passenger 2.2.11 → 2.2.12

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of passenger might be problematic. Click here for more details.

Files changed (287) hide show
  1. data/NEWS +29 -0
  2. data/bin/passenger-install-nginx-module +1 -1
  3. data/doc/Users guide Apache.html +38 -7
  4. data/doc/Users guide Nginx.html +36 -5
  5. data/doc/cxxapi/AbstractSpawnManager_8h-source.html +1 -1
  6. data/doc/cxxapi/Account_8h-source.html +1 -1
  7. data/doc/cxxapi/AccountsDatabase_8h-source.html +1 -1
  8. data/doc/cxxapi/AgentsStarter_8h-source.html +1 -1
  9. data/doc/cxxapi/BCrypt_8h-source.html +1 -1
  10. data/doc/cxxapi/Blowfish_8h-source.html +1 -1
  11. data/doc/cxxapi/Bucket_8h-source.html +5 -4
  12. data/doc/cxxapi/Constants_8h-source.html +2 -2
  13. data/doc/cxxapi/ContentHandler_8h-source.html +1 -1
  14. data/doc/cxxapi/DirectoryMapper_8h-source.html +10 -9
  15. data/doc/cxxapi/EventedServer_8h-source.html +415 -0
  16. data/doc/cxxapi/Exceptions_8h-source.html +1 -1
  17. data/doc/cxxapi/FileDescriptor_8h-source.html +1 -1
  18. data/doc/cxxapi/Hooks_8h-source.html +3 -2
  19. data/doc/cxxapi/HttpStatusExtractor_8h-source.html +324 -323
  20. data/doc/cxxapi/Logging_8h-source.html +643 -546
  21. data/doc/cxxapi/MessageChannel_8h-source.html +429 -404
  22. data/doc/cxxapi/MessageClient_8h-source.html +1 -1
  23. data/doc/cxxapi/MessageReadersWriters_8h-source.html +451 -0
  24. data/doc/cxxapi/MessageServer_8h-source.html +1 -1
  25. data/doc/cxxapi/PoolOptions_8h-source.html +287 -279
  26. data/doc/cxxapi/Process_8h-source.html +178 -167
  27. data/doc/cxxapi/RandomGenerator_8h-source.html +45 -18
  28. data/doc/cxxapi/ResourceLocator_8h-source.html +3 -3
  29. data/doc/cxxapi/ScgiRequestParser_8h-source.html +344 -343
  30. data/doc/cxxapi/ServerInstanceDir_8h-source.html +1 -1
  31. data/doc/cxxapi/Session_8h-source.html +387 -374
  32. data/doc/cxxapi/SpawnManager_8h-source.html +547 -538
  33. data/doc/cxxapi/StaticContentHandler_8h-source.html +1 -1
  34. data/doc/cxxapi/StaticString_8h-source.html +1 -1
  35. data/doc/cxxapi/StringListCreator_8h-source.html +1 -1
  36. data/doc/cxxapi/Utils_8h-source.html +545 -520
  37. data/doc/cxxapi/annotated.html +11 -8
  38. data/doc/cxxapi/apache2_2Configuration_8h-source.html +1 -1
  39. data/doc/cxxapi/classAgentWatcher-members.html +1 -1
  40. data/doc/cxxapi/classAgentWatcher.html +1 -1
  41. data/doc/cxxapi/classClient-members.html +1 -1
  42. data/doc/cxxapi/classClient.html +1 -1
  43. data/doc/cxxapi/classHooks-members.html +16 -13
  44. data/doc/cxxapi/classHooks.html +69 -73
  45. data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager-members.html +1 -1
  46. data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager.html +1 -1
  47. data/doc/cxxapi/classPassenger_1_1ArgumentException-members.html +1 -1
  48. data/doc/cxxapi/classPassenger_1_1ArgumentException.html +1 -1
  49. data/doc/cxxapi/classPassenger_1_1ArrayReader-members.html +33 -0
  50. data/doc/cxxapi/classPassenger_1_1ArrayReader.html +47 -0
  51. data/doc/cxxapi/classPassenger_1_1BufferedUpload-members.html +1 -1
  52. data/doc/cxxapi/classPassenger_1_1BufferedUpload.html +1 -1
  53. data/doc/cxxapi/classPassenger_1_1BusyException-members.html +1 -1
  54. data/doc/cxxapi/classPassenger_1_1BusyException.html +1 -1
  55. data/doc/cxxapi/classPassenger_1_1ConfigurationException-members.html +1 -1
  56. data/doc/cxxapi/classPassenger_1_1ConfigurationException.html +1 -1
  57. data/doc/cxxapi/classPassenger_1_1DirectoryMapper-members.html +18 -15
  58. data/doc/cxxapi/classPassenger_1_1DirectoryMapper.html +89 -99
  59. data/doc/cxxapi/classPassenger_1_1EOFException-members.html +1 -1
  60. data/doc/cxxapi/classPassenger_1_1EOFException.html +1 -1
  61. data/doc/cxxapi/classPassenger_1_1EventFd-members.html +1 -1
  62. data/doc/cxxapi/classPassenger_1_1EventFd.html +1 -1
  63. data/doc/cxxapi/classPassenger_1_1FileDescriptor-members.html +1 -1
  64. data/doc/cxxapi/classPassenger_1_1FileDescriptor.html +1 -1
  65. data/doc/cxxapi/classPassenger_1_1FileNotFoundException-members.html +1 -1
  66. data/doc/cxxapi/classPassenger_1_1FileNotFoundException.html +1 -1
  67. data/doc/cxxapi/classPassenger_1_1FileSystemException-members.html +1 -1
  68. data/doc/cxxapi/classPassenger_1_1FileSystemException.html +1 -1
  69. data/doc/cxxapi/classPassenger_1_1HttpStatusExtractor-members.html +1 -1
  70. data/doc/cxxapi/classPassenger_1_1HttpStatusExtractor.html +2 -2
  71. data/doc/cxxapi/classPassenger_1_1IOException-members.html +1 -1
  72. data/doc/cxxapi/classPassenger_1_1IOException.html +1 -1
  73. data/doc/cxxapi/classPassenger_1_1MemZeroGuard-members.html +1 -1
  74. data/doc/cxxapi/classPassenger_1_1MemZeroGuard.html +1 -1
  75. data/doc/cxxapi/classPassenger_1_1MessageChannel-members.html +1 -1
  76. data/doc/cxxapi/classPassenger_1_1MessageChannel.html +1 -1
  77. data/doc/cxxapi/classPassenger_1_1MessageServer-members.html +1 -1
  78. data/doc/cxxapi/classPassenger_1_1MessageServer.html +1 -1
  79. data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext-members.html +1 -1
  80. data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext.html +1 -1
  81. data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext-members.html +1 -1
  82. data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext.html +1 -1
  83. data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler-members.html +1 -1
  84. data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler.html +1 -1
  85. data/doc/cxxapi/classPassenger_1_1Process-members.html +4 -3
  86. data/doc/cxxapi/classPassenger_1_1Process.html +40 -9
  87. data/doc/cxxapi/classPassenger_1_1RandomGenerator-members.html +2 -1
  88. data/doc/cxxapi/classPassenger_1_1RandomGenerator.html +29 -2
  89. data/doc/cxxapi/classPassenger_1_1RuntimeException-members.html +1 -1
  90. data/doc/cxxapi/classPassenger_1_1RuntimeException.html +1 -1
  91. data/doc/cxxapi/classPassenger_1_1ScalarReader-members.html +33 -0
  92. data/doc/cxxapi/classPassenger_1_1ScalarReader.html +47 -0
  93. data/doc/cxxapi/classPassenger_1_1ScgiRequestParser-members.html +1 -1
  94. data/doc/cxxapi/classPassenger_1_1ScgiRequestParser.html +1 -1
  95. data/doc/cxxapi/classPassenger_1_1SecurityException-members.html +1 -1
  96. data/doc/cxxapi/classPassenger_1_1SecurityException.html +1 -1
  97. data/doc/cxxapi/classPassenger_1_1Session-members.html +1 -1
  98. data/doc/cxxapi/classPassenger_1_1Session.html +2 -2
  99. data/doc/cxxapi/classPassenger_1_1SpawnException-members.html +1 -1
  100. data/doc/cxxapi/classPassenger_1_1SpawnException.html +1 -1
  101. data/doc/cxxapi/classPassenger_1_1SpawnManager-members.html +2 -2
  102. data/doc/cxxapi/classPassenger_1_1SpawnManager.html +6 -18
  103. data/doc/cxxapi/classPassenger_1_1StandardSession-members.html +1 -1
  104. data/doc/cxxapi/classPassenger_1_1StandardSession.html +1 -1
  105. data/doc/cxxapi/classPassenger_1_1StaticString-members.html +1 -1
  106. data/doc/cxxapi/classPassenger_1_1StaticString.html +1 -1
  107. data/doc/cxxapi/classPassenger_1_1SystemException-members.html +1 -1
  108. data/doc/cxxapi/classPassenger_1_1SystemException.html +1 -1
  109. data/doc/cxxapi/classPassenger_1_1TimeRetrievalException-members.html +1 -1
  110. data/doc/cxxapi/classPassenger_1_1TimeRetrievalException.html +1 -1
  111. data/doc/cxxapi/classPassenger_1_1TimeoutException-members.html +1 -1
  112. data/doc/cxxapi/classPassenger_1_1TimeoutException.html +1 -1
  113. data/doc/cxxapi/classPassenger_1_1Uint16Reader-members.html +33 -0
  114. data/doc/cxxapi/classPassenger_1_1Uint16Reader.html +47 -0
  115. data/doc/cxxapi/classPassenger_1_1Uint32Reader-members.html +33 -0
  116. data/doc/cxxapi/classPassenger_1_1Uint32Reader.html +47 -0
  117. data/doc/cxxapi/classServer-members.html +1 -1
  118. data/doc/cxxapi/classServer.html +1 -1
  119. data/doc/cxxapi/classServerInstanceDirToucher-members.html +1 -1
  120. data/doc/cxxapi/classServerInstanceDirToucher.html +1 -1
  121. data/doc/cxxapi/classes.html +13 -10
  122. data/doc/cxxapi/definitions_8h-source.html +2 -1
  123. data/doc/cxxapi/doxygen.css +91 -0
  124. data/doc/cxxapi/files.html +16 -13
  125. data/doc/cxxapi/functions.html +32 -21
  126. data/doc/cxxapi/functions_enum.html +1 -1
  127. data/doc/cxxapi/functions_eval.html +1 -1
  128. data/doc/cxxapi/functions_func.html +32 -21
  129. data/doc/cxxapi/functions_vars.html +3 -1
  130. data/doc/cxxapi/graph_legend.html +20 -15
  131. data/doc/cxxapi/graph_legend.png +0 -0
  132. data/doc/cxxapi/group__Configuration.html +51 -64
  133. data/doc/cxxapi/group__Configuration.png +0 -0
  134. data/doc/cxxapi/group__Core.html +24 -26
  135. data/doc/cxxapi/group__Core.png +0 -0
  136. data/doc/cxxapi/group__Exceptions.html +1 -1
  137. data/doc/cxxapi/group__Hooks.html +17 -15
  138. data/doc/cxxapi/group__Hooks.png +0 -0
  139. data/doc/cxxapi/group__Support.html +11 -9
  140. data/doc/cxxapi/hierarchy.html +6 -1
  141. data/doc/cxxapi/index.html +10 -5
  142. data/doc/cxxapi/inherit__graph__10.map +1 -1
  143. data/doc/cxxapi/inherit__graph__10.md5 +1 -1
  144. data/doc/cxxapi/inherit__graph__10.png +0 -0
  145. data/doc/cxxapi/inherit__graph__11.map +1 -1
  146. data/doc/cxxapi/inherit__graph__11.md5 +1 -1
  147. data/doc/cxxapi/inherit__graph__11.png +0 -0
  148. data/doc/cxxapi/inherit__graph__12.map +1 -1
  149. data/doc/cxxapi/inherit__graph__12.md5 +1 -1
  150. data/doc/cxxapi/inherit__graph__12.png +0 -0
  151. data/doc/cxxapi/inherit__graph__13.map +1 -1
  152. data/doc/cxxapi/inherit__graph__13.md5 +1 -1
  153. data/doc/cxxapi/inherit__graph__13.png +0 -0
  154. data/doc/cxxapi/inherit__graph__14.map +1 -3
  155. data/doc/cxxapi/inherit__graph__14.md5 +1 -1
  156. data/doc/cxxapi/inherit__graph__14.png +0 -0
  157. data/doc/cxxapi/inherit__graph__15.map +1 -1
  158. data/doc/cxxapi/inherit__graph__15.md5 +1 -1
  159. data/doc/cxxapi/inherit__graph__15.png +0 -0
  160. data/doc/cxxapi/inherit__graph__16.map +3 -1
  161. data/doc/cxxapi/inherit__graph__16.md5 +1 -1
  162. data/doc/cxxapi/inherit__graph__16.png +0 -0
  163. data/doc/cxxapi/inherit__graph__17.map +1 -1
  164. data/doc/cxxapi/inherit__graph__17.md5 +1 -1
  165. data/doc/cxxapi/inherit__graph__17.png +0 -0
  166. data/doc/cxxapi/inherit__graph__18.map +1 -2
  167. data/doc/cxxapi/inherit__graph__18.md5 +1 -1
  168. data/doc/cxxapi/inherit__graph__18.png +0 -0
  169. data/doc/cxxapi/inherit__graph__19.map +1 -1
  170. data/doc/cxxapi/inherit__graph__19.md5 +1 -1
  171. data/doc/cxxapi/inherit__graph__19.png +0 -0
  172. data/doc/cxxapi/inherit__graph__20.map +2 -1
  173. data/doc/cxxapi/inherit__graph__20.md5 +1 -1
  174. data/doc/cxxapi/inherit__graph__20.png +0 -0
  175. data/doc/cxxapi/inherit__graph__21.map +1 -1
  176. data/doc/cxxapi/inherit__graph__21.md5 +1 -1
  177. data/doc/cxxapi/inherit__graph__21.png +0 -0
  178. data/doc/cxxapi/inherit__graph__22.map +1 -1
  179. data/doc/cxxapi/inherit__graph__22.md5 +1 -1
  180. data/doc/cxxapi/inherit__graph__22.png +0 -0
  181. data/doc/cxxapi/inherit__graph__23.map +1 -1
  182. data/doc/cxxapi/inherit__graph__23.md5 +1 -1
  183. data/doc/cxxapi/inherit__graph__23.png +0 -0
  184. data/doc/cxxapi/inherit__graph__24.map +1 -1
  185. data/doc/cxxapi/inherit__graph__24.md5 +1 -1
  186. data/doc/cxxapi/inherit__graph__24.png +0 -0
  187. data/doc/cxxapi/inherit__graph__25.map +1 -1
  188. data/doc/cxxapi/inherit__graph__25.md5 +1 -1
  189. data/doc/cxxapi/inherit__graph__25.png +0 -0
  190. data/doc/cxxapi/inherit__graph__26.map +1 -1
  191. data/doc/cxxapi/inherit__graph__26.md5 +1 -1
  192. data/doc/cxxapi/inherit__graph__26.png +0 -0
  193. data/doc/cxxapi/inherit__graph__27.map +1 -2
  194. data/doc/cxxapi/inherit__graph__27.md5 +1 -1
  195. data/doc/cxxapi/inherit__graph__27.png +0 -0
  196. data/doc/cxxapi/inherit__graph__28.map +1 -1
  197. data/doc/cxxapi/inherit__graph__28.md5 +1 -1
  198. data/doc/cxxapi/inherit__graph__28.png +0 -0
  199. data/doc/cxxapi/inherit__graph__29.map +1 -1
  200. data/doc/cxxapi/inherit__graph__29.md5 +1 -1
  201. data/doc/cxxapi/inherit__graph__29.png +0 -0
  202. data/doc/cxxapi/inherit__graph__30.map +2 -1
  203. data/doc/cxxapi/inherit__graph__30.md5 +1 -1
  204. data/doc/cxxapi/inherit__graph__30.png +0 -0
  205. data/doc/cxxapi/inherit__graph__31.map +1 -3
  206. data/doc/cxxapi/inherit__graph__31.md5 +1 -1
  207. data/doc/cxxapi/inherit__graph__31.png +0 -0
  208. data/doc/cxxapi/inherit__graph__32.map +1 -1
  209. data/doc/cxxapi/inherit__graph__32.md5 +1 -1
  210. data/doc/cxxapi/inherit__graph__32.png +0 -0
  211. data/doc/cxxapi/inherit__graph__33.map +1 -1
  212. data/doc/cxxapi/inherit__graph__33.md5 +1 -1
  213. data/doc/cxxapi/inherit__graph__33.png +0 -0
  214. data/doc/cxxapi/inherit__graph__34.map +3 -1
  215. data/doc/cxxapi/inherit__graph__34.md5 +1 -1
  216. data/doc/cxxapi/inherit__graph__34.png +0 -0
  217. data/doc/cxxapi/inherit__graph__35.map +1 -1
  218. data/doc/cxxapi/inherit__graph__35.md5 +1 -1
  219. data/doc/cxxapi/inherit__graph__35.png +0 -0
  220. data/doc/cxxapi/inherit__graph__36.map +1 -1
  221. data/doc/cxxapi/inherit__graph__36.md5 +1 -1
  222. data/doc/cxxapi/inherit__graph__36.png +0 -0
  223. data/doc/cxxapi/inherit__graph__37.map +1 -0
  224. data/doc/cxxapi/inherit__graph__37.md5 +1 -0
  225. data/doc/cxxapi/inherit__graph__37.png +0 -0
  226. data/doc/cxxapi/inherit__graph__38.map +1 -0
  227. data/doc/cxxapi/inherit__graph__38.md5 +1 -0
  228. data/doc/cxxapi/inherit__graph__38.png +0 -0
  229. data/doc/cxxapi/inherit__graph__39.map +1 -0
  230. data/doc/cxxapi/inherit__graph__39.md5 +1 -0
  231. data/doc/cxxapi/inherit__graph__39.png +0 -0
  232. data/doc/cxxapi/inherit__graph__5.map +1 -1
  233. data/doc/cxxapi/inherit__graph__5.md5 +1 -1
  234. data/doc/cxxapi/inherit__graph__5.png +0 -0
  235. data/doc/cxxapi/inherit__graph__6.map +1 -1
  236. data/doc/cxxapi/inherit__graph__6.md5 +1 -1
  237. data/doc/cxxapi/inherit__graph__6.png +0 -0
  238. data/doc/cxxapi/inherit__graph__7.map +1 -1
  239. data/doc/cxxapi/inherit__graph__7.md5 +1 -1
  240. data/doc/cxxapi/inherit__graph__7.png +0 -0
  241. data/doc/cxxapi/inherit__graph__8.map +1 -1
  242. data/doc/cxxapi/inherit__graph__8.md5 +1 -1
  243. data/doc/cxxapi/inherit__graph__8.png +0 -0
  244. data/doc/cxxapi/inherit__graph__9.map +1 -1
  245. data/doc/cxxapi/inherit__graph__9.md5 +1 -1
  246. data/doc/cxxapi/inherit__graph__9.png +0 -0
  247. data/doc/cxxapi/inherits.html +46 -31
  248. data/doc/cxxapi/main.html +12 -11
  249. data/doc/cxxapi/modules.html +16 -13
  250. data/doc/cxxapi/namespacePassenger.html +82 -5
  251. data/doc/cxxapi/namespacemembers.html +5 -3
  252. data/doc/cxxapi/namespacemembers_enum.html +1 -1
  253. data/doc/cxxapi/namespacemembers_eval.html +1 -1
  254. data/doc/cxxapi/namespacemembers_func.html +5 -3
  255. data/doc/cxxapi/namespacemembers_type.html +1 -1
  256. data/doc/cxxapi/namespaces.html +2 -2
  257. data/doc/cxxapi/nginx_2Configuration_8h-source.html +43 -42
  258. data/doc/cxxapi/ngx__http__passenger__module_8h-source.html +1 -1
  259. data/doc/cxxapi/structPassenger_1_1AnythingToString-members.html +1 -1
  260. data/doc/cxxapi/structPassenger_1_1AnythingToString.html +1 -1
  261. data/doc/cxxapi/structPassenger_1_1AnythingToString_3_01vector_3_01StaticString_01_4_01_4-members.html +33 -0
  262. data/doc/cxxapi/structPassenger_1_1AnythingToString_3_01vector_3_01StaticString_01_4_01_4.html +52 -0
  263. data/doc/cxxapi/structPassenger_1_1AnythingToString_3_01vector_3_01string_01_4_01_4-members.html +1 -1
  264. data/doc/cxxapi/structPassenger_1_1AnythingToString_3_01vector_3_01string_01_4_01_4.html +1 -1
  265. data/doc/cxxapi/structPassenger_1_1MessageServer_1_1DisconnectEventBroadcastGuard-members.html +1 -1
  266. data/doc/cxxapi/structPassenger_1_1MessageServer_1_1DisconnectEventBroadcastGuard.html +1 -1
  267. data/doc/cxxapi/structPassenger_1_1PoolOptions-members.html +3 -2
  268. data/doc/cxxapi/structPassenger_1_1PoolOptions.html +25 -5
  269. data/doc/cxxapi/structPassenger_1_1StaticString_1_1Hash-members.html +1 -1
  270. data/doc/cxxapi/structPassenger_1_1StaticString_1_1Hash.html +1 -1
  271. data/doc/cxxapi/tabs.css +1 -1
  272. data/doc/cxxapi/tree.html +20 -17
  273. data/doc/users_guide_snippets/tips.txt +28 -0
  274. data/ext/apache2/Hooks.cpp +19 -2
  275. data/ext/common/Utils.cpp +15 -4
  276. data/ext/common/Version.h +1 -1
  277. data/ext/nginx/StaticContentHandler.c +6 -3
  278. data/ext/phusion_passenger/native_support.c +25 -0
  279. data/lib/phusion_passenger/abstract_request_handler.rb +1 -0
  280. data/lib/phusion_passenger/constants.rb +1 -1
  281. data/lib/phusion_passenger/events.rb +3 -0
  282. data/lib/phusion_passenger/platform_info.rb +2 -1
  283. data/lib/phusion_passenger/rack/application_spawner.rb +2 -0
  284. data/lib/phusion_passenger/railz/application_spawner.rb +8 -0
  285. data/lib/phusion_passenger/utils.rb +69 -9
  286. data/lib/phusion_passenger/utils/rewindable_input.rb +10 -6
  287. metadata +55 -17
@@ -77,546 +77,555 @@
77
77
  <a name="l00055"></a>00055 <span class="preprocessor">#include "Exceptions.h"</span>
78
78
  <a name="l00056"></a>00056 <span class="preprocessor">#include "Logging.h"</span>
79
79
  <a name="l00057"></a>00057 <span class="preprocessor">#include "Utils/Base64.h"</span>
80
- <a name="l00058"></a>00058
81
- <a name="l00059"></a>00059 <span class="keyword">namespace </span>Passenger {
82
- <a name="l00060"></a>00060
83
- <a name="l00061"></a>00061 <span class="keyword">using namespace </span>std;
84
- <a name="l00062"></a>00062 <span class="keyword">using namespace </span>boost;
85
- <a name="l00063"></a>00063 <span class="keyword">using namespace </span>oxt;
86
- <a name="l00064"></a>00064 <span class="comment"></span>
87
- <a name="l00065"></a>00065 <span class="comment">/**</span>
88
- <a name="l00066"></a>00066 <span class="comment"> * An AbstractSpawnManager implementation.</span>
89
- <a name="l00067"></a>00067 <span class="comment"> *</span>
90
- <a name="l00068"></a>00068 <span class="comment"> * Internally, this class makes use of a spawn server, which is written in Ruby. This server</span>
91
- <a name="l00069"></a>00069 <span class="comment"> * is automatically started when a SpawnManager instance is created, and automatically</span>
92
- <a name="l00070"></a>00070 <span class="comment"> * shutdown when that instance is destroyed. The existance of the spawn server is almost</span>
93
- <a name="l00071"></a>00071 <span class="comment"> * totally transparent to users of this class. Spawn requests are sent to the server,</span>
94
- <a name="l00072"></a>00072 <span class="comment"> * and details about the spawned process is returned.</span>
95
- <a name="l00073"></a>00073 <span class="comment"> *</span>
96
- <a name="l00074"></a>00074 <span class="comment"> * If the spawn server dies during the middle of an operation, it will be restarted.</span>
97
- <a name="l00075"></a>00075 <span class="comment"> * See spawn() for full details.</span>
98
- <a name="l00076"></a>00076 <span class="comment"> *</span>
99
- <a name="l00077"></a>00077 <span class="comment"> * The communication channel with the server is anonymous, i.e. no other processes</span>
100
- <a name="l00078"></a>00078 <span class="comment"> * can access the communication channel, so communication is guaranteed to be safe</span>
101
- <a name="l00079"></a>00079 <span class="comment"> * (unless, of course, if the spawn server itself is a trojan).</span>
102
- <a name="l00080"></a>00080 <span class="comment"> *</span>
103
- <a name="l00081"></a>00081 <span class="comment"> * The server will try to keep the spawning time as small as possible, by keeping</span>
104
- <a name="l00082"></a>00082 <span class="comment"> * corresponding Ruby on Rails frameworks and application code in memory. So the second</span>
105
- <a name="l00083"></a>00083 <span class="comment"> * time a process of the same application is spawned, the spawn time is significantly</span>
106
- <a name="l00084"></a>00084 <span class="comment"> * lower than the first time. Nevertheless, spawning is a relatively expensive operation</span>
107
- <a name="l00085"></a>00085 <span class="comment"> * (compared to the processing of a typical HTTP request/response), and so should be</span>
108
- <a name="l00086"></a>00086 <span class="comment"> * avoided whenever possible.</span>
109
- <a name="l00087"></a>00087 <span class="comment"> *</span>
110
- <a name="l00088"></a>00088 <span class="comment"> * See the documentation of the spawn server for full implementation details.</span>
111
- <a name="l00089"></a>00089 <span class="comment"> *</span>
112
- <a name="l00090"></a>00090 <span class="comment"> * @ingroup Support</span>
113
- <a name="l00091"></a>00091 <span class="comment"> */</span>
114
- <a name="l00092"></a><a class="code" href="classPassenger_1_1SpawnManager.html">00092</a> <span class="keyword">class </span><a class="code" href="classPassenger_1_1SpawnManager.html" title="An AbstractSpawnManager implementation.">SpawnManager</a>: <span class="keyword">public</span> <a class="code" href="classPassenger_1_1AbstractSpawnManager.html" title="Spawning of application processes.">AbstractSpawnManager</a> {
115
- <a name="l00093"></a>00093 <span class="keyword">private</span>:
116
- <a name="l00094"></a>00094 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> SERVER_SOCKET_FD = 3;
117
- <a name="l00095"></a>00095 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> OWNER_SOCKET_FD = 4;
118
- <a name="l00096"></a>00096 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> HIGHEST_FD = OWNER_SOCKET_FD;
119
- <a name="l00097"></a>00097
120
- <a name="l00098"></a>00098 <span class="keywordtype">string</span> spawnServerCommand;
121
- <a name="l00099"></a>00099 ServerInstanceDir::GenerationPtr generation;
122
- <a name="l00100"></a>00100 AccountsDatabasePtr accountsDatabase;
123
- <a name="l00101"></a>00101 <span class="keywordtype">string</span> rubyCommand;
124
- <a name="l00102"></a>00102 <span class="keywordtype">string</span> loggingAgentAddress;
125
- <a name="l00103"></a>00103 <span class="keywordtype">string</span> loggingAgentUsername;
126
- <a name="l00104"></a>00104 <span class="keywordtype">string</span> loggingAgentPassword;
127
- <a name="l00105"></a>00105
128
- <a name="l00106"></a>00106 boost::mutex lock;
129
- <a name="l00107"></a>00107 <a class="code" href="classPassenger_1_1RandomGenerator.html" title="A random data generator.">RandomGenerator</a> random;
130
- <a name="l00108"></a>00108
131
- <a name="l00109"></a>00109 pid_t pid;
132
- <a name="l00110"></a>00110 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> ownerSocket;
133
- <a name="l00111"></a>00111 <span class="keywordtype">string</span> socketFilename;
134
- <a name="l00112"></a>00112 <span class="keywordtype">string</span> socketPassword;
135
- <a name="l00113"></a>00113 <span class="keywordtype">bool</span> serverNeedsRestart;
136
- <a name="l00114"></a>00114
137
- <a name="l00115"></a>00115 <span class="keyword">static</span> <span class="keywordtype">void</span> deleteAccount(AccountsDatabasePtr accountsDatabase, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;username) {
138
- <a name="l00116"></a>00116 accountsDatabase-&gt;remove(username);
139
- <a name="l00117"></a>00117 }
140
- <a name="l00118"></a>00118 <span class="comment"></span>
141
- <a name="l00119"></a>00119 <span class="comment"> /**</span>
142
- <a name="l00120"></a>00120 <span class="comment"> * Restarts the spawn server.</span>
143
- <a name="l00121"></a>00121 <span class="comment"> *</span>
144
- <a name="l00122"></a>00122 <span class="comment"> * @pre System call interruption is disabled.</span>
145
- <a name="l00123"></a>00123 <span class="comment"> * @throws RuntimeException An error occurred while creating a Unix server socket.</span>
146
- <a name="l00124"></a>00124 <span class="comment"> * @throws SystemException An error occured while trying to setup the spawn server.</span>
147
- <a name="l00125"></a>00125 <span class="comment"> * @throws IOException An error occurred while generating random data.</span>
148
- <a name="l00126"></a>00126 <span class="comment"> */</span>
149
- <a name="l00127"></a>00127 <span class="keywordtype">void</span> restartServer() {
150
- <a name="l00128"></a>00128 TRACE_POINT();
151
- <a name="l00129"></a>00129 <span class="keywordflow">if</span> (pid != 0) {
152
- <a name="l00130"></a>00130 UPDATE_TRACE_POINT();
153
- <a name="l00131"></a>00131 ownerSocket.<a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">close</a>();
154
- <a name="l00132"></a>00132
155
- <a name="l00133"></a>00133 <span class="comment">/* Wait at most 5 seconds for the spawn server to exit.</span>
156
- <a name="l00134"></a>00134 <span class="comment"> * If that doesn't work, kill it, then wait at most 5 seconds</span>
157
- <a name="l00135"></a>00135 <span class="comment"> * for it to exit.</span>
158
- <a name="l00136"></a>00136 <span class="comment"> */</span>
159
- <a name="l00137"></a>00137 time_t begin = syscalls::time(NULL);
160
- <a name="l00138"></a>00138 <span class="keywordtype">bool</span> done = <span class="keyword">false</span>;
161
- <a name="l00139"></a>00139 <span class="keywordflow">while</span> (!done &amp;&amp; syscalls::time(NULL) - begin &lt; 5) {
162
- <a name="l00140"></a>00140 <span class="keywordflow">if</span> (syscalls::waitpid(pid, NULL, WNOHANG) &gt; 0) {
163
- <a name="l00141"></a>00141 done = <span class="keyword">true</span>;
164
- <a name="l00142"></a>00142 } <span class="keywordflow">else</span> {
165
- <a name="l00143"></a>00143 syscalls::usleep(100000);
166
- <a name="l00144"></a>00144 }
167
- <a name="l00145"></a>00145 }
168
- <a name="l00146"></a>00146 UPDATE_TRACE_POINT();
169
- <a name="l00147"></a>00147 <span class="keywordflow">if</span> (!done) {
170
- <a name="l00148"></a>00148 UPDATE_TRACE_POINT();
171
- <a name="l00149"></a>00149 P_TRACE(2, <span class="stringliteral">"Spawn server did not exit in time, killing it..."</span>);
172
- <a name="l00150"></a>00150 syscalls::kill(pid, SIGTERM);
173
- <a name="l00151"></a>00151 begin = syscalls::time(NULL);
174
- <a name="l00152"></a>00152 <span class="keywordflow">while</span> (syscalls::time(NULL) - begin &lt; 5) {
175
- <a name="l00153"></a>00153 <span class="keywordflow">if</span> (syscalls::waitpid(pid, NULL, WNOHANG) &gt; 0) {
176
- <a name="l00154"></a>00154 <span class="keywordflow">break</span>;
177
- <a name="l00155"></a>00155 } <span class="keywordflow">else</span> {
178
- <a name="l00156"></a>00156 syscalls::usleep(100000);
179
- <a name="l00157"></a>00157 }
180
- <a name="l00158"></a>00158 }
181
- <a name="l00159"></a>00159 }
182
- <a name="l00160"></a>00160 pid = 0;
183
- <a name="l00161"></a>00161 }
184
- <a name="l00162"></a>00162
185
- <a name="l00163"></a>00163 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> serverSocket;
186
- <a name="l00164"></a>00164 <span class="keywordtype">string</span> socketFilename;
187
- <a name="l00165"></a>00165 <span class="keywordtype">string</span> socketPassword;
188
- <a name="l00166"></a>00166 <span class="keywordtype">int</span> ret, fds[2];
189
- <a name="l00167"></a>00167
190
- <a name="l00168"></a>00168 UPDATE_TRACE_POINT();
191
- <a name="l00169"></a>00169 socketFilename = generation-&gt;getPath() + <span class="stringliteral">"/spawn-server/socket."</span> +
192
- <a name="l00170"></a>00170 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(getpid()) + <span class="stringliteral">"."</span> +
193
- <a name="l00171"></a>00171 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>((<span class="keywordtype">unsigned</span> <span class="keywordtype">long</span> <span class="keywordtype">long</span>) <span class="keyword">this</span>);
194
- <a name="l00172"></a>00172 socketPassword = Base64::encode(random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(32));
195
- <a name="l00173"></a>00173 serverSocket = <a class="code" href="group__Support.html#g5d3b342a34d81e7506744e90ee2f9b8b" title="Create a new Unix server socket which is bounded to filename.">createUnixServer</a>(socketFilename.c_str());
196
- <a name="l00174"></a>00174 <span class="keywordflow">do</span> {
197
- <a name="l00175"></a>00175 ret = chmod(socketFilename.c_str(), S_IRUSR | S_IWUSR | S_IXUSR);
198
- <a name="l00176"></a>00176 } <span class="keywordflow">while</span> (ret == -1 &amp;&amp; errno == EINTR);
199
- <a name="l00177"></a>00177 <span class="keywordflow">if</span> (ret == -1) {
200
- <a name="l00178"></a>00178 <span class="keywordtype">int</span> e = errno;
201
- <a name="l00179"></a>00179 syscalls::unlink(socketFilename.c_str());
202
- <a name="l00180"></a>00180 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1FileSystemException.html" title="A filesystem error, as returned by the operating system.">FileSystemException</a>(<span class="stringliteral">"Cannot set permissions on '"</span> + socketFilename + <span class="stringliteral">"'"</span>,
203
- <a name="l00181"></a>00181 e, socketFilename);
204
- <a name="l00182"></a>00182 }
205
- <a name="l00183"></a>00183
206
- <a name="l00184"></a>00184 <span class="keywordflow">if</span> (syscalls::socketpair(AF_UNIX, SOCK_STREAM, 0, fds) == -1) {
207
- <a name="l00185"></a>00185 <span class="keywordtype">int</span> e = errno;
208
- <a name="l00186"></a>00186 syscalls::unlink(socketFilename.c_str());
209
- <a name="l00187"></a>00187 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(<span class="stringliteral">"Cannot create a Unix socket"</span>, e);
210
- <a name="l00188"></a>00188 }
211
- <a name="l00189"></a>00189
212
- <a name="l00190"></a>00190 UPDATE_TRACE_POINT();
213
- <a name="l00191"></a>00191 pid = syscalls::fork();
214
- <a name="l00192"></a>00192 <span class="keywordflow">if</span> (pid == 0) {
215
- <a name="l00193"></a>00193 dup2(serverSocket, HIGHEST_FD + 1);
216
- <a name="l00194"></a>00194 dup2(fds[1], HIGHEST_FD + 2);
217
- <a name="l00195"></a>00195 dup2(HIGHEST_FD + 1, SERVER_SOCKET_FD);
218
- <a name="l00196"></a>00196 dup2(HIGHEST_FD + 2, OWNER_SOCKET_FD);
219
- <a name="l00197"></a>00197
220
- <a name="l00198"></a>00198 <span class="comment">// Close all unnecessary file descriptors</span>
221
- <a name="l00199"></a>00199 <span class="keywordflow">for</span> (<span class="keywordtype">long</span> i = sysconf(_SC_OPEN_MAX) - 1; i &gt; HIGHEST_FD; i--) {
222
- <a name="l00200"></a>00200 close(i);
223
- <a name="l00201"></a>00201 }
224
- <a name="l00202"></a>00202
225
- <a name="l00203"></a>00203 execlp(rubyCommand.c_str(),
226
- <a name="l00204"></a>00204 rubyCommand.c_str(),
227
- <a name="l00205"></a>00205 spawnServerCommand.c_str(),
228
- <a name="l00206"></a>00206 <span class="comment">/* The spawn server changes the process names of the subservers</span>
229
- <a name="l00207"></a>00207 <span class="comment"> * that it starts, for better usability. However, the process name length</span>
230
- <a name="l00208"></a>00208 <span class="comment"> * (as shown by ps) is limited. Here, we try to expand that limit by</span>
231
- <a name="l00209"></a>00209 <span class="comment"> * deliberately passing a useless whitespace string to the spawn server.</span>
232
- <a name="l00210"></a>00210 <span class="comment"> * This argument is ignored by the spawn server. This works on some</span>
233
- <a name="l00211"></a>00211 <span class="comment"> * systems, such as Ubuntu Linux.</span>
234
- <a name="l00212"></a>00212 <span class="comment"> */</span>
235
- <a name="l00213"></a>00213 <span class="stringliteral">" "</span>,
236
- <a name="l00214"></a>00214 (<span class="keywordtype">char</span> *) NULL);
237
- <a name="l00215"></a>00215 <span class="keywordtype">int</span> e = errno;
238
- <a name="l00216"></a>00216 fprintf(stderr, <span class="stringliteral">"*** Passenger ERROR (%s:%d):\n"</span>
239
- <a name="l00217"></a>00217 <span class="stringliteral">"Could not start the spawn server: %s: %s (%d)\n"</span>,
240
- <a name="l00218"></a>00218 __FILE__, __LINE__,
241
- <a name="l00219"></a>00219 rubyCommand.c_str(), strerror(e), e);
242
- <a name="l00220"></a>00220 fflush(stderr);
243
- <a name="l00221"></a>00221 _exit(1);
244
- <a name="l00222"></a>00222 } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (pid == -1) {
245
- <a name="l00223"></a>00223 <span class="keywordtype">int</span> e = errno;
246
- <a name="l00224"></a>00224 syscalls::unlink(socketFilename.c_str());
247
- <a name="l00225"></a>00225 syscalls::close(fds[0]);
248
- <a name="l00226"></a>00226 syscalls::close(fds[1]);
249
- <a name="l00227"></a>00227 pid = 0;
250
- <a name="l00228"></a>00228 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(<span class="stringliteral">"Unable to fork a process"</span>, e);
251
- <a name="l00229"></a>00229 } <span class="keywordflow">else</span> {
252
- <a name="l00230"></a>00230 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> ownerSocket = fds[0];
253
- <a name="l00231"></a>00231 syscalls::close(fds[1]);
254
- <a name="l00232"></a>00232 serverSocket.close();
255
- <a name="l00233"></a>00233
256
- <a name="l00234"></a>00234 <span class="comment">// Pass arguments to spawn server.</span>
257
- <a name="l00235"></a>00235 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> ownerSocketChannel(ownerSocket);
258
- <a name="l00236"></a>00236 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(socketFilename + <span class="stringliteral">"\n"</span>);
259
- <a name="l00237"></a>00237 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(socketPassword + <span class="stringliteral">"\n"</span>);
260
- <a name="l00238"></a>00238 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(generation-&gt;getPath() + <span class="stringliteral">"\n"</span>);
261
- <a name="l00239"></a>00239 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(loggingAgentAddress + <span class="stringliteral">"\n"</span>);
262
- <a name="l00240"></a>00240 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(loggingAgentUsername + <span class="stringliteral">"\n"</span>);
263
- <a name="l00241"></a>00241 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(Base64::encode(loggingAgentPassword) + <span class="stringliteral">"\n"</span>);
264
- <a name="l00242"></a>00242
265
- <a name="l00243"></a>00243 this-&gt;ownerSocket = ownerSocket;
266
- <a name="l00244"></a>00244 this-&gt;socketFilename = socketFilename;
267
- <a name="l00245"></a>00245 this-&gt;socketPassword = socketPassword;
268
- <a name="l00246"></a>00246 <a class="code" href="classPassenger_1_1SpawnManager.html#b4dfcb11c58b07d8144b18dd99abed45" title="A method which is called after the spawn server has started.">spawnServerStarted</a>();
269
- <a name="l00247"></a>00247 }
270
- <a name="l00248"></a>00248 }
271
- <a name="l00249"></a>00249 <span class="comment"></span>
272
- <a name="l00250"></a>00250 <span class="comment"> /**</span>
273
- <a name="l00251"></a>00251 <span class="comment"> * Connects to the spawn server and returns the connection.</span>
274
- <a name="l00252"></a>00252 <span class="comment"> *</span>
275
- <a name="l00253"></a>00253 <span class="comment"> * @throws RuntimeException</span>
276
- <a name="l00254"></a>00254 <span class="comment"> * @throws SystemException</span>
277
- <a name="l00255"></a>00255 <span class="comment"> * @throws boost::thread_interrupted</span>
278
- <a name="l00256"></a>00256 <span class="comment"> */</span>
279
- <a name="l00257"></a>00257 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> connect()<span class="keyword"> const </span>{
280
- <a name="l00258"></a>00258 TRACE_POINT();
281
- <a name="l00259"></a>00259 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> fd = <a class="code" href="group__Support.html#gae3993db9dfa0b05c97b67ccefd2b619" title="Connect to a Unix server socket at filename.">connectToUnixServer</a>(socketFilename.c_str());
282
- <a name="l00260"></a>00260 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> channel(fd);
283
- <a name="l00261"></a>00261 UPDATE_TRACE_POINT();
284
- <a name="l00262"></a>00262 channel.writeScalar(socketPassword);
285
- <a name="l00263"></a>00263 <span class="keywordflow">return</span> fd;
286
- <a name="l00264"></a>00264 }
287
- <a name="l00265"></a>00265 <span class="comment"></span>
288
- <a name="l00266"></a>00266 <span class="comment"> /**</span>
289
- <a name="l00267"></a>00267 <span class="comment"> * Send the spawn command to the spawn server.</span>
290
- <a name="l00268"></a>00268 <span class="comment"> *</span>
291
- <a name="l00269"></a>00269 <span class="comment"> * @param PoolOptions The spawn options to use.</span>
292
- <a name="l00270"></a>00270 <span class="comment"> * @return A Process smart pointer, representing the spawned process.</span>
293
- <a name="l00271"></a>00271 <span class="comment"> * @throws SpawnException Something went wrong.</span>
294
- <a name="l00272"></a>00272 <span class="comment"> * @throws Anything thrown by options.environmentVariables-&gt;getItems().</span>
295
- <a name="l00273"></a>00273 <span class="comment"> * @throws boost::thread_interrupted</span>
296
- <a name="l00274"></a>00274 <span class="comment"> */</span>
297
- <a name="l00275"></a>00275 <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a> sendSpawnCommand(<span class="keyword">const</span> <a class="code" href="structPassenger_1_1PoolOptions.html" title="This struct encapsulates information for ApplicationPool::get() and for SpawnManager::spawn()...">PoolOptions</a> &amp;options) {
298
- <a name="l00276"></a>00276 TRACE_POINT();
299
- <a name="l00277"></a>00277 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> connection;
300
- <a name="l00278"></a>00278 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> channel;
301
- <a name="l00279"></a>00279
302
- <a name="l00280"></a>00280 <span class="keywordflow">try</span> {
303
- <a name="l00281"></a>00281 connection = connect();
304
- <a name="l00282"></a>00282 channel = <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a>(connection);
305
- <a name="l00283"></a>00283 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
306
- <a name="l00284"></a>00284 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not connect to the spawn server: "</span>) +
307
- <a name="l00285"></a>00285 e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
308
- <a name="l00286"></a>00286 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> std::exception &amp;e) {
309
- <a name="l00287"></a>00287 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not connect to the spawn server: "</span>) +
310
- <a name="l00288"></a>00288 e.what());
311
- <a name="l00289"></a>00289 }
312
- <a name="l00290"></a>00290
313
- <a name="l00291"></a>00291 UPDATE_TRACE_POINT();
314
- <a name="l00292"></a>00292 vector&lt;string&gt; args;
315
- <a name="l00293"></a>00293 <span class="keywordtype">string</span> appRoot;
316
- <a name="l00294"></a>00294 pid_t appPid;
317
- <a name="l00295"></a>00295 <span class="keywordtype">int</span> i, nServerSockets, ownerPipe;
318
- <a name="l00296"></a>00296 Process::SocketInfoMap serverSockets;
319
- <a name="l00297"></a>00297 <span class="keywordtype">string</span> detachKey = Base64::encode(random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(32));
320
- <a name="l00298"></a>00298 <span class="comment">// The connect password must be a URL-friendly string because users will</span>
321
- <a name="l00299"></a>00299 <span class="comment">// insert it in HTTP headers.</span>
322
- <a name="l00300"></a>00300 <span class="keywordtype">string</span> connectPassword = Base64::encodeForUrl(random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(32));
323
- <a name="l00301"></a>00301 AccountPtr account;
324
- <a name="l00302"></a>00302 function&lt;void ()&gt; destructionCallback;
325
- <a name="l00303"></a>00303
326
- <a name="l00304"></a>00304 <span class="keywordflow">try</span> {
327
- <a name="l00305"></a>00305 args.push_back(<span class="stringliteral">"spawn_application"</span>);
328
- <a name="l00306"></a>00306 options.<a class="code" href="structPassenger_1_1PoolOptions.html#437e077b0ab037f4e86680e9164a694e" title="Append the information in this PoolOptions object to the given string vector.">toVector</a>(args);
329
- <a name="l00307"></a>00307
330
- <a name="l00308"></a>00308 args.push_back(<span class="stringliteral">"detach_key"</span>);
331
- <a name="l00309"></a>00309 args.push_back(detachKey);
332
- <a name="l00310"></a>00310 args.push_back(<span class="stringliteral">"connect_password"</span>);
333
- <a name="l00311"></a>00311 args.push_back(connectPassword);
334
- <a name="l00312"></a>00312 <span class="keywordflow">if</span> (accountsDatabase != NULL) {
335
- <a name="l00313"></a>00313 <span class="keywordtype">string</span> username = <span class="stringliteral">"_backend-"</span> + <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(accountsDatabase-&gt;getUniqueNumber());
336
- <a name="l00314"></a>00314 <span class="keywordtype">string</span> password = random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(MESSAGE_SERVER_MAX_PASSWORD_SIZE);
337
- <a name="l00315"></a>00315 account = accountsDatabase-&gt;add(username, password, <span class="keyword">false</span>, options.<a class="code" href="structPassenger_1_1PoolOptions.html#45665fffc8a20ea60a3fcafaff8627ab" title="Any rights that the spawned application process may have.">rights</a>);
338
- <a name="l00316"></a>00316 destructionCallback = boost::bind(&amp;SpawnManager::deleteAccount,
339
- <a name="l00317"></a>00317 accountsDatabase, username);
340
- <a name="l00318"></a>00318
341
- <a name="l00319"></a>00319 args.push_back(<span class="stringliteral">"pool_account_username"</span>);
342
- <a name="l00320"></a>00320 args.push_back(username);
343
- <a name="l00321"></a>00321 args.push_back(<span class="stringliteral">"pool_account_password_base64"</span>);
344
- <a name="l00322"></a>00322 args.push_back(Base64::encode(password));
345
- <a name="l00323"></a>00323 }
346
- <a name="l00324"></a>00324
347
- <a name="l00325"></a>00325 channel.<a class="code" href="classPassenger_1_1MessageChannel.html#150f68bc47e060dcc6ca35bee047914d" title="Send an array message, which consists of the given elements, over the underlying...">write</a>(args);
348
- <a name="l00326"></a>00326 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
349
- <a name="l00327"></a>00327 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not write 'spawn_application' "</span>
350
- <a name="l00328"></a>00328 <span class="stringliteral">"command to the spawn server: "</span>) + e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
351
- <a name="l00329"></a>00329 }
352
- <a name="l00330"></a>00330
353
- <a name="l00331"></a>00331 <span class="keywordflow">try</span> {
354
- <a name="l00332"></a>00332 UPDATE_TRACE_POINT();
355
- <a name="l00333"></a>00333 <span class="comment">// Read status.</span>
356
- <a name="l00334"></a>00334 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#129659b60d1a663337873d2af944431e" title="Read an array message from the underlying file descriptor.">read</a>(args)) {
357
- <a name="l00335"></a>00335 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
358
- <a name="l00336"></a>00336 }
359
- <a name="l00337"></a>00337 <span class="keywordflow">if</span> (args.size() != 1) {
360
- <a name="l00338"></a>00338 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
361
- <a name="l00339"></a>00339 }
362
- <a name="l00340"></a>00340 <span class="keywordflow">if</span> (args[0] == <span class="stringliteral">"error_page"</span>) {
363
- <a name="l00341"></a>00341 UPDATE_TRACE_POINT();
364
- <a name="l00342"></a>00342 <span class="keywordtype">string</span> errorPage;
365
- <a name="l00343"></a>00343
366
- <a name="l00344"></a>00344 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#1f3cf672b566166ed920c2bba4d19604" title="Read a scalar message from the underlying file descriptor.">readScalar</a>(errorPage)) {
367
- <a name="l00345"></a>00345 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
368
- <a name="l00346"></a>00346 }
369
- <a name="l00347"></a>00347 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"An error occured while spawning the application."</span>,
370
- <a name="l00348"></a>00348 errorPage);
371
- <a name="l00349"></a>00349 } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (args[0] != <span class="stringliteral">"ok"</span>) {
372
- <a name="l00350"></a>00350 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
373
- <a name="l00351"></a>00351 }
374
- <a name="l00352"></a>00352
375
- <a name="l00353"></a>00353 <span class="comment">// Read application info.</span>
376
- <a name="l00354"></a>00354 UPDATE_TRACE_POINT();
377
- <a name="l00355"></a>00355 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#129659b60d1a663337873d2af944431e" title="Read an array message from the underlying file descriptor.">read</a>(args)) {
378
- <a name="l00356"></a>00356 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
379
- <a name="l00357"></a>00357 }
380
- <a name="l00358"></a>00358 <span class="keywordflow">if</span> (args.size() != 3) {
381
- <a name="l00359"></a>00359 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
382
- <a name="l00360"></a>00360 }
383
- <a name="l00361"></a>00361
384
- <a name="l00362"></a>00362 appRoot = args[0];
385
- <a name="l00363"></a>00363 appPid = (pid_t) atoll(args[1].c_str());
386
- <a name="l00364"></a>00364 nServerSockets = <a class="code" href="group__Support.html#g73b17b509ee44938a56bf16cbf82fb48" title="Converts the given string to an integer.">atoi</a>(args[2]);
387
- <a name="l00365"></a>00365
388
- <a name="l00366"></a>00366 UPDATE_TRACE_POINT();
389
- <a name="l00367"></a>00367 <span class="keywordflow">for</span> (i = 0; i &lt; nServerSockets; i++) {
390
- <a name="l00368"></a>00368 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#129659b60d1a663337873d2af944431e" title="Read an array message from the underlying file descriptor.">read</a>(args)) {
391
- <a name="l00369"></a>00369 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
392
- <a name="l00370"></a>00370 }
393
- <a name="l00371"></a>00371 <span class="keywordflow">if</span> (args.size() != 3) {
394
- <a name="l00372"></a>00372 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
395
- <a name="l00373"></a>00373 }
396
- <a name="l00374"></a>00374 serverSockets[args[0]] = Process::SocketInfo(args[1], args[2]);
397
- <a name="l00375"></a>00375 }
398
- <a name="l00376"></a>00376 <span class="keywordflow">if</span> (serverSockets.find(<span class="stringliteral">"main"</span>) == serverSockets.end()) {
399
- <a name="l00377"></a>00377 UPDATE_TRACE_POINT();
400
- <a name="l00378"></a>00378 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
401
- <a name="l00379"></a>00379 }
402
- <a name="l00380"></a>00380 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
403
- <a name="l00381"></a>00381 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not read from the spawn server: "</span>) + e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
404
- <a name="l00382"></a>00382 }
405
- <a name="l00383"></a>00383
406
- <a name="l00384"></a>00384 UPDATE_TRACE_POINT();
407
- <a name="l00385"></a>00385 <span class="keywordflow">try</span> {
408
- <a name="l00386"></a>00386 ownerPipe = channel.<a class="code" href="classPassenger_1_1MessageChannel.html#1a28ba9d078f38f2831673092ac9dcbc" title="Receive a file descriptor, which had been passed over the underlying file descriptor...">readFileDescriptor</a>();
409
- <a name="l00387"></a>00387 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
410
- <a name="l00388"></a>00388 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not receive the spawned "</span>
411
- <a name="l00389"></a>00389 <span class="stringliteral">"application's owner pipe from the spawn server: "</span>) +
412
- <a name="l00390"></a>00390 e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
413
- <a name="l00391"></a>00391 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
414
- <a name="l00392"></a>00392 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not receive the spawned "</span>
415
- <a name="l00393"></a>00393 <span class="stringliteral">"application's owner pipe from the spawn server: "</span>) +
416
- <a name="l00394"></a>00394 e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
417
- <a name="l00395"></a>00395 }
418
- <a name="l00396"></a>00396
419
- <a name="l00397"></a>00397 UPDATE_TRACE_POINT();
420
- <a name="l00398"></a>00398 <span class="keywordflow">return</span> <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a>(<span class="keyword">new</span> <a class="code" href="classPassenger_1_1Process.html" title="Represents a single application process, as spawned by SpawnManager or by ApplicationPool::Interface...">Process</a>(appRoot, appPid, ownerPipe, serverSockets,
421
- <a name="l00399"></a>00399 detachKey, connectPassword, destructionCallback));
422
- <a name="l00400"></a>00400 }
423
- <a name="l00401"></a>00401 <span class="comment"></span>
424
- <a name="l00402"></a>00402 <span class="comment"> /**</span>
425
- <a name="l00403"></a>00403 <span class="comment"> * @throws boost::thread_interrupted</span>
426
- <a name="l00404"></a>00404 <span class="comment"> * @throws Anything thrown by options.environmentVariables-&gt;getItems().</span>
427
- <a name="l00405"></a>00405 <span class="comment"> */</span>
428
- <a name="l00406"></a>00406 <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a>
429
- <a name="l00407"></a>00407 handleSpawnException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a> &amp;e, <span class="keyword">const</span> <a class="code" href="structPassenger_1_1PoolOptions.html" title="This struct encapsulates information for ApplicationPool::get() and for SpawnManager::spawn()...">PoolOptions</a> &amp;options) {
430
- <a name="l00408"></a>00408 TRACE_POINT();
431
- <a name="l00409"></a>00409 <span class="keywordtype">bool</span> restarted;
432
- <a name="l00410"></a>00410 <span class="keywordflow">try</span> {
433
- <a name="l00411"></a>00411 P_DEBUG(<span class="stringliteral">"Spawn server died. Attempting to restart it..."</span>);
434
- <a name="l00412"></a>00412 this_thread::disable_syscall_interruption dsi;
435
- <a name="l00413"></a>00413 restartServer();
436
- <a name="l00414"></a>00414 P_DEBUG(<span class="stringliteral">"Restart seems to be successful."</span>);
437
- <a name="l00415"></a>00415 restarted = <span class="keyword">true</span>;
438
- <a name="l00416"></a>00416 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
439
- <a name="l00417"></a>00417 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
440
- <a name="l00418"></a>00418 restarted = <span class="keyword">false</span>;
441
- <a name="l00419"></a>00419 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
442
- <a name="l00420"></a>00420 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1SystemException.html#ee405d1078ff3aa1045bc60b1c93d7a3">what</a>());
443
- <a name="l00421"></a>00421 restarted = <span class="keyword">false</span>;
444
- <a name="l00422"></a>00422 }
445
- <a name="l00423"></a>00423 <span class="keywordflow">if</span> (restarted) {
446
- <a name="l00424"></a>00424 <span class="keywordflow">return</span> sendSpawnCommand(options);
447
- <a name="l00425"></a>00425 } <span class="keywordflow">else</span> {
448
- <a name="l00426"></a>00426 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server died unexpectedly, and restarting it failed."</span>);
449
- <a name="l00427"></a>00427 }
450
- <a name="l00428"></a>00428 }
451
- <a name="l00429"></a>00429 <span class="comment"></span>
452
- <a name="l00430"></a>00430 <span class="comment"> /**</span>
453
- <a name="l00431"></a>00431 <span class="comment"> * Send the reload command to the spawn server.</span>
454
- <a name="l00432"></a>00432 <span class="comment"> *</span>
455
- <a name="l00433"></a>00433 <span class="comment"> * @param appRoot The application root to reload.</span>
456
- <a name="l00434"></a>00434 <span class="comment"> * @throws RuntimeException </span>
457
- <a name="l00435"></a>00435 <span class="comment"> * @throws SystemException</span>
458
- <a name="l00436"></a>00436 <span class="comment"> * @throws boost::thread_interrupted</span>
459
- <a name="l00437"></a>00437 <span class="comment"> */</span>
460
- <a name="l00438"></a>00438 <span class="keywordtype">void</span> sendReloadCommand(<span class="keyword">const</span> <span class="keywordtype">string</span> &amp;appRoot) {
461
- <a name="l00439"></a>00439 TRACE_POINT();
462
- <a name="l00440"></a>00440 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> connection;
463
- <a name="l00441"></a>00441 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> channel;
464
- <a name="l00442"></a>00442
465
- <a name="l00443"></a>00443 <span class="keywordflow">try</span> {
466
- <a name="l00444"></a>00444 connection = connect();
467
- <a name="l00445"></a>00445 channel = <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a>(connection);
468
- <a name="l00446"></a>00446 } <span class="keywordflow">catch</span> (<a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
469
- <a name="l00447"></a>00447 e.<a class="code" href="classPassenger_1_1SystemException.html#6fc359f27a084158b7111ef35bec114f">setBriefMessage</a>(<span class="stringliteral">"Could not connect to the spawn server"</span>);
470
- <a name="l00448"></a>00448 <span class="keywordflow">throw</span>;
471
- <a name="l00449"></a>00449 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1RuntimeException.html" title="A generic runtime exception.">RuntimeException</a> &amp;e) {
472
- <a name="l00450"></a>00450 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1RuntimeException.html" title="A generic runtime exception.">RuntimeException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not connect to the spawn server: "</span>) +
473
- <a name="l00451"></a>00451 e.<a class="code" href="classPassenger_1_1RuntimeException.html#b8fc54bfa94be7e6435af440f9fd72d0">what</a>());
474
- <a name="l00452"></a>00452 }
475
- <a name="l00453"></a>00453
476
- <a name="l00454"></a>00454 <span class="keywordflow">try</span> {
477
- <a name="l00455"></a>00455 channel.<a class="code" href="classPassenger_1_1MessageChannel.html#150f68bc47e060dcc6ca35bee047914d" title="Send an array message, which consists of the given elements, over the underlying...">write</a>(<span class="stringliteral">"reload"</span>, appRoot.c_str(), NULL);
478
- <a name="l00456"></a>00456 } <span class="keywordflow">catch</span> (<a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
479
- <a name="l00457"></a>00457 e.<a class="code" href="classPassenger_1_1SystemException.html#6fc359f27a084158b7111ef35bec114f">setBriefMessage</a>(<span class="stringliteral">"Could not write 'reload' command to the spawn server"</span>);
480
- <a name="l00458"></a>00458 <span class="keywordflow">throw</span>;
481
- <a name="l00459"></a>00459 }
482
- <a name="l00460"></a>00460 }
483
- <a name="l00461"></a>00461
484
- <a name="l00462"></a>00462 <span class="keywordtype">void</span> handleReloadException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;appRoot) {
485
- <a name="l00463"></a>00463 TRACE_POINT();
486
- <a name="l00464"></a>00464 <span class="keywordtype">bool</span> restarted;
487
- <a name="l00465"></a>00465 <span class="keywordflow">try</span> {
488
- <a name="l00466"></a>00466 P_DEBUG(<span class="stringliteral">"Spawn server died. Attempting to restart it..."</span>);
489
- <a name="l00467"></a>00467 restartServer();
490
- <a name="l00468"></a>00468 P_DEBUG(<span class="stringliteral">"Restart seems to be successful."</span>);
491
- <a name="l00469"></a>00469 restarted = <span class="keyword">true</span>;
492
- <a name="l00470"></a>00470 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
493
- <a name="l00471"></a>00471 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
494
- <a name="l00472"></a>00472 restarted = <span class="keyword">false</span>;
495
- <a name="l00473"></a>00473 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
496
- <a name="l00474"></a>00474 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1SystemException.html#ee405d1078ff3aa1045bc60b1c93d7a3">what</a>());
497
- <a name="l00475"></a>00475 restarted = <span class="keyword">false</span>;
498
- <a name="l00476"></a>00476 }
499
- <a name="l00477"></a>00477 <span class="keywordflow">if</span> (restarted) {
500
- <a name="l00478"></a>00478 <span class="keywordflow">return</span> sendReloadCommand(appRoot);
501
- <a name="l00479"></a>00479 } <span class="keywordflow">else</span> {
502
- <a name="l00480"></a>00480 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server died unexpectedly, and restarting it failed."</span>);
503
- <a name="l00481"></a>00481 }
504
- <a name="l00482"></a>00482 }
505
- <a name="l00483"></a>00483
506
- <a name="l00484"></a>00484 <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> prependMessageToException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;message) {
507
- <a name="l00485"></a>00485 <span class="keywordflow">return</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a>(message + <span class="stringliteral">": "</span> + e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
508
- <a name="l00486"></a>00486 }
509
- <a name="l00487"></a>00487
510
- <a name="l00488"></a>00488 <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> prependMessageToException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;message) {
511
- <a name="l00489"></a>00489 <span class="keywordflow">return</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(message + <span class="stringliteral">": "</span> + e.<a class="code" href="classPassenger_1_1SystemException.html#84fa2ab4f5b7b96704734fcdfdaa0269" title="Returns a brief version of the exception message.">brief</a>(), e.<a class="code" href="classPassenger_1_1SystemException.html#ee7a6672bf79b72a4c3ee70c57d6a47c" title="The value of errno at the time the error occured.">code</a>());
80
+ <a name="l00058"></a>00058 <span class="preprocessor">#include "Utils/SystemTime.h"</span>
81
+ <a name="l00059"></a>00059
82
+ <a name="l00060"></a>00060 <span class="keyword">namespace </span>Passenger {
83
+ <a name="l00061"></a>00061
84
+ <a name="l00062"></a>00062 <span class="keyword">using namespace </span>std;
85
+ <a name="l00063"></a>00063 <span class="keyword">using namespace </span>boost;
86
+ <a name="l00064"></a>00064 <span class="keyword">using namespace </span>oxt;
87
+ <a name="l00065"></a>00065 <span class="comment"></span>
88
+ <a name="l00066"></a>00066 <span class="comment">/**</span>
89
+ <a name="l00067"></a>00067 <span class="comment"> * An AbstractSpawnManager implementation.</span>
90
+ <a name="l00068"></a>00068 <span class="comment"> *</span>
91
+ <a name="l00069"></a>00069 <span class="comment"> * Internally, this class makes use of a spawn server, which is written in Ruby. This server</span>
92
+ <a name="l00070"></a>00070 <span class="comment"> * is automatically started when a SpawnManager instance is created, and automatically</span>
93
+ <a name="l00071"></a>00071 <span class="comment"> * shutdown when that instance is destroyed. The existance of the spawn server is almost</span>
94
+ <a name="l00072"></a>00072 <span class="comment"> * totally transparent to users of this class. Spawn requests are sent to the server,</span>
95
+ <a name="l00073"></a>00073 <span class="comment"> * and details about the spawned process is returned.</span>
96
+ <a name="l00074"></a>00074 <span class="comment"> *</span>
97
+ <a name="l00075"></a>00075 <span class="comment"> * If the spawn server dies during the middle of an operation, it will be restarted.</span>
98
+ <a name="l00076"></a>00076 <span class="comment"> * See spawn() for full details.</span>
99
+ <a name="l00077"></a>00077 <span class="comment"> *</span>
100
+ <a name="l00078"></a>00078 <span class="comment"> * The communication channel with the server is anonymous, i.e. no other processes</span>
101
+ <a name="l00079"></a>00079 <span class="comment"> * can access the communication channel, so communication is guaranteed to be safe</span>
102
+ <a name="l00080"></a>00080 <span class="comment"> * (unless, of course, if the spawn server itself is a trojan).</span>
103
+ <a name="l00081"></a>00081 <span class="comment"> *</span>
104
+ <a name="l00082"></a>00082 <span class="comment"> * The server will try to keep the spawning time as small as possible, by keeping</span>
105
+ <a name="l00083"></a>00083 <span class="comment"> * corresponding Ruby on Rails frameworks and application code in memory. So the second</span>
106
+ <a name="l00084"></a>00084 <span class="comment"> * time a process of the same application is spawned, the spawn time is significantly</span>
107
+ <a name="l00085"></a>00085 <span class="comment"> * lower than the first time. Nevertheless, spawning is a relatively expensive operation</span>
108
+ <a name="l00086"></a>00086 <span class="comment"> * (compared to the processing of a typical HTTP request/response), and so should be</span>
109
+ <a name="l00087"></a>00087 <span class="comment"> * avoided whenever possible.</span>
110
+ <a name="l00088"></a>00088 <span class="comment"> *</span>
111
+ <a name="l00089"></a>00089 <span class="comment"> * See the documentation of the spawn server for full implementation details.</span>
112
+ <a name="l00090"></a>00090 <span class="comment"> *</span>
113
+ <a name="l00091"></a>00091 <span class="comment"> * @ingroup Support</span>
114
+ <a name="l00092"></a>00092 <span class="comment"> */</span>
115
+ <a name="l00093"></a><a class="code" href="classPassenger_1_1SpawnManager.html">00093</a> <span class="keyword">class </span><a class="code" href="classPassenger_1_1SpawnManager.html" title="An AbstractSpawnManager implementation.">SpawnManager</a>: <span class="keyword">public</span> <a class="code" href="classPassenger_1_1AbstractSpawnManager.html" title="Spawning of application processes.">AbstractSpawnManager</a> {
116
+ <a name="l00094"></a>00094 <span class="keyword">private</span>:
117
+ <a name="l00095"></a>00095 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> SERVER_SOCKET_FD = 3;
118
+ <a name="l00096"></a>00096 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> OWNER_SOCKET_FD = 4;
119
+ <a name="l00097"></a>00097 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> HIGHEST_FD = OWNER_SOCKET_FD;
120
+ <a name="l00098"></a>00098
121
+ <a name="l00099"></a>00099 <span class="keywordtype">string</span> spawnServerCommand;
122
+ <a name="l00100"></a>00100 ServerInstanceDir::GenerationPtr generation;
123
+ <a name="l00101"></a>00101 AccountsDatabasePtr accountsDatabase;
124
+ <a name="l00102"></a>00102 <span class="keywordtype">string</span> rubyCommand;
125
+ <a name="l00103"></a>00103 AnalyticsLoggerPtr analyticsLogger;
126
+ <a name="l00104"></a>00104
127
+ <a name="l00105"></a>00105 boost::mutex lock;
128
+ <a name="l00106"></a>00106 <a class="code" href="classPassenger_1_1RandomGenerator.html" title="A random data generator.">RandomGenerator</a> random;
129
+ <a name="l00107"></a>00107
130
+ <a name="l00108"></a>00108 pid_t pid;
131
+ <a name="l00109"></a>00109 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> ownerSocket;
132
+ <a name="l00110"></a>00110 <span class="keywordtype">string</span> socketFilename;
133
+ <a name="l00111"></a>00111 <span class="keywordtype">string</span> socketPassword;
134
+ <a name="l00112"></a>00112 <span class="keywordtype">bool</span> serverNeedsRestart;
135
+ <a name="l00113"></a>00113
136
+ <a name="l00114"></a>00114 <span class="keyword">static</span> <span class="keywordtype">void</span> deleteAccount(AccountsDatabasePtr accountsDatabase, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;username) {
137
+ <a name="l00115"></a>00115 accountsDatabase-&gt;remove(username);
138
+ <a name="l00116"></a>00116 }
139
+ <a name="l00117"></a>00117 <span class="comment"></span>
140
+ <a name="l00118"></a>00118 <span class="comment"> /**</span>
141
+ <a name="l00119"></a>00119 <span class="comment"> * Restarts the spawn server.</span>
142
+ <a name="l00120"></a>00120 <span class="comment"> *</span>
143
+ <a name="l00121"></a>00121 <span class="comment"> * @pre System call interruption is disabled.</span>
144
+ <a name="l00122"></a>00122 <span class="comment"> * @throws RuntimeException An error occurred while creating a Unix server socket.</span>
145
+ <a name="l00123"></a>00123 <span class="comment"> * @throws SystemException An error occured while trying to setup the spawn server.</span>
146
+ <a name="l00124"></a>00124 <span class="comment"> * @throws IOException An error occurred while generating random data.</span>
147
+ <a name="l00125"></a>00125 <span class="comment"> */</span>
148
+ <a name="l00126"></a>00126 <span class="keywordtype">void</span> restartServer() {
149
+ <a name="l00127"></a>00127 TRACE_POINT();
150
+ <a name="l00128"></a>00128 <span class="keywordflow">if</span> (pid != 0) {
151
+ <a name="l00129"></a>00129 UPDATE_TRACE_POINT();
152
+ <a name="l00130"></a>00130 ownerSocket.<a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">close</a>();
153
+ <a name="l00131"></a>00131
154
+ <a name="l00132"></a>00132 <span class="comment">/* Wait at most 5 seconds for the spawn server to exit.</span>
155
+ <a name="l00133"></a>00133 <span class="comment"> * If that doesn't work, kill it, then wait at most 5 seconds</span>
156
+ <a name="l00134"></a>00134 <span class="comment"> * for it to exit.</span>
157
+ <a name="l00135"></a>00135 <span class="comment"> */</span>
158
+ <a name="l00136"></a>00136 time_t begin = syscalls::time(NULL);
159
+ <a name="l00137"></a>00137 <span class="keywordtype">bool</span> done = <span class="keyword">false</span>;
160
+ <a name="l00138"></a>00138 <span class="keywordflow">while</span> (!done &amp;&amp; syscalls::time(NULL) - begin &lt; 5) {
161
+ <a name="l00139"></a>00139 <span class="keywordflow">if</span> (syscalls::waitpid(pid, NULL, WNOHANG) &gt; 0) {
162
+ <a name="l00140"></a>00140 done = <span class="keyword">true</span>;
163
+ <a name="l00141"></a>00141 } <span class="keywordflow">else</span> {
164
+ <a name="l00142"></a>00142 syscalls::usleep(100000);
165
+ <a name="l00143"></a>00143 }
166
+ <a name="l00144"></a>00144 }
167
+ <a name="l00145"></a>00145 UPDATE_TRACE_POINT();
168
+ <a name="l00146"></a>00146 <span class="keywordflow">if</span> (!done) {
169
+ <a name="l00147"></a>00147 UPDATE_TRACE_POINT();
170
+ <a name="l00148"></a>00148 P_TRACE(2, <span class="stringliteral">"Spawn server did not exit in time, killing it..."</span>);
171
+ <a name="l00149"></a>00149 syscalls::kill(pid, SIGTERM);
172
+ <a name="l00150"></a>00150 begin = syscalls::time(NULL);
173
+ <a name="l00151"></a>00151 <span class="keywordflow">while</span> (syscalls::time(NULL) - begin &lt; 5) {
174
+ <a name="l00152"></a>00152 <span class="keywordflow">if</span> (syscalls::waitpid(pid, NULL, WNOHANG) &gt; 0) {
175
+ <a name="l00153"></a>00153 <span class="keywordflow">break</span>;
176
+ <a name="l00154"></a>00154 } <span class="keywordflow">else</span> {
177
+ <a name="l00155"></a>00155 syscalls::usleep(100000);
178
+ <a name="l00156"></a>00156 }
179
+ <a name="l00157"></a>00157 }
180
+ <a name="l00158"></a>00158 }
181
+ <a name="l00159"></a>00159 pid = 0;
182
+ <a name="l00160"></a>00160 }
183
+ <a name="l00161"></a>00161
184
+ <a name="l00162"></a>00162 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> serverSocket;
185
+ <a name="l00163"></a>00163 <span class="keywordtype">string</span> socketFilename;
186
+ <a name="l00164"></a>00164 <span class="keywordtype">string</span> socketPassword;
187
+ <a name="l00165"></a>00165 <span class="keywordtype">int</span> ret, fds[2];
188
+ <a name="l00166"></a>00166
189
+ <a name="l00167"></a>00167 UPDATE_TRACE_POINT();
190
+ <a name="l00168"></a>00168 socketFilename = generation-&gt;getPath() + <span class="stringliteral">"/spawn-server/socket."</span> +
191
+ <a name="l00169"></a>00169 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(getpid()) + <span class="stringliteral">"."</span> +
192
+ <a name="l00170"></a>00170 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>((<span class="keywordtype">unsigned</span> <span class="keywordtype">long</span> <span class="keywordtype">long</span>) <span class="keyword">this</span>);
193
+ <a name="l00171"></a>00171 socketPassword = Base64::encode(random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(32));
194
+ <a name="l00172"></a>00172 serverSocket = <a class="code" href="group__Support.html#g5d3b342a34d81e7506744e90ee2f9b8b" title="Create a new Unix server socket which is bounded to filename.">createUnixServer</a>(socketFilename.c_str());
195
+ <a name="l00173"></a>00173 <span class="keywordflow">do</span> {
196
+ <a name="l00174"></a>00174 ret = chmod(socketFilename.c_str(), S_IRUSR | S_IWUSR | S_IXUSR);
197
+ <a name="l00175"></a>00175 } <span class="keywordflow">while</span> (ret == -1 &amp;&amp; errno == EINTR);
198
+ <a name="l00176"></a>00176 <span class="keywordflow">if</span> (ret == -1) {
199
+ <a name="l00177"></a>00177 <span class="keywordtype">int</span> e = errno;
200
+ <a name="l00178"></a>00178 syscalls::unlink(socketFilename.c_str());
201
+ <a name="l00179"></a>00179 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1FileSystemException.html" title="A filesystem error, as returned by the operating system.">FileSystemException</a>(<span class="stringliteral">"Cannot set permissions on '"</span> + socketFilename + <span class="stringliteral">"'"</span>,
202
+ <a name="l00180"></a>00180 e, socketFilename);
203
+ <a name="l00181"></a>00181 }
204
+ <a name="l00182"></a>00182
205
+ <a name="l00183"></a>00183 <span class="keywordflow">if</span> (syscalls::socketpair(AF_UNIX, SOCK_STREAM, 0, fds) == -1) {
206
+ <a name="l00184"></a>00184 <span class="keywordtype">int</span> e = errno;
207
+ <a name="l00185"></a>00185 syscalls::unlink(socketFilename.c_str());
208
+ <a name="l00186"></a>00186 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(<span class="stringliteral">"Cannot create a Unix socket"</span>, e);
209
+ <a name="l00187"></a>00187 }
210
+ <a name="l00188"></a>00188
211
+ <a name="l00189"></a>00189 UPDATE_TRACE_POINT();
212
+ <a name="l00190"></a>00190 pid = syscalls::fork();
213
+ <a name="l00191"></a>00191 <span class="keywordflow">if</span> (pid == 0) {
214
+ <a name="l00192"></a>00192 dup2(serverSocket, HIGHEST_FD + 1);
215
+ <a name="l00193"></a>00193 dup2(fds[1], HIGHEST_FD + 2);
216
+ <a name="l00194"></a>00194 dup2(HIGHEST_FD + 1, SERVER_SOCKET_FD);
217
+ <a name="l00195"></a>00195 dup2(HIGHEST_FD + 2, OWNER_SOCKET_FD);
218
+ <a name="l00196"></a>00196
219
+ <a name="l00197"></a>00197 <span class="comment">// Close all unnecessary file descriptors</span>
220
+ <a name="l00198"></a>00198 <span class="keywordflow">for</span> (<span class="keywordtype">long</span> i = sysconf(_SC_OPEN_MAX) - 1; i &gt; HIGHEST_FD; i--) {
221
+ <a name="l00199"></a>00199 close(i);
222
+ <a name="l00200"></a>00200 }
223
+ <a name="l00201"></a>00201
224
+ <a name="l00202"></a>00202 execlp(rubyCommand.c_str(),
225
+ <a name="l00203"></a>00203 rubyCommand.c_str(),
226
+ <a name="l00204"></a>00204 spawnServerCommand.c_str(),
227
+ <a name="l00205"></a>00205 <span class="comment">/* The spawn server changes the process names of the subservers</span>
228
+ <a name="l00206"></a>00206 <span class="comment"> * that it starts, for better usability. However, the process name length</span>
229
+ <a name="l00207"></a>00207 <span class="comment"> * (as shown by ps) is limited. Here, we try to expand that limit by</span>
230
+ <a name="l00208"></a>00208 <span class="comment"> * deliberately passing a useless whitespace string to the spawn server.</span>
231
+ <a name="l00209"></a>00209 <span class="comment"> * This argument is ignored by the spawn server. This works on some</span>
232
+ <a name="l00210"></a>00210 <span class="comment"> * systems, such as Ubuntu Linux.</span>
233
+ <a name="l00211"></a>00211 <span class="comment"> */</span>
234
+ <a name="l00212"></a>00212 <span class="stringliteral">" "</span>,
235
+ <a name="l00213"></a>00213 (<span class="keywordtype">char</span> *) NULL);
236
+ <a name="l00214"></a>00214 <span class="keywordtype">int</span> e = errno;
237
+ <a name="l00215"></a>00215 fprintf(stderr, <span class="stringliteral">"*** Passenger ERROR (%s:%d):\n"</span>
238
+ <a name="l00216"></a>00216 <span class="stringliteral">"Could not start the spawn server: %s: %s (%d)\n"</span>,
239
+ <a name="l00217"></a>00217 __FILE__, __LINE__,
240
+ <a name="l00218"></a>00218 rubyCommand.c_str(), strerror(e), e);
241
+ <a name="l00219"></a>00219 fflush(stderr);
242
+ <a name="l00220"></a>00220 _exit(1);
243
+ <a name="l00221"></a>00221 } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (pid == -1) {
244
+ <a name="l00222"></a>00222 <span class="keywordtype">int</span> e = errno;
245
+ <a name="l00223"></a>00223 syscalls::unlink(socketFilename.c_str());
246
+ <a name="l00224"></a>00224 syscalls::close(fds[0]);
247
+ <a name="l00225"></a>00225 syscalls::close(fds[1]);
248
+ <a name="l00226"></a>00226 pid = 0;
249
+ <a name="l00227"></a>00227 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(<span class="stringliteral">"Unable to fork a process"</span>, e);
250
+ <a name="l00228"></a>00228 } <span class="keywordflow">else</span> {
251
+ <a name="l00229"></a>00229 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> ownerSocket = fds[0];
252
+ <a name="l00230"></a>00230 syscalls::close(fds[1]);
253
+ <a name="l00231"></a>00231 serverSocket.close();
254
+ <a name="l00232"></a>00232
255
+ <a name="l00233"></a>00233 <span class="comment">// Pass arguments to spawn server.</span>
256
+ <a name="l00234"></a>00234 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> ownerSocketChannel(ownerSocket);
257
+ <a name="l00235"></a>00235 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(socketFilename + <span class="stringliteral">"\n"</span>);
258
+ <a name="l00236"></a>00236 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(socketPassword + <span class="stringliteral">"\n"</span>);
259
+ <a name="l00237"></a>00237 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(generation-&gt;getPath() + <span class="stringliteral">"\n"</span>);
260
+ <a name="l00238"></a>00238 <span class="keywordflow">if</span> (analyticsLogger != NULL) {
261
+ <a name="l00239"></a>00239 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(analyticsLogger-&gt;getAddress() + <span class="stringliteral">"\n"</span>);
262
+ <a name="l00240"></a>00240 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(analyticsLogger-&gt;getUsername() + <span class="stringliteral">"\n"</span>);
263
+ <a name="l00241"></a>00241 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(Base64::encode(analyticsLogger-&gt;getPassword()) + <span class="stringliteral">"\n"</span>);
264
+ <a name="l00242"></a>00242 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(analyticsLogger-&gt;getNodeName() + <span class="stringliteral">"\n"</span>);
265
+ <a name="l00243"></a>00243 } <span class="keywordflow">else</span> {
266
+ <a name="l00244"></a>00244 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(<span class="stringliteral">"\n"</span>);
267
+ <a name="l00245"></a>00245 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(<span class="stringliteral">"\n"</span>);
268
+ <a name="l00246"></a>00246 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(<span class="stringliteral">"\n"</span>);
269
+ <a name="l00247"></a>00247 ownerSocketChannel.<a class="code" href="classPassenger_1_1MessageChannel.html#069314e4c7e1fe8c8ab36e16d2cc5fef" title="Send a block of data over the underlying file descriptor.">writeRaw</a>(<span class="stringliteral">"\n"</span>);
270
+ <a name="l00248"></a>00248 }
271
+ <a name="l00249"></a>00249
272
+ <a name="l00250"></a>00250 this-&gt;ownerSocket = ownerSocket;
273
+ <a name="l00251"></a>00251 this-&gt;socketFilename = socketFilename;
274
+ <a name="l00252"></a>00252 this-&gt;socketPassword = socketPassword;
275
+ <a name="l00253"></a>00253 <a class="code" href="classPassenger_1_1SpawnManager.html#b4dfcb11c58b07d8144b18dd99abed45" title="A method which is called after the spawn server has started.">spawnServerStarted</a>();
276
+ <a name="l00254"></a>00254 }
277
+ <a name="l00255"></a>00255 }
278
+ <a name="l00256"></a>00256 <span class="comment"></span>
279
+ <a name="l00257"></a>00257 <span class="comment"> /**</span>
280
+ <a name="l00258"></a>00258 <span class="comment"> * Connects to the spawn server and returns the connection.</span>
281
+ <a name="l00259"></a>00259 <span class="comment"> *</span>
282
+ <a name="l00260"></a>00260 <span class="comment"> * @throws RuntimeException</span>
283
+ <a name="l00261"></a>00261 <span class="comment"> * @throws SystemException</span>
284
+ <a name="l00262"></a>00262 <span class="comment"> * @throws boost::thread_interrupted</span>
285
+ <a name="l00263"></a>00263 <span class="comment"> */</span>
286
+ <a name="l00264"></a>00264 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> connect()<span class="keyword"> const </span>{
287
+ <a name="l00265"></a>00265 TRACE_POINT();
288
+ <a name="l00266"></a>00266 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> fd = <a class="code" href="group__Support.html#gae3993db9dfa0b05c97b67ccefd2b619" title="Connect to a Unix server socket at filename.">connectToUnixServer</a>(socketFilename.c_str());
289
+ <a name="l00267"></a>00267 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> channel(fd);
290
+ <a name="l00268"></a>00268 UPDATE_TRACE_POINT();
291
+ <a name="l00269"></a>00269 channel.writeScalar(socketPassword);
292
+ <a name="l00270"></a>00270 <span class="keywordflow">return</span> fd;
293
+ <a name="l00271"></a>00271 }
294
+ <a name="l00272"></a>00272 <span class="comment"></span>
295
+ <a name="l00273"></a>00273 <span class="comment"> /**</span>
296
+ <a name="l00274"></a>00274 <span class="comment"> * Send the spawn command to the spawn server.</span>
297
+ <a name="l00275"></a>00275 <span class="comment"> *</span>
298
+ <a name="l00276"></a>00276 <span class="comment"> * @param PoolOptions The spawn options to use.</span>
299
+ <a name="l00277"></a>00277 <span class="comment"> * @return A Process smart pointer, representing the spawned process.</span>
300
+ <a name="l00278"></a>00278 <span class="comment"> * @throws SpawnException Something went wrong.</span>
301
+ <a name="l00279"></a>00279 <span class="comment"> * @throws Anything thrown by options.environmentVariables-&gt;getItems().</span>
302
+ <a name="l00280"></a>00280 <span class="comment"> * @throws boost::thread_interrupted</span>
303
+ <a name="l00281"></a>00281 <span class="comment"> */</span>
304
+ <a name="l00282"></a>00282 <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a> sendSpawnCommand(<span class="keyword">const</span> <a class="code" href="structPassenger_1_1PoolOptions.html" title="This struct encapsulates information for ApplicationPool::get() and for SpawnManager::spawn()...">PoolOptions</a> &amp;options) {
305
+ <a name="l00283"></a>00283 TRACE_POINT();
306
+ <a name="l00284"></a>00284 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> connection;
307
+ <a name="l00285"></a>00285 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> channel;
308
+ <a name="l00286"></a>00286
309
+ <a name="l00287"></a>00287 <span class="keywordflow">try</span> {
310
+ <a name="l00288"></a>00288 connection = connect();
311
+ <a name="l00289"></a>00289 channel = <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a>(connection);
312
+ <a name="l00290"></a>00290 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
313
+ <a name="l00291"></a>00291 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not connect to the spawn server: "</span>) +
314
+ <a name="l00292"></a>00292 e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
315
+ <a name="l00293"></a>00293 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> std::exception &amp;e) {
316
+ <a name="l00294"></a>00294 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not connect to the spawn server: "</span>) +
317
+ <a name="l00295"></a>00295 e.what());
318
+ <a name="l00296"></a>00296 }
319
+ <a name="l00297"></a>00297
320
+ <a name="l00298"></a>00298 UPDATE_TRACE_POINT();
321
+ <a name="l00299"></a>00299 vector&lt;string&gt; args;
322
+ <a name="l00300"></a>00300 <span class="keywordtype">string</span> appRoot;
323
+ <a name="l00301"></a>00301 pid_t appPid;
324
+ <a name="l00302"></a>00302 <span class="keywordtype">int</span> i, nServerSockets, ownerPipe;
325
+ <a name="l00303"></a>00303 Process::SocketInfoMap serverSockets;
326
+ <a name="l00304"></a>00304 <span class="keywordtype">string</span> detachKey = Base64::encode(random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(32));
327
+ <a name="l00305"></a>00305 <span class="comment">// The connect password must be a URL-friendly string because users will</span>
328
+ <a name="l00306"></a>00306 <span class="comment">// insert it in HTTP headers.</span>
329
+ <a name="l00307"></a>00307 <span class="keywordtype">string</span> connectPassword = Base64::encodeForUrl(random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(32));
330
+ <a name="l00308"></a>00308 <span class="keywordtype">string</span> gupid = random.<a class="code" href="classPassenger_1_1RandomGenerator.html#d9d9a489f6dbfabafb35da326f5ed967" title="Generates a random string of size bytes which is also valid ASCII.">generateAsciiString</a>(11) + <span class="stringliteral">"-"</span> + <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(SystemTime::get());
331
+ <a name="l00309"></a>00309 AccountPtr account;
332
+ <a name="l00310"></a>00310 function&lt;void ()&gt; destructionCallback;
333
+ <a name="l00311"></a>00311
334
+ <a name="l00312"></a>00312 <span class="keywordflow">try</span> {
335
+ <a name="l00313"></a>00313 args.push_back(<span class="stringliteral">"spawn_application"</span>);
336
+ <a name="l00314"></a>00314 options.<a class="code" href="structPassenger_1_1PoolOptions.html#437e077b0ab037f4e86680e9164a694e" title="Append the information in this PoolOptions object to the given string vector.">toVector</a>(args);
337
+ <a name="l00315"></a>00315
338
+ <a name="l00316"></a>00316 args.push_back(<span class="stringliteral">"detach_key"</span>);
339
+ <a name="l00317"></a>00317 args.push_back(detachKey);
340
+ <a name="l00318"></a>00318 args.push_back(<span class="stringliteral">"connect_password"</span>);
341
+ <a name="l00319"></a>00319 args.push_back(connectPassword);
342
+ <a name="l00320"></a>00320 <span class="keywordflow">if</span> (accountsDatabase != NULL) {
343
+ <a name="l00321"></a>00321 <span class="keywordtype">string</span> username = <span class="stringliteral">"_backend-"</span> + <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(accountsDatabase-&gt;getUniqueNumber());
344
+ <a name="l00322"></a>00322 <span class="keywordtype">string</span> password = random.<a class="code" href="classPassenger_1_1RandomGenerator.html#02cdc94154658d63241f7d65d9d2d865">generateByteString</a>(MESSAGE_SERVER_MAX_PASSWORD_SIZE);
345
+ <a name="l00323"></a>00323 account = accountsDatabase-&gt;add(username, password, <span class="keyword">false</span>, options.<a class="code" href="structPassenger_1_1PoolOptions.html#45665fffc8a20ea60a3fcafaff8627ab" title="Any rights that the spawned application process may have.">rights</a>);
346
+ <a name="l00324"></a>00324 destructionCallback = boost::bind(&amp;SpawnManager::deleteAccount,
347
+ <a name="l00325"></a>00325 accountsDatabase, username);
348
+ <a name="l00326"></a>00326
349
+ <a name="l00327"></a>00327 args.push_back(<span class="stringliteral">"pool_account_username"</span>);
350
+ <a name="l00328"></a>00328 args.push_back(username);
351
+ <a name="l00329"></a>00329 args.push_back(<span class="stringliteral">"pool_account_password_base64"</span>);
352
+ <a name="l00330"></a>00330 args.push_back(Base64::encode(password));
353
+ <a name="l00331"></a>00331 }
354
+ <a name="l00332"></a>00332
355
+ <a name="l00333"></a>00333 channel.<a class="code" href="classPassenger_1_1MessageChannel.html#150f68bc47e060dcc6ca35bee047914d" title="Send an array message, which consists of the given elements, over the underlying...">write</a>(args);
356
+ <a name="l00334"></a>00334 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
357
+ <a name="l00335"></a>00335 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not write 'spawn_application' "</span>
358
+ <a name="l00336"></a>00336 <span class="stringliteral">"command to the spawn server: "</span>) + e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
359
+ <a name="l00337"></a>00337 }
360
+ <a name="l00338"></a>00338
361
+ <a name="l00339"></a>00339 <span class="keywordflow">try</span> {
362
+ <a name="l00340"></a>00340 UPDATE_TRACE_POINT();
363
+ <a name="l00341"></a>00341 <span class="comment">// Read status.</span>
364
+ <a name="l00342"></a>00342 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#129659b60d1a663337873d2af944431e" title="Read an array message from the underlying file descriptor.">read</a>(args)) {
365
+ <a name="l00343"></a>00343 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
366
+ <a name="l00344"></a>00344 }
367
+ <a name="l00345"></a>00345 <span class="keywordflow">if</span> (args.size() != 1) {
368
+ <a name="l00346"></a>00346 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
369
+ <a name="l00347"></a>00347 }
370
+ <a name="l00348"></a>00348 <span class="keywordflow">if</span> (args[0] == <span class="stringliteral">"error_page"</span>) {
371
+ <a name="l00349"></a>00349 UPDATE_TRACE_POINT();
372
+ <a name="l00350"></a>00350 <span class="keywordtype">string</span> errorPage;
373
+ <a name="l00351"></a>00351
374
+ <a name="l00352"></a>00352 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#1f3cf672b566166ed920c2bba4d19604" title="Read a scalar message from the underlying file descriptor.">readScalar</a>(errorPage)) {
375
+ <a name="l00353"></a>00353 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
376
+ <a name="l00354"></a>00354 }
377
+ <a name="l00355"></a>00355 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"An error occured while spawning the application."</span>,
378
+ <a name="l00356"></a>00356 errorPage);
379
+ <a name="l00357"></a>00357 } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (args[0] != <span class="stringliteral">"ok"</span>) {
380
+ <a name="l00358"></a>00358 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
381
+ <a name="l00359"></a>00359 }
382
+ <a name="l00360"></a>00360
383
+ <a name="l00361"></a>00361 <span class="comment">// Read application info.</span>
384
+ <a name="l00362"></a>00362 UPDATE_TRACE_POINT();
385
+ <a name="l00363"></a>00363 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#129659b60d1a663337873d2af944431e" title="Read an array message from the underlying file descriptor.">read</a>(args)) {
386
+ <a name="l00364"></a>00364 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
387
+ <a name="l00365"></a>00365 }
388
+ <a name="l00366"></a>00366 <span class="keywordflow">if</span> (args.size() != 3) {
389
+ <a name="l00367"></a>00367 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
390
+ <a name="l00368"></a>00368 }
391
+ <a name="l00369"></a>00369
392
+ <a name="l00370"></a>00370 appRoot = args[0];
393
+ <a name="l00371"></a>00371 appPid = (pid_t) atoll(args[1].c_str());
394
+ <a name="l00372"></a>00372 nServerSockets = <a class="code" href="group__Support.html#g73b17b509ee44938a56bf16cbf82fb48" title="Converts the given string to an integer.">atoi</a>(args[2]);
395
+ <a name="l00373"></a>00373
396
+ <a name="l00374"></a>00374 UPDATE_TRACE_POINT();
397
+ <a name="l00375"></a>00375 <span class="keywordflow">for</span> (i = 0; i &lt; nServerSockets; i++) {
398
+ <a name="l00376"></a>00376 <span class="keywordflow">if</span> (!channel.<a class="code" href="classPassenger_1_1MessageChannel.html#129659b60d1a663337873d2af944431e" title="Read an array message from the underlying file descriptor.">read</a>(args)) {
399
+ <a name="l00377"></a>00377 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server has exited unexpectedly."</span>);
400
+ <a name="l00378"></a>00378 }
401
+ <a name="l00379"></a>00379 <span class="keywordflow">if</span> (args.size() != 3) {
402
+ <a name="l00380"></a>00380 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
403
+ <a name="l00381"></a>00381 }
404
+ <a name="l00382"></a>00382 serverSockets[args[0]] = Process::SocketInfo(args[1], args[2]);
405
+ <a name="l00383"></a>00383 }
406
+ <a name="l00384"></a>00384 <span class="keywordflow">if</span> (serverSockets.find(<span class="stringliteral">"main"</span>) == serverSockets.end()) {
407
+ <a name="l00385"></a>00385 UPDATE_TRACE_POINT();
408
+ <a name="l00386"></a>00386 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server sent an invalid message."</span>);
409
+ <a name="l00387"></a>00387 }
410
+ <a name="l00388"></a>00388 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
411
+ <a name="l00389"></a>00389 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not read from the spawn server: "</span>) + e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
412
+ <a name="l00390"></a>00390 }
413
+ <a name="l00391"></a>00391
414
+ <a name="l00392"></a>00392 UPDATE_TRACE_POINT();
415
+ <a name="l00393"></a>00393 <span class="keywordflow">try</span> {
416
+ <a name="l00394"></a>00394 ownerPipe = channel.<a class="code" href="classPassenger_1_1MessageChannel.html#1a28ba9d078f38f2831673092ac9dcbc" title="Receive a file descriptor, which had been passed over the underlying file descriptor...">readFileDescriptor</a>();
417
+ <a name="l00395"></a>00395 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
418
+ <a name="l00396"></a>00396 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not receive the spawned "</span>
419
+ <a name="l00397"></a>00397 <span class="stringliteral">"application's owner pipe from the spawn server: "</span>) +
420
+ <a name="l00398"></a>00398 e.<a class="code" href="classPassenger_1_1SystemException.html#0ef41d755fea2c9b1db2252e7c418dc8" title="Returns the system&amp;#39;s error message.">sys</a>());
421
+ <a name="l00399"></a>00399 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
422
+ <a name="l00400"></a>00400 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not receive the spawned "</span>
423
+ <a name="l00401"></a>00401 <span class="stringliteral">"application's owner pipe from the spawn server: "</span>) +
424
+ <a name="l00402"></a>00402 e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
425
+ <a name="l00403"></a>00403 }
426
+ <a name="l00404"></a>00404
427
+ <a name="l00405"></a>00405 UPDATE_TRACE_POINT();
428
+ <a name="l00406"></a>00406 <span class="keywordflow">return</span> <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a>(<span class="keyword">new</span> <a class="code" href="classPassenger_1_1Process.html" title="Represents a single application process, as spawned by SpawnManager or by ApplicationPool::Interface...">Process</a>(appRoot, appPid, ownerPipe, serverSockets,
429
+ <a name="l00407"></a>00407 detachKey, connectPassword, gupid, destructionCallback));
430
+ <a name="l00408"></a>00408 }
431
+ <a name="l00409"></a>00409 <span class="comment"></span>
432
+ <a name="l00410"></a>00410 <span class="comment"> /**</span>
433
+ <a name="l00411"></a>00411 <span class="comment"> * @throws boost::thread_interrupted</span>
434
+ <a name="l00412"></a>00412 <span class="comment"> * @throws Anything thrown by options.environmentVariables-&gt;getItems().</span>
435
+ <a name="l00413"></a>00413 <span class="comment"> */</span>
436
+ <a name="l00414"></a>00414 <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a>
437
+ <a name="l00415"></a>00415 handleSpawnException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a> &amp;e, <span class="keyword">const</span> <a class="code" href="structPassenger_1_1PoolOptions.html" title="This struct encapsulates information for ApplicationPool::get() and for SpawnManager::spawn()...">PoolOptions</a> &amp;options) {
438
+ <a name="l00416"></a>00416 TRACE_POINT();
439
+ <a name="l00417"></a>00417 <span class="keywordtype">bool</span> restarted;
440
+ <a name="l00418"></a>00418 <span class="keywordflow">try</span> {
441
+ <a name="l00419"></a>00419 P_DEBUG(<span class="stringliteral">"Spawn server died. Attempting to restart it..."</span>);
442
+ <a name="l00420"></a>00420 this_thread::disable_syscall_interruption dsi;
443
+ <a name="l00421"></a>00421 restartServer();
444
+ <a name="l00422"></a>00422 P_DEBUG(<span class="stringliteral">"Restart seems to be successful."</span>);
445
+ <a name="l00423"></a>00423 restarted = <span class="keyword">true</span>;
446
+ <a name="l00424"></a>00424 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
447
+ <a name="l00425"></a>00425 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
448
+ <a name="l00426"></a>00426 restarted = <span class="keyword">false</span>;
449
+ <a name="l00427"></a>00427 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
450
+ <a name="l00428"></a>00428 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1SystemException.html#ee405d1078ff3aa1045bc60b1c93d7a3">what</a>());
451
+ <a name="l00429"></a>00429 restarted = <span class="keyword">false</span>;
452
+ <a name="l00430"></a>00430 }
453
+ <a name="l00431"></a>00431 <span class="keywordflow">if</span> (restarted) {
454
+ <a name="l00432"></a>00432 <span class="keywordflow">return</span> sendSpawnCommand(options);
455
+ <a name="l00433"></a>00433 } <span class="keywordflow">else</span> {
456
+ <a name="l00434"></a>00434 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server died unexpectedly, and restarting it failed."</span>);
457
+ <a name="l00435"></a>00435 }
458
+ <a name="l00436"></a>00436 }
459
+ <a name="l00437"></a>00437 <span class="comment"></span>
460
+ <a name="l00438"></a>00438 <span class="comment"> /**</span>
461
+ <a name="l00439"></a>00439 <span class="comment"> * Send the reload command to the spawn server.</span>
462
+ <a name="l00440"></a>00440 <span class="comment"> *</span>
463
+ <a name="l00441"></a>00441 <span class="comment"> * @param appRoot The application root to reload.</span>
464
+ <a name="l00442"></a>00442 <span class="comment"> * @throws RuntimeException </span>
465
+ <a name="l00443"></a>00443 <span class="comment"> * @throws SystemException</span>
466
+ <a name="l00444"></a>00444 <span class="comment"> * @throws boost::thread_interrupted</span>
467
+ <a name="l00445"></a>00445 <span class="comment"> */</span>
468
+ <a name="l00446"></a>00446 <span class="keywordtype">void</span> sendReloadCommand(<span class="keyword">const</span> <span class="keywordtype">string</span> &amp;appRoot) {
469
+ <a name="l00447"></a>00447 TRACE_POINT();
470
+ <a name="l00448"></a>00448 <a class="code" href="classPassenger_1_1FileDescriptor.html" title="Wrapper class around a file descriptor integer, for RAII behavior.">FileDescriptor</a> connection;
471
+ <a name="l00449"></a>00449 <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a> channel;
472
+ <a name="l00450"></a>00450
473
+ <a name="l00451"></a>00451 <span class="keywordflow">try</span> {
474
+ <a name="l00452"></a>00452 connection = connect();
475
+ <a name="l00453"></a>00453 channel = <a class="code" href="classPassenger_1_1MessageChannel.html" title="Convenience class for I/O operations on file descriptors.">MessageChannel</a>(connection);
476
+ <a name="l00454"></a>00454 } <span class="keywordflow">catch</span> (<a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
477
+ <a name="l00455"></a>00455 e.<a class="code" href="classPassenger_1_1SystemException.html#6fc359f27a084158b7111ef35bec114f">setBriefMessage</a>(<span class="stringliteral">"Could not connect to the spawn server"</span>);
478
+ <a name="l00456"></a>00456 <span class="keywordflow">throw</span>;
479
+ <a name="l00457"></a>00457 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1RuntimeException.html" title="A generic runtime exception.">RuntimeException</a> &amp;e) {
480
+ <a name="l00458"></a>00458 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1RuntimeException.html" title="A generic runtime exception.">RuntimeException</a>(<span class="keywordtype">string</span>(<span class="stringliteral">"Could not connect to the spawn server: "</span>) +
481
+ <a name="l00459"></a>00459 e.<a class="code" href="classPassenger_1_1RuntimeException.html#b8fc54bfa94be7e6435af440f9fd72d0">what</a>());
482
+ <a name="l00460"></a>00460 }
483
+ <a name="l00461"></a>00461
484
+ <a name="l00462"></a>00462 <span class="keywordflow">try</span> {
485
+ <a name="l00463"></a>00463 channel.<a class="code" href="classPassenger_1_1MessageChannel.html#150f68bc47e060dcc6ca35bee047914d" title="Send an array message, which consists of the given elements, over the underlying...">write</a>(<span class="stringliteral">"reload"</span>, appRoot.c_str(), NULL);
486
+ <a name="l00464"></a>00464 } <span class="keywordflow">catch</span> (<a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
487
+ <a name="l00465"></a>00465 e.<a class="code" href="classPassenger_1_1SystemException.html#6fc359f27a084158b7111ef35bec114f">setBriefMessage</a>(<span class="stringliteral">"Could not write 'reload' command to the spawn server"</span>);
488
+ <a name="l00466"></a>00466 <span class="keywordflow">throw</span>;
489
+ <a name="l00467"></a>00467 }
490
+ <a name="l00468"></a>00468 }
491
+ <a name="l00469"></a>00469
492
+ <a name="l00470"></a>00470 <span class="keywordtype">void</span> handleReloadException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;appRoot) {
493
+ <a name="l00471"></a>00471 TRACE_POINT();
494
+ <a name="l00472"></a>00472 <span class="keywordtype">bool</span> restarted;
495
+ <a name="l00473"></a>00473 <span class="keywordflow">try</span> {
496
+ <a name="l00474"></a>00474 P_DEBUG(<span class="stringliteral">"Spawn server died. Attempting to restart it..."</span>);
497
+ <a name="l00475"></a>00475 restartServer();
498
+ <a name="l00476"></a>00476 P_DEBUG(<span class="stringliteral">"Restart seems to be successful."</span>);
499
+ <a name="l00477"></a>00477 restarted = <span class="keyword">true</span>;
500
+ <a name="l00478"></a>00478 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
501
+ <a name="l00479"></a>00479 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
502
+ <a name="l00480"></a>00480 restarted = <span class="keyword">false</span>;
503
+ <a name="l00481"></a>00481 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
504
+ <a name="l00482"></a>00482 P_DEBUG(<span class="stringliteral">"Restart failed: "</span> &lt;&lt; e.<a class="code" href="classPassenger_1_1SystemException.html#ee405d1078ff3aa1045bc60b1c93d7a3">what</a>());
505
+ <a name="l00483"></a>00483 restarted = <span class="keyword">false</span>;
506
+ <a name="l00484"></a>00484 }
507
+ <a name="l00485"></a>00485 <span class="keywordflow">if</span> (restarted) {
508
+ <a name="l00486"></a>00486 <span class="keywordflow">return</span> sendReloadCommand(appRoot);
509
+ <a name="l00487"></a>00487 } <span class="keywordflow">else</span> {
510
+ <a name="l00488"></a>00488 <span class="keywordflow">throw</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a>(<span class="stringliteral">"The spawn server died unexpectedly, and restarting it failed."</span>);
511
+ <a name="l00489"></a>00489 }
512
512
  <a name="l00490"></a>00490 }
513
- <a name="l00491"></a>00491
514
- <a name="l00492"></a>00492 <span class="keyword">protected</span>:<span class="comment"></span>
515
- <a name="l00493"></a>00493 <span class="comment"> /**</span>
516
- <a name="l00494"></a>00494 <span class="comment"> * A method which is called after the spawn server has started.</span>
517
- <a name="l00495"></a>00495 <span class="comment"> * It doesn't do anything by default and serves as a hook for unit tests.</span>
518
- <a name="l00496"></a>00496 <span class="comment"> */</span>
519
- <a name="l00497"></a><a class="code" href="classPassenger_1_1SpawnManager.html#b4dfcb11c58b07d8144b18dd99abed45">00497</a> <span class="keyword">virtual</span> <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1SpawnManager.html#b4dfcb11c58b07d8144b18dd99abed45" title="A method which is called after the spawn server has started.">spawnServerStarted</a>() { }
520
- <a name="l00498"></a>00498
521
- <a name="l00499"></a>00499 <span class="keyword">public</span>:<span class="comment"></span>
522
- <a name="l00500"></a>00500 <span class="comment"> /**</span>
523
- <a name="l00501"></a>00501 <span class="comment"> * Construct a new SpawnManager.</span>
524
- <a name="l00502"></a>00502 <span class="comment"> *</span>
525
- <a name="l00503"></a>00503 <span class="comment"> * @param spawnServerCommand The filename of the spawn server to use.</span>
526
- <a name="l00504"></a>00504 <span class="comment"> * @param generation The server instance dir generation in which</span>
527
- <a name="l00505"></a>00505 <span class="comment"> * generation-specific are stored.</span>
528
- <a name="l00506"></a>00506 <span class="comment"> * @param accountsDatabase An accounts database. SpawnManager will automatically</span>
529
- <a name="l00507"></a>00507 <span class="comment"> * create a new account for each spawned process, assigning</span>
530
- <a name="l00508"></a>00508 <span class="comment"> * it the rights as set in the PoolOptions object. This</span>
531
- <a name="l00509"></a>00509 <span class="comment"> * account is also automatically deleted when no longer needed.</span>
532
- <a name="l00510"></a>00510 <span class="comment"> * May be a null pointer.</span>
533
- <a name="l00511"></a>00511 <span class="comment"> * @param rubyCommand The Ruby interpreter's command.</span>
534
- <a name="l00512"></a>00512 <span class="comment"> * @throws RuntimeException An error occurred while creating a Unix server socket.</span>
535
- <a name="l00513"></a>00513 <span class="comment"> * @throws SystemException An error occured while trying to setup the spawn server.</span>
536
- <a name="l00514"></a>00514 <span class="comment"> * @throws IOException An error occurred while generating random data.</span>
537
- <a name="l00515"></a>00515 <span class="comment"> */</span>
538
- <a name="l00516"></a><a class="code" href="classPassenger_1_1SpawnManager.html#5805c2919457e5bcb0eaf4f7107ba555">00516</a> <a class="code" href="classPassenger_1_1SpawnManager.html#5805c2919457e5bcb0eaf4f7107ba555" title="Construct a new SpawnManager.">SpawnManager</a>(<span class="keyword">const</span> <span class="keywordtype">string</span> &amp;spawnServerCommand,
539
- <a name="l00517"></a>00517 <span class="keyword">const</span> ServerInstanceDir::GenerationPtr &amp;generation,
540
- <a name="l00518"></a>00518 <span class="keyword">const</span> AccountsDatabasePtr &amp;accountsDatabase = AccountsDatabasePtr(),
541
- <a name="l00519"></a>00519 <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;rubyCommand = <span class="stringliteral">"ruby"</span>,
542
- <a name="l00520"></a>00520 <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;loggingAgentAddress = <span class="stringliteral">""</span>,
543
- <a name="l00521"></a>00521 <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;loggingAgentUsername = <span class="stringliteral">""</span>,
544
- <a name="l00522"></a>00522 <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;loggingAgentPassword = <span class="stringliteral">""</span>
545
- <a name="l00523"></a>00523 ) {
546
- <a name="l00524"></a>00524 TRACE_POINT();
547
- <a name="l00525"></a>00525 this-&gt;spawnServerCommand = spawnServerCommand;
548
- <a name="l00526"></a>00526 this-&gt;generation = generation;
549
- <a name="l00527"></a>00527 this-&gt;accountsDatabase = accountsDatabase;
550
- <a name="l00528"></a>00528 this-&gt;rubyCommand = rubyCommand;
551
- <a name="l00529"></a>00529 this-&gt;loggingAgentAddress = loggingAgentAddress;
552
- <a name="l00530"></a>00530 this-&gt;loggingAgentUsername = loggingAgentUsername;
553
- <a name="l00531"></a>00531 this-&gt;loggingAgentPassword = loggingAgentPassword;
554
- <a name="l00532"></a>00532 pid = 0;
555
- <a name="l00533"></a>00533 this_thread::disable_interruption di;
556
- <a name="l00534"></a>00534 this_thread::disable_syscall_interruption dsi;
557
- <a name="l00535"></a>00535 <span class="keywordflow">try</span> {
558
- <a name="l00536"></a>00536 restartServer();
559
- <a name="l00537"></a>00537 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
560
- <a name="l00538"></a>00538 <span class="keywordflow">throw</span> prependMessageToException(e, <span class="stringliteral">"Could not start the spawn server"</span>);
561
- <a name="l00539"></a>00539 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
562
- <a name="l00540"></a>00540 <span class="keywordflow">throw</span> prependMessageToException(e, <span class="stringliteral">"Could not start the spawn server"</span>);
563
- <a name="l00541"></a>00541 }
564
- <a name="l00542"></a>00542 }
565
- <a name="l00543"></a>00543
566
- <a name="l00544"></a>00544 <span class="keyword">virtual</span> ~<a class="code" href="classPassenger_1_1SpawnManager.html" title="An AbstractSpawnManager implementation.">SpawnManager</a>() {
567
- <a name="l00545"></a>00545 TRACE_POINT();
568
- <a name="l00546"></a>00546 <span class="keywordflow">if</span> (pid != 0) {
569
- <a name="l00547"></a>00547 UPDATE_TRACE_POINT();
570
- <a name="l00548"></a>00548 this_thread::disable_interruption di;
571
- <a name="l00549"></a>00549 this_thread::disable_syscall_interruption dsi;
572
- <a name="l00550"></a>00550 syscalls::unlink(socketFilename.c_str());
573
- <a name="l00551"></a>00551 ownerSocket.<a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">close</a>();
574
- <a name="l00552"></a>00552 syscalls::waitpid(pid, NULL, 0);
575
- <a name="l00553"></a>00553 }
576
- <a name="l00554"></a>00554 }
577
- <a name="l00555"></a>00555
578
- <a name="l00556"></a><a class="code" href="classPassenger_1_1SpawnManager.html#88a94ba4d41dab492cb1dc8a0f998973">00556</a> <span class="keyword">virtual</span> <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a> <a class="code" href="classPassenger_1_1SpawnManager.html#88a94ba4d41dab492cb1dc8a0f998973" title="Spawn a new application process.">spawn</a>(<span class="keyword">const</span> <a class="code" href="structPassenger_1_1PoolOptions.html" title="This struct encapsulates information for ApplicationPool::get() and for SpawnManager::spawn()...">PoolOptions</a> &amp;options) {
579
- <a name="l00557"></a>00557 TRACE_POINT();
580
- <a name="l00558"></a>00558 boost::mutex::scoped_lock l(lock);
581
- <a name="l00559"></a>00559 <span class="keywordflow">try</span> {
582
- <a name="l00560"></a>00560 <span class="keywordflow">return</span> sendSpawnCommand(options);
583
- <a name="l00561"></a>00561 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a> &amp;e) {
584
- <a name="l00562"></a>00562 <span class="keywordflow">if</span> (e.<a class="code" href="classPassenger_1_1SpawnException.html#e65dc272e183fd9e5637a4091cc6bbf4" title="Check whether an error page is available.">hasErrorPage</a>()) {
585
- <a name="l00563"></a>00563 <span class="keywordflow">throw</span>;
586
- <a name="l00564"></a>00564 } <span class="keywordflow">else</span> {
587
- <a name="l00565"></a>00565 <span class="keywordflow">return</span> handleSpawnException(e, options);
588
- <a name="l00566"></a>00566 }
589
- <a name="l00567"></a>00567 }
590
- <a name="l00568"></a>00568 }
591
- <a name="l00569"></a>00569
592
- <a name="l00570"></a><a class="code" href="classPassenger_1_1SpawnManager.html#ad71b2ea9431f3286f23a6e6f80e0981">00570</a> <span class="keyword">virtual</span> <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1SpawnManager.html#ad71b2ea9431f3286f23a6e6f80e0981" title="Shutdown the ApplicationSpawner server that&amp;#39;s running at the given application...">reload</a>(<span class="keyword">const</span> <span class="keywordtype">string</span> &amp;appRoot) {
593
- <a name="l00571"></a>00571 TRACE_POINT();
594
- <a name="l00572"></a>00572 this_thread::disable_interruption di;
595
- <a name="l00573"></a>00573 this_thread::disable_syscall_interruption dsi;
596
- <a name="l00574"></a>00574 <span class="keywordflow">try</span> {
597
- <a name="l00575"></a>00575 <span class="keywordflow">return</span> sendReloadCommand(appRoot);
598
- <a name="l00576"></a>00576 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
599
- <a name="l00577"></a>00577 <span class="keywordflow">return</span> handleReloadException(e, appRoot);
600
- <a name="l00578"></a>00578 }
601
- <a name="l00579"></a>00579 }
602
- <a name="l00580"></a>00580
603
- <a name="l00581"></a><a class="code" href="classPassenger_1_1SpawnManager.html#9210212c23c5a8ee5ee4dd5c9dcdf57d">00581</a> <span class="keyword">virtual</span> <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1SpawnManager.html#9210212c23c5a8ee5ee4dd5c9dcdf57d" title="Forcefully kill the spawn server.">killSpawnServer</a>()<span class="keyword"> const </span>{
604
- <a name="l00582"></a>00582 kill(pid, SIGKILL);
605
- <a name="l00583"></a>00583 }
606
- <a name="l00584"></a>00584
607
- <a name="l00585"></a><a class="code" href="classPassenger_1_1SpawnManager.html#f34df8cfccfc64507999f1b48bc5a126">00585</a> <span class="keyword">virtual</span> pid_t <a class="code" href="classPassenger_1_1SpawnManager.html#f34df8cfccfc64507999f1b48bc5a126" title="Returns the spawn server&amp;#39;s PID.">getServerPid</a>()<span class="keyword"> const </span>{
608
- <a name="l00586"></a>00586 <span class="keywordflow">return</span> pid;
609
- <a name="l00587"></a>00587 }
610
- <a name="l00588"></a>00588 };
611
- <a name="l00589"></a>00589 <span class="comment"></span>
612
- <a name="l00590"></a>00590 <span class="comment">/** Convenient alias for SpawnManager smart pointer. */</span>
613
- <a name="l00591"></a><a class="code" href="namespacePassenger.html#6ab2d566ff87a2676c23d8aa40204587">00591</a> <span class="keyword">typedef</span> shared_ptr&lt;SpawnManager&gt; <a class="code" href="namespacePassenger.html#6ab2d566ff87a2676c23d8aa40204587" title="Convenient alias for SpawnManager smart pointer.">SpawnManagerPtr</a>;
614
- <a name="l00592"></a>00592
615
- <a name="l00593"></a>00593 } <span class="comment">// namespace Passenger</span>
616
- <a name="l00594"></a>00594
617
- <a name="l00595"></a>00595 <span class="preprocessor">#endif </span><span class="comment">/* _PASSENGER_SPAWN_MANAGER_H_ */</span>
513
+ <a name="l00491"></a>00491
514
+ <a name="l00492"></a>00492 <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> prependMessageToException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;message) {
515
+ <a name="l00493"></a>00493 <span class="keywordflow">return</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a>(message + <span class="stringliteral">": "</span> + e.<a class="code" href="classPassenger_1_1IOException.html#1b65d0fdb9bc5136f5b7df759c14768a">what</a>());
516
+ <a name="l00494"></a>00494 }
517
+ <a name="l00495"></a>00495
518
+ <a name="l00496"></a>00496 <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> prependMessageToException(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e, <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;message) {
519
+ <a name="l00497"></a>00497 <span class="keywordflow">return</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a>(message + <span class="stringliteral">": "</span> + e.<a class="code" href="classPassenger_1_1SystemException.html#84fa2ab4f5b7b96704734fcdfdaa0269" title="Returns a brief version of the exception message.">brief</a>(), e.<a class="code" href="classPassenger_1_1SystemException.html#ee7a6672bf79b72a4c3ee70c57d6a47c" title="The value of errno at the time the error occured.">code</a>());
520
+ <a name="l00498"></a>00498 }
521
+ <a name="l00499"></a>00499
522
+ <a name="l00500"></a>00500 <span class="keyword">protected</span>:<span class="comment"></span>
523
+ <a name="l00501"></a>00501 <span class="comment"> /**</span>
524
+ <a name="l00502"></a>00502 <span class="comment"> * A method which is called after the spawn server has started.</span>
525
+ <a name="l00503"></a>00503 <span class="comment"> * It doesn't do anything by default and serves as a hook for unit tests.</span>
526
+ <a name="l00504"></a>00504 <span class="comment"> */</span>
527
+ <a name="l00505"></a><a class="code" href="classPassenger_1_1SpawnManager.html#b4dfcb11c58b07d8144b18dd99abed45">00505</a> <span class="keyword">virtual</span> <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1SpawnManager.html#b4dfcb11c58b07d8144b18dd99abed45" title="A method which is called after the spawn server has started.">spawnServerStarted</a>() { }
528
+ <a name="l00506"></a>00506
529
+ <a name="l00507"></a>00507 <span class="keyword">public</span>:<span class="comment"></span>
530
+ <a name="l00508"></a>00508 <span class="comment"> /**</span>
531
+ <a name="l00509"></a>00509 <span class="comment"> * Construct a new SpawnManager.</span>
532
+ <a name="l00510"></a>00510 <span class="comment"> *</span>
533
+ <a name="l00511"></a>00511 <span class="comment"> * @param spawnServerCommand The filename of the spawn server to use.</span>
534
+ <a name="l00512"></a>00512 <span class="comment"> * @param generation The server instance dir generation in which</span>
535
+ <a name="l00513"></a>00513 <span class="comment"> * generation-specific are stored.</span>
536
+ <a name="l00514"></a>00514 <span class="comment"> * @param accountsDatabase An accounts database. SpawnManager will automatically</span>
537
+ <a name="l00515"></a>00515 <span class="comment"> * create a new account for each spawned process, assigning</span>
538
+ <a name="l00516"></a>00516 <span class="comment"> * it the rights as set in the PoolOptions object. This</span>
539
+ <a name="l00517"></a>00517 <span class="comment"> * account is also automatically deleted when no longer needed.</span>
540
+ <a name="l00518"></a>00518 <span class="comment"> * May be a null pointer.</span>
541
+ <a name="l00519"></a>00519 <span class="comment"> * @param rubyCommand The Ruby interpreter's command.</span>
542
+ <a name="l00520"></a>00520 <span class="comment"> * @throws RuntimeException An error occurred while creating a Unix server socket.</span>
543
+ <a name="l00521"></a>00521 <span class="comment"> * @throws SystemException An error occured while trying to setup the spawn server.</span>
544
+ <a name="l00522"></a>00522 <span class="comment"> * @throws IOException An error occurred while generating random data.</span>
545
+ <a name="l00523"></a>00523 <span class="comment"> */</span>
546
+ <a name="l00524"></a><a class="code" href="classPassenger_1_1SpawnManager.html#9dcfe537ec598125fc83a1625ba50552">00524</a> <a class="code" href="classPassenger_1_1SpawnManager.html#9dcfe537ec598125fc83a1625ba50552" title="Construct a new SpawnManager.">SpawnManager</a>(<span class="keyword">const</span> <span class="keywordtype">string</span> &amp;spawnServerCommand,
547
+ <a name="l00525"></a>00525 <span class="keyword">const</span> ServerInstanceDir::GenerationPtr &amp;generation,
548
+ <a name="l00526"></a>00526 <span class="keyword">const</span> AccountsDatabasePtr &amp;accountsDatabase = AccountsDatabasePtr(),
549
+ <a name="l00527"></a>00527 <span class="keyword">const</span> <span class="keywordtype">string</span> &amp;rubyCommand = <span class="stringliteral">"ruby"</span>,
550
+ <a name="l00528"></a>00528 <span class="keyword">const</span> AnalyticsLoggerPtr &amp;analyticsLogger = AnalyticsLoggerPtr()
551
+ <a name="l00529"></a>00529 ) {
552
+ <a name="l00530"></a>00530 TRACE_POINT();
553
+ <a name="l00531"></a>00531 this-&gt;spawnServerCommand = spawnServerCommand;
554
+ <a name="l00532"></a>00532 this-&gt;generation = generation;
555
+ <a name="l00533"></a>00533 this-&gt;accountsDatabase = accountsDatabase;
556
+ <a name="l00534"></a>00534 this-&gt;rubyCommand = rubyCommand;
557
+ <a name="l00535"></a>00535 this-&gt;analyticsLogger = analyticsLogger;
558
+ <a name="l00536"></a>00536 pid = 0;
559
+ <a name="l00537"></a>00537 this_thread::disable_interruption di;
560
+ <a name="l00538"></a>00538 this_thread::disable_syscall_interruption dsi;
561
+ <a name="l00539"></a>00539 <span class="keywordflow">try</span> {
562
+ <a name="l00540"></a>00540 restartServer();
563
+ <a name="l00541"></a>00541 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1IOException.html" title="Represents an error that occured during an I/O operation.">IOException</a> &amp;e) {
564
+ <a name="l00542"></a>00542 <span class="keywordflow">throw</span> prependMessageToException(e, <span class="stringliteral">"Could not start the spawn server"</span>);
565
+ <a name="l00543"></a>00543 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
566
+ <a name="l00544"></a>00544 <span class="keywordflow">throw</span> prependMessageToException(e, <span class="stringliteral">"Could not start the spawn server"</span>);
567
+ <a name="l00545"></a>00545 }
568
+ <a name="l00546"></a>00546 }
569
+ <a name="l00547"></a>00547
570
+ <a name="l00548"></a>00548 <span class="keyword">virtual</span> ~<a class="code" href="classPassenger_1_1SpawnManager.html" title="An AbstractSpawnManager implementation.">SpawnManager</a>() {
571
+ <a name="l00549"></a>00549 TRACE_POINT();
572
+ <a name="l00550"></a>00550 <span class="keywordflow">if</span> (pid != 0) {
573
+ <a name="l00551"></a>00551 UPDATE_TRACE_POINT();
574
+ <a name="l00552"></a>00552 this_thread::disable_interruption di;
575
+ <a name="l00553"></a>00553 this_thread::disable_syscall_interruption dsi;
576
+ <a name="l00554"></a>00554 syscalls::unlink(socketFilename.c_str());
577
+ <a name="l00555"></a>00555 ownerSocket.<a class="code" href="classPassenger_1_1FileDescriptor.html#d7c1a99531181878cbab74a7400c5432" title="Close the underlying file descriptor.">close</a>();
578
+ <a name="l00556"></a>00556 syscalls::waitpid(pid, NULL, 0);
579
+ <a name="l00557"></a>00557 }
580
+ <a name="l00558"></a>00558 }
581
+ <a name="l00559"></a>00559
582
+ <a name="l00560"></a><a class="code" href="classPassenger_1_1SpawnManager.html#88a94ba4d41dab492cb1dc8a0f998973">00560</a> <span class="keyword">virtual</span> <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a> <a class="code" href="classPassenger_1_1SpawnManager.html#88a94ba4d41dab492cb1dc8a0f998973" title="Spawn a new application process.">spawn</a>(<span class="keyword">const</span> <a class="code" href="structPassenger_1_1PoolOptions.html" title="This struct encapsulates information for ApplicationPool::get() and for SpawnManager::spawn()...">PoolOptions</a> &amp;options) {
583
+ <a name="l00561"></a>00561 TRACE_POINT();
584
+ <a name="l00562"></a>00562 AnalyticsScopeLog scope(options.<a class="code" href="structPassenger_1_1PoolOptions.html#ce54185a9981f103a036a7ac3e768612" title="An analytics log object to log things to.">log</a>, <span class="stringliteral">"spawn app process"</span>);
585
+ <a name="l00563"></a>00563 <a class="code" href="namespacePassenger.html#ced1f248465136b313c6dec5fe6ca492" title="Convenient alias for Process smart pointer.">ProcessPtr</a> result;
586
+ <a name="l00564"></a>00564 boost::mutex::scoped_lock l(lock);
587
+ <a name="l00565"></a>00565
588
+ <a name="l00566"></a>00566 <span class="keywordflow">try</span> {
589
+ <a name="l00567"></a>00567 result = sendSpawnCommand(options);
590
+ <a name="l00568"></a>00568 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SpawnException.html" title="Thrown when SpawnManager or ApplicationPool fails to spawn an application instance...">SpawnException</a> &amp;e) {
591
+ <a name="l00569"></a>00569 <span class="keywordflow">if</span> (e.<a class="code" href="classPassenger_1_1SpawnException.html#e65dc272e183fd9e5637a4091cc6bbf4" title="Check whether an error page is available.">hasErrorPage</a>()) {
592
+ <a name="l00570"></a>00570 <span class="keywordflow">throw</span>;
593
+ <a name="l00571"></a>00571 } <span class="keywordflow">else</span> {
594
+ <a name="l00572"></a>00572 result = handleSpawnException(e, options);
595
+ <a name="l00573"></a>00573 }
596
+ <a name="l00574"></a>00574 }
597
+ <a name="l00575"></a>00575 scope.success();
598
+ <a name="l00576"></a>00576 <span class="keywordflow">return</span> result;
599
+ <a name="l00577"></a>00577 }
600
+ <a name="l00578"></a>00578
601
+ <a name="l00579"></a><a class="code" href="classPassenger_1_1SpawnManager.html#ad71b2ea9431f3286f23a6e6f80e0981">00579</a> <span class="keyword">virtual</span> <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1SpawnManager.html#ad71b2ea9431f3286f23a6e6f80e0981" title="Shutdown the ApplicationSpawner server that&amp;#39;s running at the given application...">reload</a>(<span class="keyword">const</span> <span class="keywordtype">string</span> &amp;appRoot) {
602
+ <a name="l00580"></a>00580 TRACE_POINT();
603
+ <a name="l00581"></a>00581 this_thread::disable_interruption di;
604
+ <a name="l00582"></a>00582 this_thread::disable_syscall_interruption dsi;
605
+ <a name="l00583"></a>00583 <span class="keywordflow">try</span> {
606
+ <a name="l00584"></a>00584 <span class="keywordflow">return</span> sendReloadCommand(appRoot);
607
+ <a name="l00585"></a>00585 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> <a class="code" href="classPassenger_1_1SystemException.html" title="Represents an error returned by a system call or a standard library call.">SystemException</a> &amp;e) {
608
+ <a name="l00586"></a>00586 <span class="keywordflow">return</span> handleReloadException(e, appRoot);
609
+ <a name="l00587"></a>00587 }
610
+ <a name="l00588"></a>00588 }
611
+ <a name="l00589"></a>00589
612
+ <a name="l00590"></a><a class="code" href="classPassenger_1_1SpawnManager.html#9210212c23c5a8ee5ee4dd5c9dcdf57d">00590</a> <span class="keyword">virtual</span> <span class="keywordtype">void</span> <a class="code" href="classPassenger_1_1SpawnManager.html#9210212c23c5a8ee5ee4dd5c9dcdf57d" title="Forcefully kill the spawn server.">killSpawnServer</a>()<span class="keyword"> const </span>{
613
+ <a name="l00591"></a>00591 kill(pid, SIGKILL);
614
+ <a name="l00592"></a>00592 }
615
+ <a name="l00593"></a>00593
616
+ <a name="l00594"></a><a class="code" href="classPassenger_1_1SpawnManager.html#f34df8cfccfc64507999f1b48bc5a126">00594</a> <span class="keyword">virtual</span> pid_t <a class="code" href="classPassenger_1_1SpawnManager.html#f34df8cfccfc64507999f1b48bc5a126" title="Returns the spawn server&amp;#39;s PID.">getServerPid</a>()<span class="keyword"> const </span>{
617
+ <a name="l00595"></a>00595 <span class="keywordflow">return</span> pid;
618
+ <a name="l00596"></a>00596 }
619
+ <a name="l00597"></a>00597 };
620
+ <a name="l00598"></a>00598 <span class="comment"></span>
621
+ <a name="l00599"></a>00599 <span class="comment">/** Convenient alias for SpawnManager smart pointer. */</span>
622
+ <a name="l00600"></a><a class="code" href="namespacePassenger.html#6ab2d566ff87a2676c23d8aa40204587">00600</a> <span class="keyword">typedef</span> shared_ptr&lt;SpawnManager&gt; <a class="code" href="namespacePassenger.html#6ab2d566ff87a2676c23d8aa40204587" title="Convenient alias for SpawnManager smart pointer.">SpawnManagerPtr</a>;
623
+ <a name="l00601"></a>00601
624
+ <a name="l00602"></a>00602 } <span class="comment">// namespace Passenger</span>
625
+ <a name="l00603"></a>00603
626
+ <a name="l00604"></a>00604 <span class="preprocessor">#endif </span><span class="comment">/* _PASSENGER_SPAWN_MANAGER_H_ */</span>
618
627
  </pre></div></div>
619
- <hr size="1"><address style="text-align: right;"><small>Generated on Sun Feb 21 12:22:46 2010 for Passenger by&nbsp;
628
+ <hr size="1"><address style="text-align: right;"><small>Generated on Sun Mar 28 14:11:56 2010 for Passenger by&nbsp;
620
629
  <a href="http://www.doxygen.org/index.html">
621
630
  <img src="doxygen.png" alt="doxygen" align="middle" border="0"></a> 1.5.8 </small></address>
622
631
  </body>