poolparty 0.2.18 → 0.2.69

Sign up to get free protection for your applications and to get access to all the features.
Files changed (335) hide show
  1. data/Manifest.txt +142 -47
  2. data/PostInstall.txt +5 -5
  3. data/README.txt +25 -4
  4. data/Rakefile +46 -2
  5. data/bin/cloud +12 -21
  6. data/bin/cloud-add-keypair +18 -13
  7. data/bin/cloud-configure +11 -18
  8. data/bin/cloud-contract +7 -17
  9. data/bin/cloud-ensure-provisioning +39 -0
  10. data/bin/cloud-expand +9 -17
  11. data/bin/cloud-handle-load +27 -0
  12. data/bin/cloud-list +3 -9
  13. data/bin/cloud-maintain +12 -18
  14. data/bin/cloud-osxcopy +3 -9
  15. data/bin/cloud-provision +20 -21
  16. data/bin/cloud-refresh +4 -13
  17. data/bin/cloud-run +3 -4
  18. data/bin/cloud-ssh +3 -2
  19. data/bin/cloud-start +18 -21
  20. data/bin/cloud-stats +17 -0
  21. data/bin/cloud-terminate +4 -7
  22. data/bin/messenger-get-current-nodes +14 -0
  23. data/bin/pool +12 -12
  24. data/bin/pool-describe +0 -1
  25. data/bin/{pool-spec → pool-generate} +1 -0
  26. data/bin/pool-init +28 -0
  27. data/bin/pool-list +14 -12
  28. data/bin/pool-start +11 -17
  29. data/bin/server-build-messenger +13 -6
  30. data/bin/server-fire-cmd +1 -1
  31. data/bin/server-get-load +5 -16
  32. data/bin/server-list-active +3 -10
  33. data/bin/server-rerun +4 -4
  34. data/bin/server-send-command +18 -0
  35. data/bin/server-show-stats +17 -0
  36. data/bin/server-start-client +29 -0
  37. data/bin/server-start-master +13 -14
  38. data/bin/server-start-node +14 -15
  39. data/bin/server-stop-client +3 -0
  40. data/bin/server-stop-master +3 -0
  41. data/bin/server-stop-node +3 -0
  42. data/bin/server-update-hosts +48 -0
  43. data/config/hoe.rb +4 -3
  44. data/examples/plugin_without_plugin_directory.rb +1 -2
  45. data/generators/poolspec/USAGE +2 -2
  46. data/generators/poolspec/poolspec_generator.rb +2 -1
  47. data/generators/poolspec/templates/pool_spec_template.erb +3 -2
  48. data/lib/erlang/messenger/Rakefile +12 -0
  49. data/lib/erlang/messenger/ebin/client.app +19 -0
  50. data/lib/erlang/messenger/ebin/master.app +1 -1
  51. data/lib/erlang/messenger/ebin/node.app +1 -1
  52. data/lib/erlang/messenger/ebin/pm_client_rel-0.1.rel +1 -0
  53. data/lib/erlang/messenger/ebin/pm_master_rel-0.1.rel +1 -1
  54. data/lib/erlang/messenger/ebin/pm_node_rel-0.1.rel +1 -1
  55. data/lib/erlang/messenger/include/defines.hrl +27 -0
  56. data/lib/erlang/messenger/lib/eunit/.svn/all-wcprops +53 -0
  57. data/lib/erlang/messenger/lib/eunit/.svn/entries +140 -0
  58. data/lib/erlang/messenger/lib/eunit/.svn/format +1 -0
  59. data/lib/erlang/messenger/lib/eunit/.svn/prop-base/NOTES.svn-base +5 -0
  60. data/lib/erlang/messenger/lib/eunit/.svn/text-base/AUTHORS.svn-base +2 -0
  61. data/lib/erlang/messenger/lib/eunit/.svn/text-base/CHANGELOG.svn-base +14 -0
  62. data/lib/erlang/messenger/lib/eunit/.svn/text-base/COPYING.svn-base +504 -0
  63. data/lib/erlang/messenger/lib/eunit/.svn/text-base/NOTES.svn-base +276 -0
  64. data/lib/erlang/messenger/lib/eunit/.svn/text-base/README.svn-base +3 -0
  65. data/lib/erlang/messenger/lib/eunit/.svn/text-base/sys.config.svn-base +9 -0
  66. data/lib/erlang/messenger/lib/eunit/.svn/text-base/vsn.mk.svn-base +1 -0
  67. data/lib/erlang/messenger/lib/eunit/doc/.svn/all-wcprops +59 -0
  68. data/lib/erlang/messenger/lib/eunit/doc/.svn/entries +142 -0
  69. data/lib/erlang/messenger/lib/eunit/doc/.svn/format +1 -0
  70. data/lib/erlang/messenger/lib/eunit/doc/.svn/prop-base/erlang.png.svn-base +5 -0
  71. data/lib/erlang/messenger/lib/eunit/doc/.svn/prop-base/eunit.html.svn-base +5 -0
  72. data/lib/erlang/messenger/lib/eunit/doc/.svn/prop-base/index.html.svn-base +5 -0
  73. data/lib/erlang/messenger/lib/eunit/doc/.svn/prop-base/modules-frame.html.svn-base +5 -0
  74. data/lib/erlang/messenger/lib/eunit/doc/.svn/prop-base/overview-summary.html.svn-base +5 -0
  75. data/lib/erlang/messenger/lib/eunit/doc/.svn/prop-base/packages-frame.html.svn-base +5 -0
  76. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/edoc-info.svn-base +3 -0
  77. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/erlang.png.svn-base +0 -0
  78. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/eunit.html.svn-base +172 -0
  79. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/index.html.svn-base +17 -0
  80. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/modules-frame.html.svn-base +12 -0
  81. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/overview-summary.html.svn-base +984 -0
  82. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/overview.edoc.svn-base +980 -0
  83. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/packages-frame.html.svn-base +11 -0
  84. data/lib/erlang/messenger/lib/eunit/doc/.svn/text-base/stylesheet.css.svn-base +55 -0
  85. data/lib/erlang/messenger/lib/eunit/ebin/.svn/all-wcprops +5 -0
  86. data/lib/erlang/messenger/lib/eunit/ebin/.svn/dir-prop-base +8 -0
  87. data/lib/erlang/messenger/lib/eunit/ebin/.svn/entries +28 -0
  88. data/lib/erlang/messenger/lib/eunit/ebin/.svn/format +1 -0
  89. data/lib/erlang/messenger/lib/eunit/examples/.svn/all-wcprops +23 -0
  90. data/lib/erlang/messenger/lib/eunit/examples/.svn/entries +66 -0
  91. data/lib/erlang/messenger/lib/eunit/examples/.svn/format +1 -0
  92. data/lib/erlang/messenger/lib/eunit/examples/.svn/prop-base/eunit_examples.erl.svn-base +5 -0
  93. data/lib/erlang/messenger/lib/eunit/examples/.svn/prop-base/fib.erl.svn-base +5 -0
  94. data/lib/erlang/messenger/lib/eunit/examples/.svn/text-base/eunit_examples.erl.svn-base +339 -0
  95. data/lib/erlang/messenger/lib/eunit/examples/.svn/text-base/fib.erl.svn-base +19 -0
  96. data/lib/erlang/messenger/lib/eunit/examples/.svn/text-base/tests.txt.svn-base +1 -0
  97. data/lib/erlang/messenger/lib/eunit/include/.svn/all-wcprops +11 -0
  98. data/lib/erlang/messenger/lib/eunit/include/.svn/entries +41 -0
  99. data/lib/erlang/messenger/lib/eunit/include/.svn/format +1 -0
  100. data/lib/erlang/messenger/lib/eunit/include/.svn/prop-base/eunit.hrl.svn-base +5 -0
  101. data/lib/erlang/messenger/lib/eunit/include/.svn/text-base/eunit.hrl.svn-base +313 -0
  102. data/lib/erlang/messenger/lib/eunit/src/.svn/all-wcprops +113 -0
  103. data/lib/erlang/messenger/lib/eunit/src/.svn/entries +259 -0
  104. data/lib/erlang/messenger/lib/eunit/src/.svn/format +1 -0
  105. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/autoload.erl.svn-base +5 -0
  106. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/code_monitor.erl.svn-base +5 -0
  107. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit.erl.svn-base +5 -0
  108. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_autoexport.erl.svn-base +5 -0
  109. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_data.erl.svn-base +5 -0
  110. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_internal.hrl.svn-base +5 -0
  111. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_lib.erl.svn-base +5 -0
  112. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_proc.erl.svn-base +5 -0
  113. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_serial.erl.svn-base +5 -0
  114. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_server.erl.svn-base +5 -0
  115. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_striptests.erl.svn-base +5 -0
  116. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_test.erl.svn-base +5 -0
  117. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_tests.erl.svn-base +5 -0
  118. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/eunit_tty.erl.svn-base +5 -0
  119. data/lib/erlang/messenger/lib/eunit/src/.svn/prop-base/file_monitor.erl.svn-base +5 -0
  120. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/autoload.erl.svn-base +388 -0
  121. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/code_monitor.erl.svn-base +243 -0
  122. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit.app.src.svn-base +21 -0
  123. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit.appup.src.svn-base +1 -0
  124. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit.erl.svn-base +196 -0
  125. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_autoexport.erl.svn-base +102 -0
  126. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_data.erl.svn-base +798 -0
  127. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_internal.hrl.svn-base +48 -0
  128. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_lib.erl.svn-base +682 -0
  129. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_proc.erl.svn-base +552 -0
  130. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_serial.erl.svn-base +157 -0
  131. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_server.erl.svn-base +340 -0
  132. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_striptests.erl.svn-base +64 -0
  133. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_test.erl.svn-base +334 -0
  134. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_tests.erl.svn-base +45 -0
  135. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/eunit_tty.erl.svn-base +272 -0
  136. data/lib/erlang/messenger/lib/eunit/src/.svn/text-base/file_monitor.erl.svn-base +409 -0
  137. data/lib/erlang/messenger/pm_client_rel-0.1.boot +0 -0
  138. data/lib/erlang/messenger/pm_client_rel-0.1.script +238 -0
  139. data/lib/erlang/messenger/pm_master_rel-0.1.boot +0 -0
  140. data/lib/erlang/messenger/pm_master_rel-0.1.script +82 -85
  141. data/lib/erlang/messenger/pm_node_rel-0.1.boot +0 -0
  142. data/lib/erlang/messenger/pm_node_rel-0.1.script +80 -85
  143. data/lib/erlang/messenger/src/client_app.erl +39 -0
  144. data/lib/erlang/messenger/src/client_server.erl +60 -0
  145. data/lib/erlang/messenger/src/pm_client.erl +48 -18
  146. data/lib/erlang/messenger/src/pm_client_old.erl +52 -0
  147. data/lib/erlang/messenger/src/pm_client_supervisor.erl +38 -0
  148. data/lib/erlang/messenger/src/pm_cluster.erl +20 -16
  149. data/lib/erlang/messenger/src/pm_event_manager.erl +27 -0
  150. data/lib/erlang/messenger/src/pm_master.erl +126 -34
  151. data/lib/erlang/messenger/src/pm_master_event_handler.erl +72 -0
  152. data/lib/erlang/messenger/src/pm_master_supervisor.erl +9 -10
  153. data/lib/erlang/messenger/src/pm_node.erl +107 -30
  154. data/lib/erlang/messenger/src/pm_node_supervisor.erl +13 -15
  155. data/lib/erlang/messenger/src/pm_packager.erl +5 -2
  156. data/lib/erlang/messenger/src/pm_strings.erl +11 -0
  157. data/lib/erlang/messenger/src/utils.erl +43 -8
  158. data/lib/erlang/messenger/useful_snippets +17 -0
  159. data/lib/poolparty/aska/aska.rb +150 -0
  160. data/lib/poolparty/base_packages/haproxy.rb +19 -26
  161. data/lib/poolparty/base_packages/heartbeat.rb +48 -47
  162. data/lib/poolparty/base_packages/poolparty.rb +88 -36
  163. data/lib/poolparty/base_packages/runit.rb +21 -0
  164. data/lib/poolparty/config/postlaunchmessage.txt +5 -0
  165. data/lib/poolparty/core/array.rb +4 -1
  166. data/lib/poolparty/core/class.rb +2 -0
  167. data/lib/poolparty/core/exception.rb +2 -1
  168. data/lib/poolparty/core/hash.rb +2 -2
  169. data/lib/poolparty/core/kernel.rb +3 -3
  170. data/lib/poolparty/core/metaid.rb +15 -0
  171. data/lib/poolparty/core/object.rb +42 -2
  172. data/lib/poolparty/core/string.rb +21 -6
  173. data/lib/poolparty/core/symbol.rb +9 -0
  174. data/lib/poolparty/dependency_resolutions/puppet.rb +20 -7
  175. data/lib/poolparty/exceptions/CloudNotFoundException.rb +7 -0
  176. data/lib/poolparty/exceptions/LoadRulesException.rb +7 -0
  177. data/lib/poolparty/exceptions/MasterException.rb +10 -0
  178. data/lib/poolparty/helpers/binary.rb +37 -4
  179. data/lib/poolparty/helpers/console.rb +25 -8
  180. data/lib/poolparty/helpers/nice_printer.rb +36 -0
  181. data/lib/poolparty/helpers/optioner.rb +51 -16
  182. data/lib/poolparty/helpers/provisioner_base.rb +117 -28
  183. data/lib/poolparty/helpers/provisioners/master.rb +102 -35
  184. data/lib/poolparty/helpers/provisioners/slave.rb +25 -9
  185. data/lib/poolparty/modules/cloud_dsl.rb +13 -0
  186. data/lib/poolparty/modules/cloud_resourcer.rb +51 -8
  187. data/lib/poolparty/modules/configurable.rb +1 -1
  188. data/lib/poolparty/modules/definable_resource.rb +2 -2
  189. data/lib/poolparty/modules/file_writer.rb +37 -6
  190. data/lib/poolparty/modules/method_missing_sugar.rb +8 -3
  191. data/lib/poolparty/modules/pretty_printer.rb +1 -0
  192. data/lib/poolparty/modules/resourcing_dsl.rb +4 -2
  193. data/lib/poolparty/modules/thread_pool.rb +106 -0
  194. data/lib/poolparty/monitors/base_monitor.rb +65 -6
  195. data/lib/poolparty/monitors/monitors/cpu_monitor.rb +1 -1
  196. data/lib/poolparty/monitors/monitors/memory_monitor.rb +28 -1
  197. data/lib/poolparty/monitors/monitors/web_monitor.rb +18 -0
  198. data/lib/poolparty/net/messenger.rb +72 -0
  199. data/lib/poolparty/net/remote.rb +5 -2
  200. data/lib/poolparty/net/remote_bases/ec2/ec2_response_object.rb +66 -0
  201. data/lib/poolparty/net/remote_bases/ec2.rb +133 -131
  202. data/lib/poolparty/net/remote_instance.rb +28 -6
  203. data/lib/poolparty/net/remoter.rb +117 -64
  204. data/lib/poolparty/net/remoter_base.rb +49 -11
  205. data/lib/poolparty/plugins/deploydirectory.rb +66 -0
  206. data/lib/poolparty/plugins/line.rb +5 -4
  207. data/lib/poolparty/plugins/rsyncmirror.rb +26 -0
  208. data/lib/poolparty/plugins/runit.rb +96 -0
  209. data/lib/poolparty/plugins/svn.rb +1 -1
  210. data/lib/poolparty/pool/base.rb +47 -18
  211. data/lib/poolparty/pool/cloud.rb +121 -65
  212. data/lib/poolparty/pool/custom_resource.rb +4 -20
  213. data/lib/poolparty/pool/loggable.rb +3 -8
  214. data/lib/poolparty/pool/plugin.rb +14 -19
  215. data/lib/poolparty/pool/plugin_model.rb +8 -8
  216. data/lib/poolparty/pool/pool.rb +28 -11
  217. data/lib/poolparty/pool/resource.rb +109 -40
  218. data/lib/poolparty/pool/resources/class_package.rb +74 -33
  219. data/lib/poolparty/pool/resources/conditional.rb +32 -10
  220. data/lib/poolparty/pool/resources/custom_service.rb +30 -0
  221. data/lib/poolparty/pool/resources/exec.rb +6 -5
  222. data/lib/poolparty/pool/resources/gem_package.rb +52 -0
  223. data/lib/poolparty/pool/resources/mount.rb +22 -0
  224. data/lib/poolparty/pool/resources/package.rb +13 -3
  225. data/lib/poolparty/pool/resources/sshkey.rb +1 -1
  226. data/lib/poolparty/pool/resources/variable.rb +4 -0
  227. data/lib/poolparty/pool/script.rb +8 -6
  228. data/lib/poolparty/templates/haproxy.conf +1 -1
  229. data/lib/poolparty/templates/messenger/client/log-run.erb +2 -0
  230. data/lib/poolparty/templates/messenger/client/run.erb +4 -0
  231. data/lib/poolparty/templates/messenger/master/log-run.erb +2 -0
  232. data/lib/poolparty/templates/messenger/master/run.erb +4 -0
  233. data/lib/poolparty/templates/messenger/node/log-run.erb +2 -0
  234. data/lib/poolparty/templates/messenger/node/run.erb +4 -0
  235. data/lib/poolparty/templates/poolparty.monitor +1 -1
  236. data/lib/poolparty/templates/puppet.conf +0 -0
  237. data/lib/poolparty/templates/puppetcleaner +6 -0
  238. data/lib/poolparty/templates/puppetrerun +9 -0
  239. data/lib/poolparty/templates/puppetrunner +8 -0
  240. data/lib/poolparty/templates/yaws.conf +19 -0
  241. data/lib/poolparty/version.rb +1 -1
  242. data/lib/poolparty.rb +35 -6
  243. data/lib/poolpartycl.rb +4 -0
  244. data/log/pool.logs +0 -0
  245. data/poolparty.gemspec +574 -61
  246. data/spec/poolparty/aska/aska_spec.rb +122 -0
  247. data/spec/poolparty/bin/console_spec.rb +18 -0
  248. data/spec/poolparty/core/hash_spec.rb +4 -4
  249. data/spec/poolparty/core/object_spec.rb +29 -0
  250. data/spec/poolparty/core/string_spec.rb +6 -0
  251. data/spec/poolparty/core/time_spec.rb +0 -3
  252. data/spec/poolparty/helpers/binary_spec.rb +2 -2
  253. data/spec/poolparty/helpers/optioner_spec.rb +1 -4
  254. data/spec/poolparty/helpers/provisioner_base_spec.rb +14 -1
  255. data/spec/poolparty/helpers/provisioners/master_spec.rb +3 -3
  256. data/spec/poolparty/modules/cloud_resourcer_spec.rb +3 -3
  257. data/spec/poolparty/modules/configurable_spec.rb +4 -1
  258. data/spec/poolparty/monitors/base_monitor_spec.rb +94 -2
  259. data/spec/poolparty/monitors/monitors/cpu_monitor_spec.rb +0 -1
  260. data/spec/poolparty/monitors/monitors/memory_monitor_spec.rb +47 -0
  261. data/spec/poolparty/{helpers → net}/messenger_spec.rb +3 -1
  262. data/spec/poolparty/net/remote_bases/ec2_spec.rb +46 -5
  263. data/spec/poolparty/net/remote_instance_spec.rb +2 -1
  264. data/spec/poolparty/net/remote_spec.rb +30 -8
  265. data/spec/poolparty/net/remoter_base_spec.rb +4 -1
  266. data/spec/poolparty/net/remoter_spec.rb +11 -108
  267. data/spec/poolparty/plugins/deploydirectory_spec.rb +51 -0
  268. data/spec/poolparty/pool/base_spec.rb +43 -33
  269. data/spec/poolparty/pool/cloud_spec.rb +299 -248
  270. data/spec/poolparty/pool/custom_resource_spec.rb +10 -5
  271. data/spec/poolparty/pool/example_spec.rb +0 -84
  272. data/spec/poolparty/pool/plugin_model_spec.rb +3 -2
  273. data/spec/poolparty/pool/plugin_spec.rb +89 -62
  274. data/spec/poolparty/pool/pool_spec.rb +17 -2
  275. data/spec/poolparty/pool/resource_spec.rb +341 -236
  276. data/spec/poolparty/pool/resources/class_package_spec.rb +102 -66
  277. data/spec/poolparty/pool/resources/conditional_spec.rb +65 -26
  278. data/spec/poolparty/pool/resources/cron_spec.rb +3 -2
  279. data/spec/poolparty/pool/resources/exec_spec.rb +1 -1
  280. data/spec/poolparty/pool/resources/gem_spec.rb +18 -14
  281. data/spec/poolparty/pool/resources/remote_file_spec.rb +1 -1
  282. data/spec/poolparty/pool/resources/variable_spec.rb +1 -1
  283. data/spec/poolparty/pool/script_spec.rb +58 -37
  284. data/spec/poolparty/pool/test_plugins/webserver.rb +1 -0
  285. data/spec/poolparty/spec_helper.rb +16 -5
  286. data/tasks/deployment.rake +19 -3
  287. data/tasks/development.rake +11 -2
  288. data/tasks/ec2.rake +9 -1
  289. data/website/index.html +2 -2
  290. metadata +167 -70
  291. data/bin/pool-provision +0 -34
  292. data/erl_crash.dump +0 -8409
  293. data/examples/with_apache_plugin.rb +0 -22
  294. data/lib/erlang/messenger/Makefile +0 -15
  295. data/lib/erlang/messenger/ebin/master_app.beam +0 -0
  296. data/lib/erlang/messenger/ebin/node_app.beam +0 -0
  297. data/lib/erlang/messenger/ebin/pm_client.beam +0 -0
  298. data/lib/erlang/messenger/ebin/pm_cluster.beam +0 -0
  299. data/lib/erlang/messenger/ebin/pm_event_handler.beam +0 -0
  300. data/lib/erlang/messenger/ebin/pm_master.beam +0 -0
  301. data/lib/erlang/messenger/ebin/pm_master_supervisor.beam +0 -0
  302. data/lib/erlang/messenger/ebin/pm_node.beam +0 -0
  303. data/lib/erlang/messenger/ebin/pm_node_supervisor.beam +0 -0
  304. data/lib/erlang/messenger/ebin/pm_packager.beam +0 -0
  305. data/lib/erlang/messenger/ebin/utils.beam +0 -0
  306. data/lib/erlang/messenger/lib/eunit/Makefile +0 -28
  307. data/lib/erlang/messenger/lib/eunit/ebin/autoload.beam +0 -0
  308. data/lib/erlang/messenger/lib/eunit/ebin/code_monitor.beam +0 -0
  309. data/lib/erlang/messenger/lib/eunit/ebin/eunit.beam +0 -0
  310. data/lib/erlang/messenger/lib/eunit/ebin/eunit_autoexport.beam +0 -0
  311. data/lib/erlang/messenger/lib/eunit/ebin/eunit_data.beam +0 -0
  312. data/lib/erlang/messenger/lib/eunit/ebin/eunit_lib.beam +0 -0
  313. data/lib/erlang/messenger/lib/eunit/ebin/eunit_proc.beam +0 -0
  314. data/lib/erlang/messenger/lib/eunit/ebin/eunit_serial.beam +0 -0
  315. data/lib/erlang/messenger/lib/eunit/ebin/eunit_server.beam +0 -0
  316. data/lib/erlang/messenger/lib/eunit/ebin/eunit_striptests.beam +0 -0
  317. data/lib/erlang/messenger/lib/eunit/ebin/eunit_test.beam +0 -0
  318. data/lib/erlang/messenger/lib/eunit/ebin/eunit_tests.beam +0 -0
  319. data/lib/erlang/messenger/lib/eunit/ebin/eunit_tty.beam +0 -0
  320. data/lib/erlang/messenger/lib/eunit/ebin/file_monitor.beam +0 -0
  321. data/lib/erlang/messenger/lib/eunit/src/Makefile +0 -46
  322. data/lib/erlang/messenger/src/pm_event_handler.erl +0 -21
  323. data/lib/poolparty/config/allowed_commands.yml +0 -1
  324. data/lib/poolparty/helpers/messenger.rb +0 -29
  325. data/lib/poolparty/plugins/git.rb +0 -39
  326. data/lib/poolparty/pool/resources/gem.rb +0 -28
  327. data/lib/poolparty/pool/tmp/.ppkeys +0 -3
  328. data/lib/poolparty/pool/tmp/happydayz +0 -1
  329. data/lib/poolparty/pool/tmp/install_master.sh +0 -33
  330. data/lib/poolparty/pool/tmp/pool.spec +0 -11
  331. data/lib/poolparty/pool/tmp/poolparty.pp +0 -600
  332. data/lib/poolparty/pool/tmp/tc-instances.list +0 -1
  333. data/lib/poolparty/templates/fileserver.conf +0 -4
  334. data/spec/poolparty/plugins/git_spec.rb +0 -33
  335. data/test_manifest.pp +0 -658
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,5 @@
1
+ K 12
2
+ svn:keywords
3
+ V 31
4
+ Date Author Id Revision HeadURL
5
+ END
@@ -0,0 +1,388 @@
1
+ %% This library is free software; you can redistribute it and/or modify
2
+ %% it under the terms of the GNU Lesser General Public License as
3
+ %% published by the Free Software Foundation; either version 2 of the
4
+ %% License, or (at your option) any later version.
5
+ %%
6
+ %% This library is distributed in the hope that it will be useful, but
7
+ %% WITHOUT ANY WARRANTY; without even the implied warranty of
8
+ %% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
9
+ %% Lesser General Public License for more details.
10
+ %%
11
+ %% You should have received a copy of the GNU Lesser General Public
12
+ %% License along with this library; if not, write to the Free Software
13
+ %% Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
14
+ %% USA
15
+ %%
16
+ %% $Id:$
17
+ %%
18
+ %% @private (for now)
19
+ %% @author Richard Carlsson <richardc@it.uu.se>
20
+ %% @copyright 2006 Richard Carlsson
21
+ %% @doc Erlang automatic code loading service
22
+
23
+ -module(autoload).
24
+
25
+ -export([start/0, start/1, stop/0, stop/1,
26
+ watch_module/1, watch_module/2, watch_module/3,
27
+ watch_file/1, watch_file/2, watch_file/3,
28
+ watch_dir/1, watch_dir/2, watch_dir/3
29
+ ]).
30
+
31
+ -include_lib("kernel/include/file.hrl").
32
+
33
+ -define(SERVER, autoload).
34
+ -define(ZERO_TIMESTAMP, {0,0,0}).
35
+
36
+ %% TODO: un-watching functionality
37
+
38
+ %% We're not trying to provide strong guarantees here. The autoloader is
39
+ %% supposed to behave like a human assistant might - race conditions may
40
+ %% cause modules to be reloaded twice, or cause a reload from the wrong
41
+ %% file (if paths are suddenly changed).
42
+
43
+
44
+ watch_dir(Path) ->
45
+ watch_dir(Path, []).
46
+
47
+ watch_dir(Path, Opts) ->
48
+ watch_dir(?SERVER, Path, Opts).
49
+
50
+ watch_dir(Server, Path, Opts) ->
51
+ command(Server, {watch, {dir, filename:flatten(Path)}, Opts}).
52
+
53
+ watch_file(Path) ->
54
+ watch_file(Path, []).
55
+
56
+ watch_file(Path, Opts) ->
57
+ watch_file(?SERVER, Path, Opts).
58
+
59
+ watch_file(Server, Path, Opts) ->
60
+ command(Server, {watch, {file, filename:flatten(Path)}, Opts}).
61
+
62
+ watch_module(Module) ->
63
+ watch_module(Module, []).
64
+
65
+ watch_module(Module, Opts) ->
66
+ watch_module(?SERVER, Module, Opts).
67
+
68
+ watch_module(Server, Module, Opts) when is_atom(Module) ->
69
+ command(Server, {watch, {module, Module}, Opts}).
70
+
71
+
72
+ command(Server, Msg) ->
73
+ ServerPid = ensure_started(Server),
74
+ ServerPid ! {command, self(), Msg},
75
+ receive
76
+ {ServerPid, Result} -> Result
77
+ end.
78
+
79
+ stop() ->
80
+ stop(?SERVER).
81
+
82
+ stop(Server) ->
83
+ Server ! stop,
84
+ ok.
85
+
86
+ ensure_started(Name) when is_atom(Name) ->
87
+ start(Name);
88
+ ensure_started(Pid) when is_pid(Pid) ->
89
+ Pid.
90
+
91
+
92
+ start() ->
93
+ start(?SERVER).
94
+
95
+ start(Name) ->
96
+ case whereis(Name) of
97
+ undefined ->
98
+ Parent = self(),
99
+ Pid = spawn(fun () -> server_init(Name, Parent) end),
100
+ receive
101
+ {Pid, ok} -> Pid;
102
+ {Pid, error} -> throw(no_server)
103
+ end;
104
+ Pid -> Pid
105
+ end.
106
+
107
+ -record(state, {name, modules, files, dirs}).
108
+
109
+ -record(module, {time = ?ZERO_TIMESTAMP, file, opts = []}).
110
+
111
+ server_init(Name, Parent) ->
112
+ Self = self(),
113
+ case catch register(Name, Self) of
114
+ true ->
115
+ Parent ! {Self, ok},
116
+ code_monitor:monitor(self()),
117
+ server(#state{name = Name,
118
+ modules = dict:new(),
119
+ files = dict:new(),
120
+ dirs = dict:new()});
121
+ _ ->
122
+ init_failure(Parent)
123
+ end.
124
+
125
+ init_failure(Parent) ->
126
+ Parent ! {self(), error},
127
+ exit(failed).
128
+
129
+ server(St) ->
130
+ receive
131
+ {file_monitor, _Ref, Msg} ->
132
+ file_event(Msg, St);
133
+ {code_monitor, Msg} ->
134
+ code_event(Msg, St);
135
+ {command, From, Cmd} ->
136
+ server_command(From, Cmd, St);
137
+ stop ->
138
+ exit(normal);
139
+ Msg ->
140
+ erlang:display({autoload_unexpected, Msg}),
141
+ server(St)
142
+ end.
143
+
144
+ server_command_reply(From, Msg) ->
145
+ From ! {self(), Msg}.
146
+
147
+ server_command(From, {watch, {dir, Path}, Opts}, St) ->
148
+ server_command_reply(From, ok),
149
+ server(monitor_dir(Path, Opts, St));
150
+ server_command(From, {watch, {file, Path}, Opts}, St) ->
151
+ server_command_reply(From, ok),
152
+ server(monitor_file(Path, Opts, St));
153
+ server_command(From, {watch, {module, M}, Opts}, St) ->
154
+ server_command_reply(From, ok),
155
+ server(monitor_module(M, Opts, St)).
156
+
157
+ file_event({exists, Path, dir, _Info, Files}, St) ->
158
+ %%erlang:display({autoload_saw_exists, dir, Path}),
159
+ server(monitor_objs(Path, Files, St));
160
+ file_event({exists, Path, file, Info, _}, St) ->
161
+ %%erlang:display({autoload_saw_exists, file, Path}),
162
+ %% treat file-exists messages just like file-changed messages
163
+ server(changed_file(Path, Info#file_info.mtime, St));
164
+ file_event({changed, Path, dir, _Info, Files}, St) ->
165
+ %%erlang:display({autoload_saw_changed, dir, Path}),
166
+ server(monitor_objs(Path, Files, St));
167
+ file_event({changed, Path, file, #file_info{}=Info, _}, St) ->
168
+ %%erlang:display({autoload_saw_changed, file, Path}),
169
+ server(changed_file(Path, Info#file_info.mtime, St));
170
+ file_event({error, _Path, file, _ErrorAtom}=_Msg, St) ->
171
+ %%erlang:display({autoload_ignoring, _Msg}),
172
+ %% just consider the files as being temporarily unavailable
173
+ server(St);
174
+ file_event(_Msg, St) ->
175
+ erlang:display({autoload_unexpected_file_event, _Msg}),
176
+ server(St).
177
+
178
+ code_event({loaded, M, Time}, St) ->
179
+ %%erlang:display({autoload_saw_loaded, M, Time}),
180
+ server(loaded_module(M, Time, St));
181
+ code_event(_Msg, St) ->
182
+ erlang:display({autoload_unexpected_code_event, _Msg}),
183
+ server(St).
184
+
185
+
186
+ %% called when asked to watch a module M
187
+
188
+ monitor_module(M, Opts, St) ->
189
+ ensure_loaded(M, "", Opts),
190
+ case code:which(M) of
191
+ non_existing ->
192
+ %% no module loaded or found in path (e.g., the user has not
193
+ %% run 'make' yet, or has not updated the path): store
194
+ %% mapping from M to undefined file, for late binding
195
+ store_record(M, #module{opts = Opts}, St);
196
+ File when is_list(File) ->
197
+ %% existing file found (already loaded or in path): monitor
198
+ %% the file and map M to that file (even if the first load
199
+ %% might turn out to be from another file)
200
+ store_record(M, #module{file = File},
201
+ monitor_file(File, Opts, St));
202
+ _ ->
203
+ %% preloaded or cover-compiled module - ignore
204
+ St
205
+ end.
206
+
207
+ %% called when a monitored directory is detected as new or changed
208
+
209
+ monitor_objs(Path, Files, St) ->
210
+ Opts = get_dir_opts(Path, St),
211
+ ObjExt = file:objfile_extension(),
212
+ Objs = [F || {added, F} <- Files, filename:extension(F) == ObjExt],
213
+ lists:foldl(
214
+ fun (F, St) ->
215
+ F1 = filename:absname(filename:join(Path, F)),
216
+ %%erlang:display({autoload_monitoring, file, F1}),
217
+ monitor_file(F1, Opts, St)
218
+ end,
219
+ St,
220
+ Objs).
221
+
222
+ get_dir_opts(Path, St) ->
223
+ case dict:find(Path, St#state.dirs) of
224
+ {ok, {_Ref, Opts}} -> Opts;
225
+ error -> []
226
+ end.
227
+
228
+ %% called when a module was detected as loaded or reloaded (either
229
+ %% caused by an auto-load or by someone else)
230
+
231
+ loaded_module(M, Time, St) ->
232
+ case find_record(M, St) of
233
+ {ok, R} ->
234
+ File = code:which(M),
235
+ case R#module.file of
236
+ undefined when is_list(File) ->
237
+ %% a watched module gets a late binding to a file
238
+ %%erlang:display({autoload_late_binding, M, File}),
239
+ Opts = R#module.opts,
240
+ update_record(M, R, Time, File,
241
+ monitor_file(File, Opts, St));
242
+ File when is_list(File) ->
243
+ %% loaded from watched file - update the load time
244
+ update_record(M, R, Time, File, St);
245
+ _ ->
246
+ St %% not loaded from the watched file
247
+ end;
248
+ error ->
249
+ St %% uninteresting module
250
+ end.
251
+
252
+
253
+ %% called whenever a file has changed and exists
254
+
255
+ changed_file(File, Time, St) ->
256
+ %% always reread the module name from the file
257
+ case obj_module(File) of
258
+ {module, M} ->
259
+ case find_record(M, St) of
260
+ {ok, R} ->
261
+ check_reload(M, File, Time, R),
262
+ St;
263
+ error ->
264
+ %% first time we see this module: map to file
265
+ %% and don't try to reload
266
+ R = #module{file = File},
267
+ store_record(M, R, St)
268
+ end;
269
+ error ->
270
+ St %% bad file - ignore until changed again
271
+ end.
272
+
273
+
274
+ %% check if a module needs to be reloaded, when its file has changed
275
+
276
+ check_reload(M, File, Time, R) ->
277
+ %%erlang:display({autoload_checking, M, File, Time, R}),
278
+ case ((R#module.file == File)
279
+ andalso is_loaded(M)
280
+ andalso (code:which(M) == File)
281
+ andalso is_newer(Time, R#module.time))
282
+ of
283
+ true ->
284
+ reload(M);
285
+ false ->
286
+ ok
287
+ end.
288
+
289
+
290
+ reload(M) ->
291
+ erlang:display({autoload_loading, M}),
292
+ code:purge(M),
293
+ code:load_file(M).
294
+
295
+ update_record(M, R, Time, File, St) ->
296
+ store_record(M, R#module{time = Time, file = File}, St).
297
+
298
+ find_record(M, St) ->
299
+ dict:find(M, St#state.modules).
300
+
301
+ store_record(M, R, St) ->
302
+ St#state{modules = dict:store(M, R, St#state.modules)}.
303
+
304
+ %% we must remember watched files/dirs, so we don't set up more than one
305
+ %% file monitor for the same path, and so we can cancel monitors; for
306
+ %% dirs, we also remember options
307
+
308
+ monitor_file(Path, Opts, St) ->
309
+ ensure_loaded([], Path, Opts),
310
+ case dict:is_key(Path, St#state.files) of
311
+ true ->
312
+ %%erlang:display({autoload_already_watching, Path}),
313
+ St;
314
+ false ->
315
+ {ok, _, Ref} = file_monitor:monitor_file(Path, self()),
316
+ St#state{files = dict:store(Path, Ref, St#state.files)}
317
+ end.
318
+
319
+ monitor_dir(Path, Opts, St) ->
320
+ case dict:is_key(Path, St#state.dirs) of
321
+ true ->
322
+ %%erlang:display({autoload_already_watching, Path}),
323
+ St;
324
+ false ->
325
+ {ok, _, Ref} = file_monitor:monitor_dir(Path, self()),
326
+ St#state{dirs = dict:store(Path, {Ref, Opts},
327
+ St#state.dirs)}
328
+ end.
329
+
330
+ %% try to ensure that a module/file is loaded (M is [] if unknown;
331
+ %% File is "" if unknown)
332
+
333
+ ensure_loaded(M, File, Opts) ->
334
+ case proplists:get_bool(load, Opts) of
335
+ true when is_atom(M) ->
336
+ ensure_loaded_1(M, File); %% known module name
337
+ true when M == [] ->
338
+ case obj_module(File) of
339
+ {module, Module} ->
340
+ ensure_loaded_1(Module, File);
341
+ error ->
342
+ ok %% couldn't load
343
+ end;
344
+ false -> ok
345
+ end.
346
+
347
+ ensure_loaded_1(M, File) ->
348
+ %% always try to load using the path first
349
+ case code:ensure_loaded(M) of
350
+ {module, _} ->
351
+ ok;
352
+ {error, _} when File == [] ->
353
+ ok; %% ignore error
354
+ {error, _} ->
355
+ %% try loading directly from the file
356
+ code:load_abs(File) %% ignore result
357
+ end.
358
+
359
+ %% find name of module stored in an object file
360
+
361
+ obj_module(File) ->
362
+ case beam_lib:info(File) of
363
+ List when is_list(List) ->
364
+ case lists:keysearch(module, 1, List) of
365
+ {value, {module, M}} ->
366
+ {module, M};
367
+ _ ->
368
+ error
369
+ end;
370
+ _ ->
371
+ error
372
+ end.
373
+
374
+ %% check if the file timestamp (on DateTime format) is newer than
375
+ %% the module load-time (on Now-format)
376
+
377
+ is_newer(DateTime, LoadTime) ->
378
+ calendar:datetime_to_gregorian_seconds(DateTime) >
379
+ calendar:datetime_to_gregorian_seconds(
380
+ calendar:now_to_local_time(LoadTime)).
381
+
382
+ %% wrapper for code:is_loaded/1
383
+
384
+ is_loaded(Module) ->
385
+ case code:is_loaded(Module) of
386
+ {file, _} -> true;
387
+ false -> false
388
+ end.
@@ -0,0 +1,243 @@
1
+ %% This library is free software; you can redistribute it and/or modify
2
+ %% it under the terms of the GNU Lesser General Public License as
3
+ %% published by the Free Software Foundation; either version 2 of the
4
+ %% License, or (at your option) any later version.
5
+ %%
6
+ %% This library is distributed in the hope that it will be useful, but
7
+ %% WITHOUT ANY WARRANTY; without even the implied warranty of
8
+ %% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
9
+ %% Lesser General Public License for more details.
10
+ %%
11
+ %% You should have received a copy of the GNU Lesser General Public
12
+ %% License along with this library; if not, write to the Free Software
13
+ %% Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
14
+ %% USA
15
+ %%
16
+ %% $Id:$
17
+ %%
18
+ %% @private (for now)
19
+ %% @author Richard Carlsson <richardc@it.uu.se>
20
+ %% @copyright 2006 Richard Carlsson
21
+ %% @doc Erlang code monitoring service
22
+
23
+ -module(code_monitor).
24
+
25
+ -export([start/0, start/1, stop/0, stop/1, monitor/1, monitor/2,
26
+ demonitor/1, demonitor/2, install_codespy/1, wiretap/3]).
27
+
28
+ -export([main/1]). %% private
29
+
30
+
31
+ -define(SERVER, code_monitor).
32
+
33
+ monitor(Pid) ->
34
+ monitor(?SERVER, Pid).
35
+
36
+ monitor(Server, Pid) when is_pid(Pid) ->
37
+ ensure_started(Server),
38
+ Server ! {monitor, Pid},
39
+ ok.
40
+
41
+ demonitor(Pid) ->
42
+ demonitor(?SERVER, Pid).
43
+
44
+ demonitor(Server, Pid) when is_pid(Pid) ->
45
+ ensure_started(Server),
46
+ Server ! {demonitor, Pid},
47
+ ok.
48
+
49
+ stop() ->
50
+ stop(?SERVER).
51
+
52
+ stop(Server) ->
53
+ Server ! stop,
54
+ ok.
55
+
56
+ ensure_started(Name) when is_atom(Name) ->
57
+ start(Name);
58
+ ensure_started(Pid) when is_pid(Pid) ->
59
+ Pid.
60
+
61
+ start() ->
62
+ start(?SERVER).
63
+
64
+ start(Name) ->
65
+ case whereis(Name) of
66
+ undefined ->
67
+ Parent = self(),
68
+ Pid = spawn(fun () -> server_init(Name, Parent) end),
69
+ receive
70
+ {Pid, ok} -> Pid;
71
+ {Pid, error} -> throw(no_server)
72
+ end;
73
+ Pid -> Pid
74
+ end.
75
+
76
+ server_init(undefined = Name, Parent) ->
77
+ %% anonymous server
78
+ server_init_1(Name, Parent);
79
+ server_init(Name, Parent) ->
80
+ case catch register(Name, self()) of
81
+ true ->
82
+ server_init_1(Name, Parent);
83
+ _ ->
84
+ init_failure(Parent)
85
+ end.
86
+
87
+ server_init_1(Name, Parent) ->
88
+ case install_codespy(self()) of
89
+ {ok, _Spy} ->
90
+ Parent ! {self(), ok},
91
+ server(Name, sets:new());
92
+ {error, _} ->
93
+ init_failure(Parent)
94
+ end.
95
+
96
+ init_failure(Parent) ->
97
+ Parent ! {self(), error},
98
+ exit(failed).
99
+
100
+ server(Name, Listeners) ->
101
+ ?MODULE:main({Name, Listeners}).
102
+
103
+ %% @private
104
+ main({Name, Listeners}) ->
105
+ receive
106
+ {code_server, {module, M}} ->
107
+ cast({loaded, M, erlang:now()}, Listeners),
108
+ server(Name, Listeners);
109
+ {monitor, Pid} when is_pid(Pid) ->
110
+ server(Name, sets:add_element(Pid, Listeners));
111
+ {demonitor, Pid} ->
112
+ server(Name, sets:del_element(Pid, Listeners));
113
+ stop ->
114
+ exit(normal);
115
+ _ ->
116
+ server(Name, Listeners)
117
+ end.
118
+
119
+ cast(M, Listeners) ->
120
+ sets:fold(fun (L, M) -> L ! M end, {code_monitor, M}, Listeners).
121
+
122
+
123
+ %% code server spy process using generic wiretap functionality
124
+
125
+ install_codespy(To) ->
126
+ wiretap(code_server, To, fun code_spy/3).
127
+
128
+ code_spy({code_call,From,{load_file,_}=Req}, Server, To) ->
129
+ handle_load(Req, From, Req, Server, To);
130
+ code_spy({code_call,From,{ensure_loaded,M}=Req}=Msg, Server, To) ->
131
+ case erlang:module_loaded(M) of
132
+ true -> Server ! Msg;
133
+ false -> handle_load(Req, From, Req, Server, To)
134
+ end;
135
+ code_spy({code_call,From,{load_abs,_,_}=Req}, Server, To) ->
136
+ handle_load(Req, From, Req, Server, To);
137
+ code_spy({code_call,From,{load_binary,_,_,_}=Req}, Server, To) ->
138
+ handle_load(Req, From, Req, Server, To);
139
+ code_spy({code_call,From,{load_native_partial,_,_}=Req}, Server, To) ->
140
+ handle_load(Req, From, Req, Server, To);
141
+ code_spy({code_call,From,{load_native_sticky,_,_,_}=Req}, Server, To) ->
142
+ handle_load(Req, From, Req, Server, To);
143
+ code_spy(Msg, Server, _To) ->
144
+ Server ! Msg.
145
+
146
+ handle_load(Req, From, Req, Server, To) ->
147
+ ReplyTo = spawn(fun () -> reply_handler(Server, From, To) end),
148
+ Server ! {code_call, ReplyTo, Req}.
149
+
150
+ %% one-shot processes - receive, pass on and die
151
+ reply_handler(Server, Client, To) ->
152
+ link(Server),
153
+ receive
154
+ {code_server, _Reply} = M ->
155
+ To ! Client ! M
156
+ end.
157
+
158
+ %% basic wiretapping of registered processes (it should be possible to
159
+ %% have several wiretaps attached to the same registered name; they will
160
+ %% transparently form a chain, without knowing about each other)
161
+
162
+ wiretap(Name, To, F) when is_atom(Name), is_pid(To), is_function(F) ->
163
+ Parent = self(),
164
+ Pid = spawn(fun () -> wiretap_init(Name, To, F, Parent) end),
165
+ receive
166
+ {Pid, Result} -> Result
167
+ end.
168
+
169
+ wiretap_init(Name, To, F, Parent) ->
170
+ case whereis(Name) of
171
+ undefined ->
172
+ Parent ! {self(), {error, undefined}},
173
+ exit(error);
174
+ Pid ->
175
+ catch unregister(Name),
176
+ catch register(Name, self()),
177
+ Self = self(),
178
+ case whereis(Name) of
179
+ Self ->
180
+ process_flag(trap_exit, true),
181
+ link(Pid),
182
+ link(To),
183
+ Parent ! {self(), {ok, self()}},
184
+ wiretap_loop(Name, To, Pid, F);
185
+ _ ->
186
+ Parent ! {self(), {error, register_failed}},
187
+ exit(error)
188
+ end
189
+ end.
190
+
191
+ wiretap_loop(Name, To, Pid, F) ->
192
+ receive
193
+ {'EXIT', Pid, _} ->
194
+ wiretap_dropped(Name, To, F);
195
+ {'EXIT', To, _} ->
196
+ wiretap_exit(Name, Pid);
197
+ Msg ->
198
+ F(Msg, Pid, To),
199
+ wiretap_loop(Name, To, Pid, F)
200
+ end.
201
+
202
+ %% note that the registered name might get stolen from the spy process,
203
+ %% e.g., by another active wiretap
204
+
205
+ wiretap_exit(Name, Pid) ->
206
+ %% the receiver died - restore things and go away invisibly
207
+ unlink(Pid),
208
+ Self = self(),
209
+ %% sadly, this is not atomic...
210
+ case whereis(Name) of
211
+ Self ->
212
+ catch unregister(Name),
213
+ catch register(Name, Pid);
214
+ _ -> ok
215
+ end,
216
+ exit(normal).
217
+
218
+ %% if the real server goes away, make sure to unregister, and keep watch
219
+ %% in order to restart the wiretap when the server comes up again
220
+
221
+ wiretap_dropped(Name, To, F) ->
222
+ Self = self(),
223
+ case whereis(Name) of
224
+ Self -> (catch unregister(Name));
225
+ _ -> ok
226
+ end,
227
+ wiretap_watch(Name, To, F).
228
+
229
+ wiretap_watch(Name, To, F) ->
230
+ receive
231
+ {'EXIT', To, _} ->
232
+ exit(normal)
233
+ after 1000 ->
234
+ case whereis(Name) of
235
+ Pid when is_pid(Pid) ->
236
+ %% this process will terminate after starting the
237
+ %% new wiretap (even it that call fails)
238
+ wiretap(Name, To, F),
239
+ exit(normal);
240
+ _ ->
241
+ wiretap_watch(Name, To, F)
242
+ end
243
+ end.