passenger 2.2.9 → 2.2.10
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.
- data/NEWS +10 -0
- data/doc/Architectural overview.html +202 -65
- data/doc/Security of user switching support.html +200 -63
- data/doc/Users guide Apache.html +266 -129
- data/doc/Users guide Nginx.html +234 -97
- data/doc/cxxapi/AbstractSpawnManager_8h-source.html +138 -0
- data/doc/cxxapi/Account_8h-source.html +202 -0
- data/doc/cxxapi/AccountsDatabase_8h-source.html +158 -0
- data/doc/cxxapi/AgentsStarter_8h-source.html +79 -0
- data/doc/cxxapi/BCrypt_8h-source.html +101 -0
- data/doc/cxxapi/BacktracesServer_8h-source.html +87 -0
- data/doc/cxxapi/Base64_8h-source.html +85 -0
- data/doc/cxxapi/Blowfish_8h-source.html +131 -0
- data/doc/cxxapi/Bucket_8h-source.html +1 -1
- data/doc/cxxapi/CachedFileStat_8h-source.html +49 -157
- data/doc/cxxapi/Configuration_8h-source.html +1 -1
- data/doc/cxxapi/Constants_8h-source.html +71 -0
- data/doc/cxxapi/ContentHandler_8h-source.html +92 -0
- data/doc/cxxapi/DirectoryMapper_8h-source.html +1 -1
- data/doc/cxxapi/Exceptions_8h-source.html +332 -236
- data/doc/cxxapi/FileChangeChecker_8h-source.html +237 -0
- data/doc/cxxapi/FileDescriptor_8h-source.html +206 -0
- data/doc/cxxapi/Hooks_8h-source.html +1 -1
- data/doc/cxxapi/HttpStatusExtractor_8h-source.html +378 -0
- data/doc/cxxapi/LoggingServer_8h-source.html +165 -0
- data/doc/cxxapi/Logging_8h-source.html +580 -112
- data/doc/cxxapi/MessageChannel_8h-source.html +789 -599
- data/doc/cxxapi/MessageClient_8h-source.html +297 -0
- data/doc/cxxapi/MessageServer_8h-source.html +607 -0
- data/doc/cxxapi/PoolOptions_8h-source.html +471 -278
- data/doc/cxxapi/Process_8h-source.html +272 -0
- data/doc/cxxapi/RandomGenerator_8h-source.html +156 -0
- data/doc/cxxapi/ResourceLocator_8h-source.html +104 -0
- data/doc/cxxapi/ScgiRequestParser_8h-source.html +399 -0
- data/doc/cxxapi/ServerInstanceDir_8h-source.html +356 -0
- data/doc/cxxapi/Session_8h-source.html +483 -0
- data/doc/cxxapi/SpawnManager_8h-source.html +590 -546
- data/doc/cxxapi/StaticContentHandler_8h-source.html +65 -0
- data/doc/cxxapi/StaticString_8h-source.html +214 -0
- data/doc/cxxapi/StringListCreator_8h-source.html +111 -0
- data/doc/cxxapi/SystemTime_8h-source.html +191 -78
- data/doc/cxxapi/Timer_8h-source.html +157 -0
- data/doc/cxxapi/Utils_8h-source.html +621 -434
- data/doc/cxxapi/Version_8h-source.html +59 -0
- data/doc/cxxapi/annotated.html +1 -1
- data/doc/cxxapi/apache2_2Configuration_8h-source.html +79 -0
- data/doc/cxxapi/classAgentWatcher-members.html +50 -0
- data/doc/cxxapi/classAgentWatcher.html +445 -0
- data/doc/cxxapi/classClient-members.html +4 -3
- data/doc/cxxapi/classClient.html +50 -46
- data/doc/cxxapi/classHooks-members.html +1 -1
- data/doc/cxxapi/classHooks.html +1 -1
- data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager-members.html +37 -0
- data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager.html +191 -0
- data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1AbstractSpawnManager__inherit__graph.png +0 -0
- data/doc/cxxapi/{classPassenger_1_1DummySpawnManager-members.html → classPassenger_1_1ArgumentException-members.html} +3 -2
- data/doc/cxxapi/classPassenger_1_1ArgumentException.html +52 -0
- data/doc/cxxapi/classPassenger_1_1BacktracesServer-members.html +36 -0
- data/doc/cxxapi/classPassenger_1_1BacktracesServer.html +107 -0
- data/doc/cxxapi/classPassenger_1_1BacktracesServer__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1BacktracesServer__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1BacktracesServer__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1BufferedUpload-members.html +3 -3
- data/doc/cxxapi/classPassenger_1_1BufferedUpload.html +18 -34
- data/doc/cxxapi/classPassenger_1_1BusyException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1BusyException.html +3 -2
- data/doc/cxxapi/classPassenger_1_1ConfigurationException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1ConfigurationException.html +3 -2
- data/doc/cxxapi/classPassenger_1_1DirectoryMapper-members.html +1 -1
- data/doc/cxxapi/classPassenger_1_1DirectoryMapper.html +1 -1
- data/doc/cxxapi/{classPassenger_1_1FileChecker-members.html → classPassenger_1_1EOFException-members.html} +3 -4
- data/doc/cxxapi/classPassenger_1_1EOFException.html +57 -0
- data/doc/cxxapi/classPassenger_1_1EOFException__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1EOFException__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1EOFException__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1EventFd-members.html +33 -0
- data/doc/cxxapi/classPassenger_1_1EventFd.html +49 -0
- data/doc/cxxapi/classPassenger_1_1FileChangeChecker-members.html +37 -0
- data/doc/cxxapi/classPassenger_1_1FileChangeChecker.html +166 -0
- data/doc/cxxapi/classPassenger_1_1FileDescriptor-members.html +37 -0
- data/doc/cxxapi/classPassenger_1_1FileDescriptor.html +159 -0
- data/doc/cxxapi/classPassenger_1_1FileNotFoundException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1FileNotFoundException.html +3 -2
- data/doc/cxxapi/classPassenger_1_1FileSystemException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1FileSystemException.html +3 -2
- data/doc/cxxapi/{classPassenger_1_1ApplicationPoolServer-members.html → classPassenger_1_1HttpStatusExtractor-members.html} +6 -5
- data/doc/cxxapi/classPassenger_1_1HttpStatusExtractor.html +128 -0
- data/doc/cxxapi/classPassenger_1_1IOException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1IOException.html +4 -3
- data/doc/cxxapi/classPassenger_1_1IOException__inherit__graph.map +2 -1
- data/doc/cxxapi/classPassenger_1_1IOException__inherit__graph.md5 +1 -1
- data/doc/cxxapi/classPassenger_1_1IOException__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1MemZeroGuard-members.html +36 -0
- data/doc/cxxapi/classPassenger_1_1MemZeroGuard.html +147 -0
- data/doc/cxxapi/classPassenger_1_1MessageChannel-members.html +12 -5
- data/doc/cxxapi/classPassenger_1_1MessageChannel.html +267 -34
- data/doc/cxxapi/classPassenger_1_1MessageServer-members.html +46 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer.html +438 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext-members.html +33 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext.html +56 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1ClientContext__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext-members.html +38 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext.html +175 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1CommonClientContext__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler-members.html +36 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler.html +176 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1MessageServer_1_1Handler__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1Process-members.html +40 -0
- data/doc/cxxapi/classPassenger_1_1Process.html +266 -0
- data/doc/cxxapi/classPassenger_1_1RandomGenerator-members.html +33 -0
- data/doc/cxxapi/classPassenger_1_1RandomGenerator.html +51 -0
- data/doc/cxxapi/classPassenger_1_1RuntimeException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1RuntimeException.html +3 -2
- data/doc/cxxapi/classPassenger_1_1ScgiRequestParser-members.html +47 -0
- data/doc/cxxapi/classPassenger_1_1ScgiRequestParser.html +302 -0
- data/doc/cxxapi/classPassenger_1_1SecurityException-members.html +33 -0
- data/doc/cxxapi/classPassenger_1_1SecurityException.html +52 -0
- data/doc/cxxapi/classPassenger_1_1Session-members.html +50 -0
- data/doc/cxxapi/classPassenger_1_1Session.html +577 -0
- data/doc/cxxapi/classPassenger_1_1Session__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1Session__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1Session__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1SpawnException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1SpawnException.html +5 -4
- data/doc/cxxapi/classPassenger_1_1SpawnManager-members.html +8 -5
- data/doc/cxxapi/classPassenger_1_1SpawnManager.html +127 -45
- data/doc/cxxapi/classPassenger_1_1SpawnManager__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1SpawnManager__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1SpawnManager__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1StandardSession-members.html +51 -0
- data/doc/cxxapi/classPassenger_1_1StandardSession.html +415 -0
- data/doc/cxxapi/classPassenger_1_1StandardSession__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1StandardSession__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1StandardSession__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1StaticString-members.html +33 -0
- data/doc/cxxapi/classPassenger_1_1StaticString.html +54 -0
- data/doc/cxxapi/classPassenger_1_1SystemException-members.html +2 -1
- data/doc/cxxapi/classPassenger_1_1SystemException.html +4 -3
- data/doc/cxxapi/classPassenger_1_1SystemException__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1SystemException__inherit__graph.md5 +1 -1
- data/doc/cxxapi/classPassenger_1_1SystemException__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1SystemTime-members.html +9 -1
- data/doc/cxxapi/classPassenger_1_1SystemTime.html +92 -8
- data/doc/cxxapi/classPassenger_1_1TimeRetrievalException-members.html +37 -0
- data/doc/cxxapi/classPassenger_1_1TimeRetrievalException.html +57 -0
- data/doc/cxxapi/classPassenger_1_1TimeRetrievalException__inherit__graph.map +1 -0
- data/doc/cxxapi/classPassenger_1_1TimeRetrievalException__inherit__graph.md5 +1 -0
- data/doc/cxxapi/classPassenger_1_1TimeRetrievalException__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1TimeoutException-members.html +33 -0
- data/doc/cxxapi/classPassenger_1_1TimeoutException.html +50 -0
- data/doc/cxxapi/{classPassenger_1_1Application-members.html → classPassenger_1_1Timer-members.html} +8 -7
- data/doc/cxxapi/classPassenger_1_1Timer.html +163 -0
- data/doc/cxxapi/classServer-members.html +33 -0
- data/doc/cxxapi/classServer.html +46 -0
- data/doc/cxxapi/classServerInstanceDirToucher-members.html +33 -0
- data/doc/cxxapi/classServerInstanceDirToucher.html +43 -0
- data/doc/cxxapi/classes.html +1 -1
- data/doc/cxxapi/definitions_8h-source.html +1 -1
- data/doc/cxxapi/files.html +1 -1
- data/doc/cxxapi/functions.html +1 -1
- data/doc/cxxapi/{functions_type.html → functions_enum.html} +7 -5
- data/doc/cxxapi/functions_eval.html +55 -0
- data/doc/cxxapi/functions_func.html +1 -1
- data/doc/cxxapi/functions_vars.html +106 -9
- data/doc/cxxapi/graph_legend.html +1 -1
- data/doc/cxxapi/group__Configuration.html +1 -1
- data/doc/cxxapi/group__Core.html +1 -1
- data/doc/cxxapi/group__Exceptions.html +17 -1
- data/doc/cxxapi/group__Hooks.html +1 -1
- data/doc/cxxapi/group__Support.html +1 -1
- data/doc/cxxapi/hierarchy.html +34 -12
- data/doc/cxxapi/inherit__graph__0.map +1 -1
- data/doc/cxxapi/inherit__graph__0.md5 +1 -1
- data/doc/cxxapi/inherit__graph__0.png +0 -0
- data/doc/cxxapi/inherit__graph__1.map +1 -1
- data/doc/cxxapi/inherit__graph__1.md5 +1 -1
- data/doc/cxxapi/inherit__graph__1.png +0 -0
- data/doc/cxxapi/inherit__graph__10.map +1 -1
- data/doc/cxxapi/inherit__graph__10.md5 +1 -1
- data/doc/cxxapi/inherit__graph__10.png +0 -0
- data/doc/cxxapi/inherit__graph__11.map +1 -1
- data/doc/cxxapi/inherit__graph__11.md5 +1 -1
- data/doc/cxxapi/inherit__graph__11.png +0 -0
- data/doc/cxxapi/inherit__graph__12.map +1 -1
- data/doc/cxxapi/inherit__graph__12.md5 +1 -1
- data/doc/cxxapi/inherit__graph__12.png +0 -0
- data/doc/cxxapi/inherit__graph__13.map +1 -1
- data/doc/cxxapi/inherit__graph__13.md5 +1 -1
- data/doc/cxxapi/inherit__graph__13.png +0 -0
- data/doc/cxxapi/inherit__graph__14.map +3 -2
- data/doc/cxxapi/inherit__graph__14.md5 +1 -1
- data/doc/cxxapi/inherit__graph__14.png +0 -0
- data/doc/cxxapi/inherit__graph__15.map +1 -1
- data/doc/cxxapi/inherit__graph__15.md5 +1 -1
- data/doc/cxxapi/inherit__graph__15.png +0 -0
- data/doc/cxxapi/inherit__graph__16.map +1 -1
- data/doc/cxxapi/inherit__graph__16.md5 +1 -1
- data/doc/cxxapi/inherit__graph__16.png +0 -0
- data/doc/cxxapi/inherit__graph__17.map +1 -1
- data/doc/cxxapi/inherit__graph__17.md5 +1 -1
- data/doc/cxxapi/inherit__graph__17.png +0 -0
- data/doc/cxxapi/inherit__graph__18.map +2 -1
- data/doc/cxxapi/inherit__graph__18.md5 +1 -1
- data/doc/cxxapi/inherit__graph__18.png +0 -0
- data/doc/cxxapi/inherit__graph__19.map +1 -1
- data/doc/cxxapi/inherit__graph__19.md5 +1 -1
- data/doc/cxxapi/inherit__graph__19.png +0 -0
- data/doc/cxxapi/inherit__graph__2.map +1 -1
- data/doc/cxxapi/inherit__graph__2.md5 +1 -1
- data/doc/cxxapi/inherit__graph__2.png +0 -0
- data/doc/cxxapi/inherit__graph__20.map +1 -2
- data/doc/cxxapi/inherit__graph__20.md5 +1 -1
- data/doc/cxxapi/inherit__graph__20.png +0 -0
- data/doc/cxxapi/inherit__graph__21.map +1 -1
- data/doc/cxxapi/inherit__graph__21.md5 +1 -1
- data/doc/cxxapi/inherit__graph__21.png +0 -0
- data/doc/cxxapi/inherit__graph__22.map +1 -0
- data/doc/cxxapi/inherit__graph__22.md5 +1 -0
- data/doc/cxxapi/inherit__graph__22.png +0 -0
- data/doc/cxxapi/inherit__graph__23.map +1 -0
- data/doc/cxxapi/inherit__graph__23.md5 +1 -0
- data/doc/cxxapi/inherit__graph__23.png +0 -0
- data/doc/cxxapi/inherit__graph__24.map +1 -0
- data/doc/cxxapi/inherit__graph__24.md5 +1 -0
- data/doc/cxxapi/inherit__graph__24.png +0 -0
- data/doc/cxxapi/inherit__graph__25.map +1 -0
- data/doc/cxxapi/inherit__graph__25.md5 +1 -0
- data/doc/cxxapi/inherit__graph__25.png +0 -0
- data/doc/cxxapi/inherit__graph__26.map +1 -0
- data/doc/cxxapi/inherit__graph__26.md5 +1 -0
- data/doc/cxxapi/inherit__graph__26.png +0 -0
- data/doc/cxxapi/inherit__graph__27.map +2 -0
- data/doc/cxxapi/inherit__graph__27.md5 +1 -0
- data/doc/cxxapi/inherit__graph__27.png +0 -0
- data/doc/cxxapi/inherit__graph__28.map +1 -0
- data/doc/cxxapi/inherit__graph__28.md5 +1 -0
- data/doc/cxxapi/inherit__graph__28.png +0 -0
- data/doc/cxxapi/inherit__graph__29.map +1 -0
- data/doc/cxxapi/inherit__graph__29.md5 +1 -0
- data/doc/cxxapi/inherit__graph__29.png +0 -0
- data/doc/cxxapi/inherit__graph__3.map +2 -1
- data/doc/cxxapi/inherit__graph__3.md5 +1 -1
- data/doc/cxxapi/inherit__graph__3.png +0 -0
- data/doc/cxxapi/inherit__graph__30.map +1 -0
- data/doc/cxxapi/inherit__graph__30.md5 +1 -0
- data/doc/cxxapi/inherit__graph__30.png +0 -0
- data/doc/cxxapi/inherit__graph__31.map +3 -0
- data/doc/cxxapi/inherit__graph__31.md5 +1 -0
- data/doc/cxxapi/inherit__graph__31.png +0 -0
- data/doc/cxxapi/inherit__graph__32.map +1 -0
- data/doc/cxxapi/inherit__graph__32.md5 +1 -0
- data/doc/cxxapi/inherit__graph__32.png +0 -0
- data/doc/cxxapi/inherit__graph__33.map +1 -0
- data/doc/cxxapi/inherit__graph__33.md5 +1 -0
- data/doc/cxxapi/inherit__graph__33.png +0 -0
- data/doc/cxxapi/inherit__graph__34.map +1 -0
- data/doc/cxxapi/inherit__graph__34.md5 +1 -0
- data/doc/cxxapi/inherit__graph__34.png +0 -0
- data/doc/cxxapi/inherit__graph__35.map +1 -0
- data/doc/cxxapi/inherit__graph__35.md5 +1 -0
- data/doc/cxxapi/inherit__graph__35.png +0 -0
- data/doc/cxxapi/inherit__graph__36.map +1 -0
- data/doc/cxxapi/inherit__graph__36.md5 +1 -0
- data/doc/cxxapi/inherit__graph__36.png +0 -0
- data/doc/cxxapi/inherit__graph__4.map +1 -1
- data/doc/cxxapi/inherit__graph__4.md5 +1 -1
- data/doc/cxxapi/inherit__graph__4.png +0 -0
- data/doc/cxxapi/inherit__graph__5.map +1 -1
- data/doc/cxxapi/inherit__graph__5.md5 +1 -1
- data/doc/cxxapi/inherit__graph__5.png +0 -0
- data/doc/cxxapi/inherit__graph__6.map +1 -2
- data/doc/cxxapi/inherit__graph__6.md5 +1 -1
- data/doc/cxxapi/inherit__graph__6.png +0 -0
- data/doc/cxxapi/inherit__graph__7.map +1 -1
- data/doc/cxxapi/inherit__graph__7.md5 +1 -1
- data/doc/cxxapi/inherit__graph__7.png +0 -0
- data/doc/cxxapi/inherit__graph__8.map +1 -1
- data/doc/cxxapi/inherit__graph__8.md5 +1 -1
- data/doc/cxxapi/inherit__graph__8.png +0 -0
- data/doc/cxxapi/inherit__graph__9.map +1 -1
- data/doc/cxxapi/inherit__graph__9.md5 +1 -1
- data/doc/cxxapi/inherit__graph__9.png +0 -0
- data/doc/cxxapi/inherits.html +81 -41
- data/doc/cxxapi/main.html +1 -1
- data/doc/cxxapi/md5_8h-source.html +121 -0
- data/doc/cxxapi/modules.html +1 -1
- data/doc/cxxapi/namespacePassenger.html +651 -0
- data/doc/cxxapi/namespacemembers.html +155 -0
- data/doc/cxxapi/namespacemembers_enum.html +46 -0
- data/doc/cxxapi/namespacemembers_eval.html +52 -0
- data/doc/cxxapi/namespacemembers_func.html +139 -0
- data/doc/cxxapi/namespacemembers_type.html +50 -0
- data/doc/cxxapi/namespaces.html +34 -0
- data/doc/cxxapi/nginx_2Configuration_8h-source.html +114 -0
- data/doc/cxxapi/ngx__http__passenger__module_8h-source.html +91 -0
- data/doc/cxxapi/structPassenger_1_1AnythingToString-members.html +2 -1
- data/doc/cxxapi/structPassenger_1_1AnythingToString.html +3 -2
- data/doc/cxxapi/structPassenger_1_1AnythingToString_3_01vector_3_01string_01_4_01_4-members.html +2 -1
- data/doc/cxxapi/structPassenger_1_1AnythingToString_3_01vector_3_01string_01_4_01_4.html +3 -2
- data/doc/cxxapi/structPassenger_1_1MessageServer_1_1DisconnectEventBroadcastGuard-members.html +33 -0
- data/doc/cxxapi/structPassenger_1_1MessageServer_1_1DisconnectEventBroadcastGuard.html +49 -0
- data/doc/cxxapi/structPassenger_1_1PoolOptions-members.html +19 -7
- data/doc/cxxapi/structPassenger_1_1PoolOptions.html +323 -53
- data/doc/cxxapi/structPassenger_1_1StaticString_1_1Hash-members.html +33 -0
- data/doc/cxxapi/structPassenger_1_1StaticString_1_1Hash.html +49 -0
- data/ext/common/MessageChannel.h +48 -4
- data/ext/common/Version.h +1 -1
- data/ext/nginx/HttpStatusExtractor.h +1 -0
- data/ext/nginx/ScgiRequestParser.h +1 -0
- data/ext/phusion_passenger/native_support.c +5 -1
- data/lib/phusion_passenger/constants.rb +1 -1
- data/lib/phusion_passenger/message_channel.rb +45 -3
- data/lib/phusion_passenger/rack/application_spawner.rb +4 -0
- data/lib/phusion_passenger/railz/application_spawner.rb +19 -1
- data/lib/phusion_passenger/utils.rb +3 -3
- data/lib/phusion_passenger/utils/rewindable_input.rb +5 -0
- data/test/MessageChannelTest.cpp +9 -1
- data/test/stub/message_channel.rb +1 -1
- data/test/stub/message_channel_2.rb +1 -1
- data/test/stub/message_channel_3.rb +2 -2
- metadata +173 -113
- data/doc/cxxapi/ApplicationPoolServer_8h-source.html +0 -833
- data/doc/cxxapi/ApplicationPool_8h-source.html +0 -226
- data/doc/cxxapi/Application_8h-source.html +0 -553
- data/doc/cxxapi/DummySpawnManager_8h-source.html +0 -130
- data/doc/cxxapi/FileChecker_8h-source.html +0 -134
- data/doc/cxxapi/StandardApplicationPool_8h-source.html +0 -842
- data/doc/cxxapi/classPassenger_1_1Application.html +0 -231
- data/doc/cxxapi/classPassenger_1_1ApplicationPool-members.html +0 -41
- data/doc/cxxapi/classPassenger_1_1ApplicationPool.html +0 -323
- data/doc/cxxapi/classPassenger_1_1ApplicationPoolServer.html +0 -227
- data/doc/cxxapi/classPassenger_1_1ApplicationPool__inherit__graph.map +0 -1
- data/doc/cxxapi/classPassenger_1_1ApplicationPool__inherit__graph.md5 +0 -1
- data/doc/cxxapi/classPassenger_1_1ApplicationPool__inherit__graph.png +0 -0
- data/doc/cxxapi/classPassenger_1_1Application_1_1Session-members.html +0 -44
- data/doc/cxxapi/classPassenger_1_1Application_1_1Session.html +0 -403
- data/doc/cxxapi/classPassenger_1_1DummySpawnManager.html +0 -53
- data/doc/cxxapi/classPassenger_1_1FileChecker.html +0 -124
- data/doc/cxxapi/classPassenger_1_1StandardApplicationPool-members.html +0 -44
- data/doc/cxxapi/classPassenger_1_1StandardApplicationPool.html +0 -342
- data/doc/cxxapi/classPassenger_1_1StandardApplicationPool__inherit__graph.map +0 -1
- data/doc/cxxapi/classPassenger_1_1StandardApplicationPool__inherit__graph.md5 +0 -1
- data/doc/cxxapi/classPassenger_1_1StandardApplicationPool__inherit__graph.png +0 -0
- data/doc/rdoc/classes/ConditionVariable.html +0 -194
- data/doc/rdoc/classes/Exception.html +0 -120
- data/doc/rdoc/classes/GC.html +0 -113
- data/doc/rdoc/classes/IO.html +0 -169
- data/doc/rdoc/classes/PhusionPassenger.html +0 -239
- data/doc/rdoc/classes/PhusionPassenger/AbstractInstaller.html +0 -153
- data/doc/rdoc/classes/PhusionPassenger/AbstractRequestHandler.html +0 -517
- data/doc/rdoc/classes/PhusionPassenger/AbstractServer.html +0 -719
- data/doc/rdoc/classes/PhusionPassenger/AbstractServer/ServerAlreadyStarted.html +0 -97
- data/doc/rdoc/classes/PhusionPassenger/AbstractServer/ServerError.html +0 -96
- data/doc/rdoc/classes/PhusionPassenger/AbstractServer/ServerNotStarted.html +0 -97
- data/doc/rdoc/classes/PhusionPassenger/AbstractServer/UnknownMessage.html +0 -96
- data/doc/rdoc/classes/PhusionPassenger/AbstractServerCollection.html +0 -598
- data/doc/rdoc/classes/PhusionPassenger/AdminTools.html +0 -140
- data/doc/rdoc/classes/PhusionPassenger/AdminTools/ControlProcess.html +0 -317
- data/doc/rdoc/classes/PhusionPassenger/AdminTools/ControlProcess/Instance.html +0 -138
- data/doc/rdoc/classes/PhusionPassenger/AppInitError.html +0 -154
- data/doc/rdoc/classes/PhusionPassenger/Application.html +0 -283
- data/doc/rdoc/classes/PhusionPassenger/ConsoleTextTemplate.html +0 -172
- data/doc/rdoc/classes/PhusionPassenger/FrameworkInitError.html +0 -145
- data/doc/rdoc/classes/PhusionPassenger/HTMLTemplate.html +0 -175
- data/doc/rdoc/classes/PhusionPassenger/InitializationError.html +0 -141
- data/doc/rdoc/classes/PhusionPassenger/InvalidPath.html +0 -92
- data/doc/rdoc/classes/PhusionPassenger/MessageChannel.html +0 -489
- data/doc/rdoc/classes/PhusionPassenger/NativeSupport.html +0 -350
- data/doc/rdoc/classes/PhusionPassenger/Rack.html +0 -91
- data/doc/rdoc/classes/PhusionPassenger/Rack/ApplicationSpawner.html +0 -188
- data/doc/rdoc/classes/PhusionPassenger/Rack/RequestHandler.html +0 -199
- data/doc/rdoc/classes/PhusionPassenger/Railz.html +0 -95
- data/doc/rdoc/classes/PhusionPassenger/Railz/ApplicationSpawner.html +0 -438
- data/doc/rdoc/classes/PhusionPassenger/Railz/ApplicationSpawner/Error.html +0 -98
- data/doc/rdoc/classes/PhusionPassenger/Railz/CGIFixed.html +0 -200
- data/doc/rdoc/classes/PhusionPassenger/Railz/FrameworkSpawner.html +0 -436
- data/doc/rdoc/classes/PhusionPassenger/Railz/FrameworkSpawner/Error.html +0 -98
- data/doc/rdoc/classes/PhusionPassenger/Railz/RequestHandler.html +0 -155
- data/doc/rdoc/classes/PhusionPassenger/SpawnManager.html +0 -402
- data/doc/rdoc/classes/PhusionPassenger/UnknownError.html +0 -125
- data/doc/rdoc/classes/PhusionPassenger/Utils.html +0 -805
- data/doc/rdoc/classes/PhusionPassenger/Utils/PseudoIO.html +0 -169
- data/doc/rdoc/classes/PhusionPassenger/Utils/RewindableInput.html +0 -247
- data/doc/rdoc/classes/PhusionPassenger/Utils/RewindableInput/Tempfile.html +0 -120
- data/doc/rdoc/classes/PhusionPassenger/VersionNotFound.html +0 -140
- data/doc/rdoc/classes/PhusionPassenger/WSGI.html +0 -89
- data/doc/rdoc/classes/PhusionPassenger/WSGI/ApplicationSpawner.html +0 -188
- data/doc/rdoc/classes/PlatformInfo.html +0 -866
- data/doc/rdoc/classes/RakeExtensions.html +0 -197
- data/doc/rdoc/classes/Signal.html +0 -139
- data/doc/rdoc/created.rid +0 -1
- data/doc/rdoc/files/DEVELOPERS_TXT.html +0 -255
- data/doc/rdoc/files/README.html +0 -157
- data/doc/rdoc/files/ext/phusion_passenger/native_support_c.html +0 -92
- data/doc/rdoc/files/lib/phusion_passenger/abstract_installer_rb.html +0 -129
- data/doc/rdoc/files/lib/phusion_passenger/abstract_request_handler_rb.html +0 -129
- data/doc/rdoc/files/lib/phusion_passenger/abstract_server_collection_rb.html +0 -126
- data/doc/rdoc/files/lib/phusion_passenger/abstract_server_rb.html +0 -128
- data/doc/rdoc/files/lib/phusion_passenger/admin_tools/control_process_rb.html +0 -130
- data/doc/rdoc/files/lib/phusion_passenger/admin_tools_rb.html +0 -122
- data/doc/rdoc/files/lib/phusion_passenger/application_rb.html +0 -127
- data/doc/rdoc/files/lib/phusion_passenger/console_text_template_rb.html +0 -126
- data/doc/rdoc/files/lib/phusion_passenger/constants_rb.html +0 -122
- data/doc/rdoc/files/lib/phusion_passenger/dependencies_rb.html +0 -134
- data/doc/rdoc/files/lib/phusion_passenger/events_rb.html +0 -122
- data/doc/rdoc/files/lib/phusion_passenger/exceptions_rb.html +0 -122
- data/doc/rdoc/files/lib/phusion_passenger/html_template_rb.html +0 -126
- data/doc/rdoc/files/lib/phusion_passenger/message_channel_rb.html +0 -120
- data/doc/rdoc/files/lib/phusion_passenger/packaging_rb.html +0 -122
- data/doc/rdoc/files/lib/phusion_passenger/platform_info_rb.html +0 -127
- data/doc/rdoc/files/lib/phusion_passenger/rack/application_spawner_rb.html +0 -133
- data/doc/rdoc/files/lib/phusion_passenger/rack/request_handler_rb.html +0 -125
- data/doc/rdoc/files/lib/phusion_passenger/railz/application_spawner_rb.html +0 -140
- data/doc/rdoc/files/lib/phusion_passenger/railz/cgi_fixed_rb.html +0 -126
- data/doc/rdoc/files/lib/phusion_passenger/railz/framework_spawner_rb.html +0 -145
- data/doc/rdoc/files/lib/phusion_passenger/railz/request_handler_rb.html +0 -125
- data/doc/rdoc/files/lib/phusion_passenger/simple_benchmarking_rb.html +0 -122
- data/doc/rdoc/files/lib/phusion_passenger/spawn_manager_rb.html +0 -159
- data/doc/rdoc/files/lib/phusion_passenger/utils/rewindable_input_rb.html +0 -100
- data/doc/rdoc/files/lib/phusion_passenger/utils_rb.html +0 -174
- data/doc/rdoc/files/lib/phusion_passenger/wsgi/application_spawner_rb.html +0 -129
- data/doc/rdoc/files/misc/rake/extensions_rb.html +0 -130
- data/doc/rdoc/fr_class_index.html +0 -93
- data/doc/rdoc/fr_file_index.html +0 -77
- data/doc/rdoc/fr_method_index.html +0 -212
- data/doc/rdoc/index.html +0 -26
- data/doc/rdoc/rdoc-style.css +0 -187
@@ -0,0 +1,399 @@
|
|
1
|
+
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
|
2
|
+
<html><head><meta http-equiv="Content-Type" content="text/html;charset=UTF-8">
|
3
|
+
<title>Passenger: ScgiRequestParser.h Source File</title>
|
4
|
+
<link href="tabs.css" rel="stylesheet" type="text/css">
|
5
|
+
<link href="doxygen.css" rel="stylesheet" type="text/css">
|
6
|
+
</head><body>
|
7
|
+
<!-- Generated by Doxygen 1.5.8 -->
|
8
|
+
<div class="navigation" id="top">
|
9
|
+
<div class="tabs">
|
10
|
+
<ul>
|
11
|
+
<li><a href="main.html"><span>Main Page</span></a></li>
|
12
|
+
<li><a href="modules.html"><span>Modules</span></a></li>
|
13
|
+
<li><a href="namespaces.html"><span>Namespaces</span></a></li>
|
14
|
+
<li><a href="annotated.html"><span>Classes</span></a></li>
|
15
|
+
<li class="current"><a href="files.html"><span>Files</span></a></li>
|
16
|
+
</ul>
|
17
|
+
</div>
|
18
|
+
<div class="tabs">
|
19
|
+
<ul>
|
20
|
+
<li><a href="files.html"><span>File List</span></a></li>
|
21
|
+
</ul>
|
22
|
+
</div>
|
23
|
+
<h1>ScgiRequestParser.h</h1><div class="fragment"><pre class="fragment"><a name="l00001"></a>00001 <span class="comment">/*</span>
|
24
|
+
<a name="l00002"></a>00002 <span class="comment"> * Phusion Passenger - http://www.modrails.com/</span>
|
25
|
+
<a name="l00003"></a>00003 <span class="comment"> * Copyright (c) 2010 Phusion</span>
|
26
|
+
<a name="l00004"></a>00004 <span class="comment"> *</span>
|
27
|
+
<a name="l00005"></a>00005 <span class="comment"> * "Phusion Passenger" is a trademark of Hongli Lai & Ninh Bui.</span>
|
28
|
+
<a name="l00006"></a>00006 <span class="comment"> *</span>
|
29
|
+
<a name="l00007"></a>00007 <span class="comment"> * Permission is hereby granted, free of charge, to any person obtaining a copy</span>
|
30
|
+
<a name="l00008"></a>00008 <span class="comment"> * of this software and associated documentation files (the "Software"), to deal</span>
|
31
|
+
<a name="l00009"></a>00009 <span class="comment"> * in the Software without restriction, including without limitation the rights</span>
|
32
|
+
<a name="l00010"></a>00010 <span class="comment"> * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell</span>
|
33
|
+
<a name="l00011"></a>00011 <span class="comment"> * copies of the Software, and to permit persons to whom the Software is</span>
|
34
|
+
<a name="l00012"></a>00012 <span class="comment"> * furnished to do so, subject to the following conditions:</span>
|
35
|
+
<a name="l00013"></a>00013 <span class="comment"> *</span>
|
36
|
+
<a name="l00014"></a>00014 <span class="comment"> * The above copyright notice and this permission notice shall be included in</span>
|
37
|
+
<a name="l00015"></a>00015 <span class="comment"> * all copies or substantial portions of the Software.</span>
|
38
|
+
<a name="l00016"></a>00016 <span class="comment"> *</span>
|
39
|
+
<a name="l00017"></a>00017 <span class="comment"> * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR</span>
|
40
|
+
<a name="l00018"></a>00018 <span class="comment"> * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,</span>
|
41
|
+
<a name="l00019"></a>00019 <span class="comment"> * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE</span>
|
42
|
+
<a name="l00020"></a>00020 <span class="comment"> * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER</span>
|
43
|
+
<a name="l00021"></a>00021 <span class="comment"> * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,</span>
|
44
|
+
<a name="l00022"></a>00022 <span class="comment"> * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN</span>
|
45
|
+
<a name="l00023"></a>00023 <span class="comment"> * THE SOFTWARE.</span>
|
46
|
+
<a name="l00024"></a>00024 <span class="comment"> */</span>
|
47
|
+
<a name="l00025"></a>00025
|
48
|
+
<a name="l00026"></a>00026 <span class="preprocessor">#include <google/dense_hash_map></span>
|
49
|
+
<a name="l00027"></a>00027
|
50
|
+
<a name="l00028"></a>00028 <span class="preprocessor">#include <string></span>
|
51
|
+
<a name="l00029"></a>00029 <span class="preprocessor">#include <map></span>
|
52
|
+
<a name="l00030"></a>00030
|
53
|
+
<a name="l00031"></a>00031 <span class="preprocessor">#include "StaticString.h"</span>
|
54
|
+
<a name="l00032"></a>00032
|
55
|
+
<a name="l00033"></a>00033 <span class="keyword">namespace </span>Passenger {
|
56
|
+
<a name="l00034"></a>00034
|
57
|
+
<a name="l00035"></a>00035 <span class="keyword">using namespace </span>std;
|
58
|
+
<a name="l00036"></a>00036 <span class="keyword">using namespace </span>google;
|
59
|
+
<a name="l00037"></a>00037 <span class="comment"></span>
|
60
|
+
<a name="l00038"></a>00038 <span class="comment">/**</span>
|
61
|
+
<a name="l00039"></a>00039 <span class="comment"> * A parser for SCGI requests. It parses the request header and ignores the</span>
|
62
|
+
<a name="l00040"></a>00040 <span class="comment"> * body data.</span>
|
63
|
+
<a name="l00041"></a>00041 <span class="comment"> *</span>
|
64
|
+
<a name="l00042"></a>00042 <span class="comment"> * You can use it by constructing a parser object, then feeding data to the</span>
|
65
|
+
<a name="l00043"></a>00043 <span class="comment"> * parser until it has reached a final state.</span>
|
66
|
+
<a name="l00044"></a>00044 <span class="comment"> *</span>
|
67
|
+
<a name="l00045"></a>00045 <span class="comment"> * Example:</span>
|
68
|
+
<a name="l00046"></a>00046 <span class="comment"> * @code</span>
|
69
|
+
<a name="l00047"></a>00047 <span class="comment"> * ScgiRequestParser parser;</span>
|
70
|
+
<a name="l00048"></a>00048 <span class="comment"> * char buf[1024 * 16];</span>
|
71
|
+
<a name="l00049"></a>00049 <span class="comment"> * ssize_t size;</span>
|
72
|
+
<a name="l00050"></a>00050 <span class="comment"> * unsigned in bytesAccepted;</span>
|
73
|
+
<a name="l00051"></a>00051 <span class="comment"> * </span>
|
74
|
+
<a name="l00052"></a>00052 <span class="comment"> * do {</span>
|
75
|
+
<a name="l00053"></a>00053 <span class="comment"> * size = read(fd, buf, sizeof(buf));</span>
|
76
|
+
<a name="l00054"></a>00054 <span class="comment"> * bytesAccepted = parser.feed(buf, size);</span>
|
77
|
+
<a name="l00055"></a>00055 <span class="comment"> * } while (parser.acceptingInput());</span>
|
78
|
+
<a name="l00056"></a>00056 <span class="comment"> * // Parser is done when its return value isn't equal to the input size.</span>
|
79
|
+
<a name="l00057"></a>00057 <span class="comment"> * </span>
|
80
|
+
<a name="l00058"></a>00058 <span class="comment"> * // Check whether a parse error occured.</span>
|
81
|
+
<a name="l00059"></a>00059 <span class="comment"> * if (parser.getState() == ScgiRequestParser::ERROR) {</span>
|
82
|
+
<a name="l00060"></a>00060 <span class="comment"> * bailOut();</span>
|
83
|
+
<a name="l00061"></a>00061 <span class="comment"> * } else {</span>
|
84
|
+
<a name="l00062"></a>00062 <span class="comment"> * // All good! Do something with the SCGI header that the parser parsed.</span>
|
85
|
+
<a name="l00063"></a>00063 <span class="comment"> * processHeader(parser.getHeaderData());</span>
|
86
|
+
<a name="l00064"></a>00064 <span class="comment"> * </span>
|
87
|
+
<a name="l00065"></a>00065 <span class="comment"> * // If the last buffer passed to the parser also contains body data,</span>
|
88
|
+
<a name="l00066"></a>00066 <span class="comment"> * // then the body data starts at 'buf + bytesAccepted'.</span>
|
89
|
+
<a name="l00067"></a>00067 <span class="comment"> * if (bytesAccepted < size) {</span>
|
90
|
+
<a name="l00068"></a>00068 <span class="comment"> * processBody(buf + bytesAccepted);</span>
|
91
|
+
<a name="l00069"></a>00069 <span class="comment"> * }</span>
|
92
|
+
<a name="l00070"></a>00070 <span class="comment"> * while (!end_of_file(fd)) {</span>
|
93
|
+
<a name="l00071"></a>00071 <span class="comment"> * ... read(...) ...</span>
|
94
|
+
<a name="l00072"></a>00072 <span class="comment"> * processBody(...);</span>
|
95
|
+
<a name="l00073"></a>00073 <span class="comment"> * }</span>
|
96
|
+
<a name="l00074"></a>00074 <span class="comment"> * }</span>
|
97
|
+
<a name="l00075"></a>00075 <span class="comment"> * @endcode</span>
|
98
|
+
<a name="l00076"></a>00076 <span class="comment"> *</span>
|
99
|
+
<a name="l00077"></a>00077 <span class="comment"> * Parser properties:</span>
|
100
|
+
<a name="l00078"></a>00078 <span class="comment"> * - A parser object can only process a single SCGI request. You must discard</span>
|
101
|
+
<a name="l00079"></a>00079 <span class="comment"> * the existing parser object and create a new one if you want to process</span>
|
102
|
+
<a name="l00080"></a>00080 <span class="comment"> * another SCGI request.</span>
|
103
|
+
<a name="l00081"></a>00081 <span class="comment"> * - This parser checks whether the header netstring is valid. It will enter</span>
|
104
|
+
<a name="l00082"></a>00082 <span class="comment"> * the error state if it encounters a parse error.</span>
|
105
|
+
<a name="l00083"></a>00083 <span class="comment"> * - However, this parser does not perform any validation of the actual header</span>
|
106
|
+
<a name="l00084"></a>00084 <span class="comment"> * contents. For example, it doesn't check that CONTENT_LENGTH is the first</span>
|
107
|
+
<a name="l00085"></a>00085 <span class="comment"> * header, or that the SCGI header is present.</span>
|
108
|
+
<a name="l00086"></a>00086 <span class="comment"> */</span>
|
109
|
+
<a name="l00087"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html">00087</a> <span class="keyword">class </span><a class="code" href="classPassenger_1_1ScgiRequestParser.html" title="A parser for SCGI requests.">ScgiRequestParser</a> {
|
110
|
+
<a name="l00088"></a>00088 <span class="keyword">public</span>:
|
111
|
+
<a name="l00089"></a>00089 <span class="keyword">enum</span> State {
|
112
|
+
<a name="l00090"></a>00090 READING_LENGTH_STRING,
|
113
|
+
<a name="l00091"></a>00091 READING_HEADER_DATA,
|
114
|
+
<a name="l00092"></a>00092 EXPECTING_COMMA,
|
115
|
+
<a name="l00093"></a>00093 DONE,
|
116
|
+
<a name="l00094"></a>00094 ERROR
|
117
|
+
<a name="l00095"></a>00095 };
|
118
|
+
<a name="l00096"></a>00096
|
119
|
+
<a name="l00097"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b">00097</a> <span class="keyword">enum</span> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b">ErrorReason</a> {
|
120
|
+
<a name="l00098"></a>00098 NONE,
|
121
|
+
<a name="l00099"></a>00099 <span class="comment"></span>
|
122
|
+
<a name="l00100"></a>00100 <span class="comment"> /** The length string is too large. */</span>
|
123
|
+
<a name="l00101"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b9ba3d5851af9ffae1e9202bc9f73ac5b">00101</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b9ba3d5851af9ffae1e9202bc9f73ac5b" title="The length string is too large.">LENGTH_STRING_TOO_LARGE</a>,
|
124
|
+
<a name="l00102"></a>00102 <span class="comment"></span>
|
125
|
+
<a name="l00103"></a>00103 <span class="comment"> /** The header is larger than the maxSize value provided to the constructor. */</span>
|
126
|
+
<a name="l00104"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b876574447523322b9d7b4921e490032d">00104</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b876574447523322b9d7b4921e490032d" title="The header is larger than the maxSize value provided to the constructor.">LIMIT_REACHED</a>,
|
127
|
+
<a name="l00105"></a>00105 <span class="comment"></span>
|
128
|
+
<a name="l00106"></a>00106 <span class="comment"> /** The length string contains an invalid character. */</span>
|
129
|
+
<a name="l00107"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bfe61dd339024beaed48f6eed6e6f3332">00107</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bfe61dd339024beaed48f6eed6e6f3332" title="The length string contains an invalid character.">INVALID_LENGTH_STRING</a>,
|
130
|
+
<a name="l00108"></a>00108 <span class="comment"></span>
|
131
|
+
<a name="l00109"></a>00109 <span class="comment"> /** A header terminator character (",") was expected, but some else</span>
|
132
|
+
<a name="l00110"></a>00110 <span class="comment"> * was encountered instead. */</span>
|
133
|
+
<a name="l00111"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b44078eaf2ae1ee592d65d122dbda4a44">00111</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b44078eaf2ae1ee592d65d122dbda4a44" title="A header terminator character (&quot;,&quot;) was expected, but some else was encountered...">HEADER_TERMINATOR_EXPECTED</a>,
|
134
|
+
<a name="l00112"></a>00112 <span class="comment"></span>
|
135
|
+
<a name="l00113"></a>00113 <span class="comment"> /** The header data itself contains errors. */</span>
|
136
|
+
<a name="l00114"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bc162062c109fe310d5923d12b6147add">00114</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bc162062c109fe310d5923d12b6147add" title="The header data itself contains errors.">INVALID_HEADER_DATA</a>
|
137
|
+
<a name="l00115"></a>00115 };
|
138
|
+
<a name="l00116"></a>00116
|
139
|
+
<a name="l00117"></a>00117 <span class="keyword">private</span>:
|
140
|
+
<a name="l00118"></a>00118 <span class="keyword">typedef</span> dense_hash_map<StaticString, StaticString, StaticString::Hash> HeaderMap;
|
141
|
+
<a name="l00119"></a>00119
|
142
|
+
<a name="l00120"></a>00120 <span class="keywordtype">unsigned</span> <span class="keywordtype">long</span> maxSize;
|
143
|
+
<a name="l00121"></a>00121
|
144
|
+
<a name="l00122"></a>00122 State state;
|
145
|
+
<a name="l00123"></a>00123 <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b">ErrorReason</a> errorReason;
|
146
|
+
<a name="l00124"></a>00124 <span class="keywordtype">char</span> lengthStringBuffer[<span class="keyword">sizeof</span>(<span class="stringliteral">"4294967296"</span>)];
|
147
|
+
<a name="l00125"></a>00125 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> lengthStringBufferSize;
|
148
|
+
<a name="l00126"></a>00126 <span class="keywordtype">unsigned</span> <span class="keywordtype">long</span> headerSize;
|
149
|
+
<a name="l00127"></a>00127 <span class="keywordtype">string</span> headerBuffer;
|
150
|
+
<a name="l00128"></a>00128 HeaderMap headers;
|
151
|
+
<a name="l00129"></a>00129
|
152
|
+
<a name="l00130"></a>00130 <span class="keyword">static</span> <span class="keyword">inline</span> <span class="keywordtype">bool</span> isDigit(<span class="keywordtype">char</span> byte) {
|
153
|
+
<a name="l00131"></a>00131 <span class="keywordflow">return</span> byte >= <span class="charliteral">'0'</span> && byte <= <span class="charliteral">'9'</span>;
|
154
|
+
<a name="l00132"></a>00132 }
|
155
|
+
<a name="l00133"></a>00133 <span class="comment"></span>
|
156
|
+
<a name="l00134"></a>00134 <span class="comment"> /**</span>
|
157
|
+
<a name="l00135"></a>00135 <span class="comment"> * Parse the given header data into key-value pairs.</span>
|
158
|
+
<a name="l00136"></a>00136 <span class="comment"> */</span>
|
159
|
+
<a name="l00137"></a>00137 <span class="keywordtype">bool</span> parseHeaderData(<span class="keyword">const</span> <span class="keywordtype">string</span> &data, HeaderMap &output) {
|
160
|
+
<a name="l00138"></a>00138 <span class="keywordtype">bool</span> isName = <span class="keyword">true</span>; <span class="comment">// Whether we're currently expecting a name or a value.</span>
|
161
|
+
<a name="l00139"></a>00139 <span class="keyword">const</span> <span class="keywordtype">char</span> *startOfString, *current, *end;
|
162
|
+
<a name="l00140"></a>00140 StaticString key, value;
|
163
|
+
<a name="l00141"></a>00141
|
164
|
+
<a name="l00142"></a>00142 <span class="keywordflow">if</span> (data.size() == 0) {
|
165
|
+
<a name="l00143"></a>00143 <span class="keywordflow">return</span> <span class="keyword">true</span>;
|
166
|
+
<a name="l00144"></a>00144 }
|
167
|
+
<a name="l00145"></a>00145
|
168
|
+
<a name="l00146"></a>00146 startOfString = data.c_str();
|
169
|
+
<a name="l00147"></a>00147 end = data.c_str() + data.size();
|
170
|
+
<a name="l00148"></a>00148
|
171
|
+
<a name="l00149"></a>00149 <span class="keywordflow">if</span> (*(end - 1) != <span class="charliteral">'\0'</span>) {
|
172
|
+
<a name="l00150"></a>00150 <span class="keywordflow">return</span> <span class="keyword">false</span>;
|
173
|
+
<a name="l00151"></a>00151 }
|
174
|
+
<a name="l00152"></a>00152
|
175
|
+
<a name="l00153"></a>00153 <span class="keywordflow">for</span> (current = data.c_str(); current != end; current++) {
|
176
|
+
<a name="l00154"></a>00154 <span class="keywordflow">if</span> (isName && *current == <span class="charliteral">'\0'</span>) {
|
177
|
+
<a name="l00155"></a>00155 key = StaticString(startOfString, current - startOfString);
|
178
|
+
<a name="l00156"></a>00156 startOfString = current + 1;
|
179
|
+
<a name="l00157"></a>00157 isName = <span class="keyword">false</span>;
|
180
|
+
<a name="l00158"></a>00158 } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (!isName && *current == <span class="charliteral">'\0'</span>) {
|
181
|
+
<a name="l00159"></a>00159 value = StaticString(startOfString, current - startOfString);
|
182
|
+
<a name="l00160"></a>00160 startOfString = current + 1;
|
183
|
+
<a name="l00161"></a>00161 isName = <span class="keyword">true</span>;
|
184
|
+
<a name="l00162"></a>00162
|
185
|
+
<a name="l00163"></a>00163 output[key] = value;
|
186
|
+
<a name="l00164"></a>00164 key = StaticString();
|
187
|
+
<a name="l00165"></a>00165 value = StaticString();
|
188
|
+
<a name="l00166"></a>00166 }
|
189
|
+
<a name="l00167"></a>00167 }
|
190
|
+
<a name="l00168"></a>00168
|
191
|
+
<a name="l00169"></a>00169 <span class="keywordflow">return</span> isName;
|
192
|
+
<a name="l00170"></a>00170 }
|
193
|
+
<a name="l00171"></a>00171 <span class="comment"></span>
|
194
|
+
<a name="l00172"></a>00172 <span class="comment"> /**</span>
|
195
|
+
<a name="l00173"></a>00173 <span class="comment"> * Process the given data, which contains header data and possibly</span>
|
196
|
+
<a name="l00174"></a>00174 <span class="comment"> * some body data as well.</span>
|
197
|
+
<a name="l00175"></a>00175 <span class="comment"> */</span>
|
198
|
+
<a name="l00176"></a>00176 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> readHeaderData(<span class="keyword">const</span> <span class="keywordtype">char</span> *data, <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> size) {
|
199
|
+
<a name="l00177"></a>00177 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> bytesToRead;
|
200
|
+
<a name="l00178"></a>00178
|
201
|
+
<a name="l00179"></a>00179 <span class="comment">// Calculate how many bytes of header data is left to be read.</span>
|
202
|
+
<a name="l00180"></a>00180 <span class="comment">// Do not read past the header data.</span>
|
203
|
+
<a name="l00181"></a>00181 <span class="keywordflow">if</span> (size < headerSize - headerBuffer.size()) {
|
204
|
+
<a name="l00182"></a>00182 bytesToRead = size;
|
205
|
+
<a name="l00183"></a>00183 } <span class="keywordflow">else</span> {
|
206
|
+
<a name="l00184"></a>00184 bytesToRead = headerSize - headerBuffer.size();
|
207
|
+
<a name="l00185"></a>00185 }
|
208
|
+
<a name="l00186"></a>00186 <span class="comment">// Append the newly received header data to the header data buffer.</span>
|
209
|
+
<a name="l00187"></a>00187 headerBuffer.append(data, bytesToRead);
|
210
|
+
<a name="l00188"></a>00188
|
211
|
+
<a name="l00189"></a>00189 <span class="keywordflow">if</span> (headerBuffer.size() == headerSize) {
|
212
|
+
<a name="l00190"></a>00190 <span class="comment">// We've received all header data. Now attempt to parse this.</span>
|
213
|
+
<a name="l00191"></a>00191 <span class="keywordflow">if</span> (bytesToRead < size) {
|
214
|
+
<a name="l00192"></a>00192 <span class="keywordflow">if</span> (data[bytesToRead] == <span class="charliteral">','</span>) {
|
215
|
+
<a name="l00193"></a>00193 <span class="keywordflow">if</span> (parseHeaderData(headerBuffer, headers)) {
|
216
|
+
<a name="l00194"></a>00194 state = DONE;
|
217
|
+
<a name="l00195"></a>00195 <span class="keywordflow">return</span> bytesToRead + 1;
|
218
|
+
<a name="l00196"></a>00196 } <span class="keywordflow">else</span> {
|
219
|
+
<a name="l00197"></a>00197 state = ERROR;
|
220
|
+
<a name="l00198"></a>00198 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bc162062c109fe310d5923d12b6147add" title="The header data itself contains errors.">INVALID_HEADER_DATA</a>;
|
221
|
+
<a name="l00199"></a>00199 <span class="keywordflow">return</span> bytesToRead;
|
222
|
+
<a name="l00200"></a>00200 }
|
223
|
+
<a name="l00201"></a>00201 } <span class="keywordflow">else</span> {
|
224
|
+
<a name="l00202"></a>00202 state = ERROR;
|
225
|
+
<a name="l00203"></a>00203 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b44078eaf2ae1ee592d65d122dbda4a44" title="A header terminator character (&quot;,&quot;) was expected, but some else was encountered...">HEADER_TERMINATOR_EXPECTED</a>;
|
226
|
+
<a name="l00204"></a>00204 <span class="keywordflow">return</span> bytesToRead;
|
227
|
+
<a name="l00205"></a>00205 }
|
228
|
+
<a name="l00206"></a>00206 } <span class="keywordflow">else</span> {
|
229
|
+
<a name="l00207"></a>00207 <span class="keywordflow">if</span> (parseHeaderData(headerBuffer, headers)) {
|
230
|
+
<a name="l00208"></a>00208 state = EXPECTING_COMMA;
|
231
|
+
<a name="l00209"></a>00209 } <span class="keywordflow">else</span> {
|
232
|
+
<a name="l00210"></a>00210 state = ERROR;
|
233
|
+
<a name="l00211"></a>00211 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bc162062c109fe310d5923d12b6147add" title="The header data itself contains errors.">INVALID_HEADER_DATA</a>;
|
234
|
+
<a name="l00212"></a>00212 }
|
235
|
+
<a name="l00213"></a>00213 <span class="keywordflow">return</span> bytesToRead;
|
236
|
+
<a name="l00214"></a>00214 }
|
237
|
+
<a name="l00215"></a>00215 } <span class="keywordflow">else</span> {
|
238
|
+
<a name="l00216"></a>00216 <span class="comment">// Not all header data has been received yet.</span>
|
239
|
+
<a name="l00217"></a>00217 <span class="keywordflow">return</span> bytesToRead;
|
240
|
+
<a name="l00218"></a>00218 }
|
241
|
+
<a name="l00219"></a>00219 }
|
242
|
+
<a name="l00220"></a>00220
|
243
|
+
<a name="l00221"></a>00221 <span class="keyword">public</span>:<span class="comment"></span>
|
244
|
+
<a name="l00222"></a>00222 <span class="comment"> /**</span>
|
245
|
+
<a name="l00223"></a>00223 <span class="comment"> * Create a new ScgiRequestParser, ready to parse a request.</span>
|
246
|
+
<a name="l00224"></a>00224 <span class="comment"> *</span>
|
247
|
+
<a name="l00225"></a>00225 <span class="comment"> * @param maxSize The maximum size that the SCGI data is allowed to</span>
|
248
|
+
<a name="l00226"></a>00226 <span class="comment"> * be, or 0 if no limit is desired.</span>
|
249
|
+
<a name="l00227"></a>00227 <span class="comment"> */</span>
|
250
|
+
<a name="l00228"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#1108e7ce9162289db8a5b14479325e7a">00228</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#1108e7ce9162289db8a5b14479325e7a" title="Create a new ScgiRequestParser, ready to parse a request.">ScgiRequestParser</a>(<span class="keywordtype">unsigned</span> <span class="keywordtype">long</span> maxSize = 0) {
|
251
|
+
<a name="l00229"></a>00229 this->maxSize = maxSize;
|
252
|
+
<a name="l00230"></a>00230 state = READING_LENGTH_STRING;
|
253
|
+
<a name="l00231"></a>00231 errorReason = NONE;
|
254
|
+
<a name="l00232"></a>00232 lengthStringBufferSize = 0;
|
255
|
+
<a name="l00233"></a>00233 headerSize = 0;
|
256
|
+
<a name="l00234"></a>00234 headers.set_empty_key(<span class="stringliteral">""</span>);
|
257
|
+
<a name="l00235"></a>00235 }
|
258
|
+
<a name="l00236"></a>00236 <span class="comment"></span>
|
259
|
+
<a name="l00237"></a>00237 <span class="comment"> /**</span>
|
260
|
+
<a name="l00238"></a>00238 <span class="comment"> * Feed SCGI request data to the parser.</span>
|
261
|
+
<a name="l00239"></a>00239 <span class="comment"> *</span>
|
262
|
+
<a name="l00240"></a>00240 <span class="comment"> * @param data The data to feed.</span>
|
263
|
+
<a name="l00241"></a>00241 <span class="comment"> * @param size The size of the data, in bytes.</span>
|
264
|
+
<a name="l00242"></a>00242 <span class="comment"> * @return The number of recognized SCGI header bytes. If this value</span>
|
265
|
+
<a name="l00243"></a>00243 <span class="comment"> * equals 'size', then it means all data in 'data' is part of</span>
|
266
|
+
<a name="l00244"></a>00244 <span class="comment"> * the SCGI header. If this value is less than size, then it</span>
|
267
|
+
<a name="l00245"></a>00245 <span class="comment"> * means only some data in 'data' is part of the SCGI header,</span>
|
268
|
+
<a name="l00246"></a>00246 <span class="comment"> * and the remaining 'size - result' bytes are part of the</span>
|
269
|
+
<a name="l00247"></a>00247 <span class="comment"> * request body.</span>
|
270
|
+
<a name="l00248"></a>00248 <span class="comment"> * @pre size > 0</span>
|
271
|
+
<a name="l00249"></a>00249 <span class="comment"> * @post result <= size</span>
|
272
|
+
<a name="l00250"></a>00250 <span class="comment"> * @post if result <= size: getState() == DONE || getState() == ERROR</span>
|
273
|
+
<a name="l00251"></a>00251 <span class="comment"> */</span>
|
274
|
+
<a name="l00252"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#0498c9529029969d5a77e1f74f95baa3">00252</a> <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#0498c9529029969d5a77e1f74f95baa3" title="Feed SCGI request data to the parser.">feed</a>(<span class="keyword">const</span> <span class="keywordtype">char</span> *data, <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> size) {
|
275
|
+
<a name="l00253"></a>00253 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> i;
|
276
|
+
<a name="l00254"></a>00254
|
277
|
+
<a name="l00255"></a>00255 <span class="keywordflow">switch</span> (state) {
|
278
|
+
<a name="l00256"></a>00256 <span class="keywordflow">case</span> READING_LENGTH_STRING:
|
279
|
+
<a name="l00257"></a>00257 <span class="comment">// Keep processing length string data...</span>
|
280
|
+
<a name="l00258"></a>00258 <span class="keywordflow">for</span> (i = 0; i < size; i++) {
|
281
|
+
<a name="l00259"></a>00259 <span class="keywordtype">char</span> byte = data[i];
|
282
|
+
<a name="l00260"></a>00260
|
283
|
+
<a name="l00261"></a>00261 <span class="keywordflow">if</span> (lengthStringBufferSize == <span class="keyword">sizeof</span>(lengthStringBuffer) - 1) {
|
284
|
+
<a name="l00262"></a>00262 <span class="comment">// ...and abort if the length string is too long.</span>
|
285
|
+
<a name="l00263"></a>00263 state = ERROR;
|
286
|
+
<a name="l00264"></a>00264 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b9ba3d5851af9ffae1e9202bc9f73ac5b" title="The length string is too large.">LENGTH_STRING_TOO_LARGE</a>;
|
287
|
+
<a name="l00265"></a>00265 <span class="keywordflow">return</span> i;
|
288
|
+
<a name="l00266"></a>00266 } <span class="keywordflow">else</span> <span class="keywordflow">if</span> (!isDigit(byte)) {
|
289
|
+
<a name="l00267"></a>00267 <span class="keywordflow">if</span> (byte == <span class="charliteral">':'</span>) {
|
290
|
+
<a name="l00268"></a>00268 <span class="comment">// ...until the end of the length string has been reached.</span>
|
291
|
+
<a name="l00269"></a>00269 state = READING_HEADER_DATA;
|
292
|
+
<a name="l00270"></a>00270 lengthStringBuffer[lengthStringBufferSize] = <span class="charliteral">'\0'</span>;
|
293
|
+
<a name="l00271"></a>00271 headerSize = <a class="code" href="group__Support.html#g7b50461f1bc2b370c956967870da2762" title="Converts the given string to a long integer.">atol</a>(lengthStringBuffer);
|
294
|
+
<a name="l00272"></a>00272 <span class="keywordflow">if</span> (maxSize > 0 && headerSize > maxSize) {
|
295
|
+
<a name="l00273"></a>00273 state = ERROR;
|
296
|
+
<a name="l00274"></a>00274 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b876574447523322b9d7b4921e490032d" title="The header is larger than the maxSize value provided to the constructor.">LIMIT_REACHED</a>;
|
297
|
+
<a name="l00275"></a>00275 } <span class="keywordflow">else</span> {
|
298
|
+
<a name="l00276"></a>00276 headerBuffer.reserve(headerSize);
|
299
|
+
<a name="l00277"></a>00277 <span class="comment">// From here on, process the rest of the data that we've</span>
|
300
|
+
<a name="l00278"></a>00278 <span class="comment">// received, as header data.</span>
|
301
|
+
<a name="l00279"></a>00279 <span class="keywordflow">return</span> readHeaderData(data + i + 1, size - i - 1) + i + 1;
|
302
|
+
<a name="l00280"></a>00280 }
|
303
|
+
<a name="l00281"></a>00281 } <span class="keywordflow">else</span> {
|
304
|
+
<a name="l00282"></a>00282 <span class="comment">// ...until we encounter a parse error.</span>
|
305
|
+
<a name="l00283"></a>00283 state = ERROR;
|
306
|
+
<a name="l00284"></a>00284 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8bfe61dd339024beaed48f6eed6e6f3332" title="The length string contains an invalid character.">INVALID_LENGTH_STRING</a>;
|
307
|
+
<a name="l00285"></a>00285 <span class="keywordflow">return</span> i;
|
308
|
+
<a name="l00286"></a>00286 }
|
309
|
+
<a name="l00287"></a>00287 } <span class="keywordflow">else</span> {
|
310
|
+
<a name="l00288"></a>00288 lengthStringBuffer[lengthStringBufferSize] = byte;
|
311
|
+
<a name="l00289"></a>00289 lengthStringBufferSize++;
|
312
|
+
<a name="l00290"></a>00290 }
|
313
|
+
<a name="l00291"></a>00291 }
|
314
|
+
<a name="l00292"></a>00292 <span class="keywordflow">return</span> i;
|
315
|
+
<a name="l00293"></a>00293
|
316
|
+
<a name="l00294"></a>00294 <span class="keywordflow">case</span> READING_HEADER_DATA:
|
317
|
+
<a name="l00295"></a>00295 <span class="keywordflow">return</span> readHeaderData(data, size);
|
318
|
+
<a name="l00296"></a>00296
|
319
|
+
<a name="l00297"></a>00297 <span class="keywordflow">case</span> EXPECTING_COMMA:
|
320
|
+
<a name="l00298"></a>00298 <span class="keywordflow">if</span> (data[0] == <span class="charliteral">','</span>) {
|
321
|
+
<a name="l00299"></a>00299 state = DONE;
|
322
|
+
<a name="l00300"></a>00300 <span class="keywordflow">return</span> 1;
|
323
|
+
<a name="l00301"></a>00301 } <span class="keywordflow">else</span> {
|
324
|
+
<a name="l00302"></a>00302 state = ERROR;
|
325
|
+
<a name="l00303"></a>00303 errorReason = <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b44078eaf2ae1ee592d65d122dbda4a44" title="A header terminator character (&quot;,&quot;) was expected, but some else was encountered...">HEADER_TERMINATOR_EXPECTED</a>;
|
326
|
+
<a name="l00304"></a>00304 <span class="keywordflow">return</span> 0;
|
327
|
+
<a name="l00305"></a>00305 }
|
328
|
+
<a name="l00306"></a>00306
|
329
|
+
<a name="l00307"></a>00307 <span class="keywordflow">default</span>:
|
330
|
+
<a name="l00308"></a>00308 <span class="keywordflow">return</span> 0;
|
331
|
+
<a name="l00309"></a>00309 }
|
332
|
+
<a name="l00310"></a>00310 }
|
333
|
+
<a name="l00311"></a>00311 <span class="comment"></span>
|
334
|
+
<a name="l00312"></a>00312 <span class="comment"> /**</span>
|
335
|
+
<a name="l00313"></a>00313 <span class="comment"> * Get the raw header data that has been processed so far.</span>
|
336
|
+
<a name="l00314"></a>00314 <span class="comment"> */</span>
|
337
|
+
<a name="l00315"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#a96dbb7c4c642ac74f30ed5b14a64e86">00315</a> <span class="keywordtype">string</span> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#a96dbb7c4c642ac74f30ed5b14a64e86" title="Get the raw header data that has been processed so far.">getHeaderData</a>()<span class="keyword"> const </span>{
|
338
|
+
<a name="l00316"></a>00316 <span class="keywordflow">return</span> headerBuffer;
|
339
|
+
<a name="l00317"></a>00317 }
|
340
|
+
<a name="l00318"></a>00318 <span class="comment"></span>
|
341
|
+
<a name="l00319"></a>00319 <span class="comment"> /**</span>
|
342
|
+
<a name="l00320"></a>00320 <span class="comment"> * Get the value of the header with the given name.</span>
|
343
|
+
<a name="l00321"></a>00321 <span class="comment"> * Lookup is case-sensitive.</span>
|
344
|
+
<a name="l00322"></a>00322 <span class="comment"> *</span>
|
345
|
+
<a name="l00323"></a>00323 <span class="comment"> * Returns the empty string if there is no such header.</span>
|
346
|
+
<a name="l00324"></a>00324 <span class="comment"> *</span>
|
347
|
+
<a name="l00325"></a>00325 <span class="comment"> * @pre getState() == DONE</span>
|
348
|
+
<a name="l00326"></a>00326 <span class="comment"> */</span>
|
349
|
+
<a name="l00327"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#82321480ad2f8ac7ba58ff865131f6b4">00327</a> <a class="code" href="classPassenger_1_1StaticString.html" title="An immutable, static byte buffer.">StaticString</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#82321480ad2f8ac7ba58ff865131f6b4" title="Get the value of the header with the given name.">getHeader</a>(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1StaticString.html" title="An immutable, static byte buffer.">StaticString</a> &name)<span class="keyword"> const </span>{
|
350
|
+
<a name="l00328"></a>00328 HeaderMap::const_iterator it(headers.find(name));
|
351
|
+
<a name="l00329"></a>00329 <span class="keywordflow">if</span> (it == headers.end()) {
|
352
|
+
<a name="l00330"></a>00330 <span class="keywordflow">return</span> <span class="stringliteral">""</span>;
|
353
|
+
<a name="l00331"></a>00331 } <span class="keywordflow">else</span> {
|
354
|
+
<a name="l00332"></a>00332 <span class="keywordflow">return</span> it->second;
|
355
|
+
<a name="l00333"></a>00333 }
|
356
|
+
<a name="l00334"></a>00334 }
|
357
|
+
<a name="l00335"></a>00335 <span class="comment"></span>
|
358
|
+
<a name="l00336"></a>00336 <span class="comment"> /**</span>
|
359
|
+
<a name="l00337"></a>00337 <span class="comment"> * Checks whether there is a header with the given name.</span>
|
360
|
+
<a name="l00338"></a>00338 <span class="comment"> * Lookup is case-sensitive.</span>
|
361
|
+
<a name="l00339"></a>00339 <span class="comment"> *</span>
|
362
|
+
<a name="l00340"></a>00340 <span class="comment"> * @pre getState() == DONE</span>
|
363
|
+
<a name="l00341"></a>00341 <span class="comment"> */</span>
|
364
|
+
<a name="l00342"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#d82e358a0ad051ab2bdd2374e9ab4ff8">00342</a> <span class="keywordtype">bool</span> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#d82e358a0ad051ab2bdd2374e9ab4ff8" title="Checks whether there is a header with the given name.">hasHeader</a>(<span class="keyword">const</span> <a class="code" href="classPassenger_1_1StaticString.html" title="An immutable, static byte buffer.">StaticString</a> &name)<span class="keyword"> const </span>{
|
365
|
+
<a name="l00343"></a>00343 <span class="keywordflow">return</span> headers.find(name) != headers.end();
|
366
|
+
<a name="l00344"></a>00344 }
|
367
|
+
<a name="l00345"></a>00345 <span class="comment"></span>
|
368
|
+
<a name="l00346"></a>00346 <span class="comment"> /**</span>
|
369
|
+
<a name="l00347"></a>00347 <span class="comment"> * Get the parser's current state.</span>
|
370
|
+
<a name="l00348"></a>00348 <span class="comment"> */</span>
|
371
|
+
<a name="l00349"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#5bd7dbce81c69ae9e3462e84daf09012">00349</a> State <a class="code" href="classPassenger_1_1ScgiRequestParser.html#5bd7dbce81c69ae9e3462e84daf09012" title="Get the parser&#39;s current state.">getState</a>()<span class="keyword"> const </span>{
|
372
|
+
<a name="l00350"></a>00350 <span class="keywordflow">return</span> state;
|
373
|
+
<a name="l00351"></a>00351 }
|
374
|
+
<a name="l00352"></a>00352 <span class="comment"></span>
|
375
|
+
<a name="l00353"></a>00353 <span class="comment"> /**</span>
|
376
|
+
<a name="l00354"></a>00354 <span class="comment"> * Returns the reason why the parser entered the error state.</span>
|
377
|
+
<a name="l00355"></a>00355 <span class="comment"> *</span>
|
378
|
+
<a name="l00356"></a>00356 <span class="comment"> * @pre getState() == ERROR</span>
|
379
|
+
<a name="l00357"></a>00357 <span class="comment"> */</span>
|
380
|
+
<a name="l00358"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#4114253f47d3e7e5c0dddc3cd11c5596">00358</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#495e2aec6deffd64fd3d53dc81379f8b">ErrorReason</a> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#4114253f47d3e7e5c0dddc3cd11c5596" title="Returns the reason why the parser entered the error state.">getErrorReason</a>()<span class="keyword"> const </span>{
|
381
|
+
<a name="l00359"></a>00359 <span class="keywordflow">return</span> errorReason;
|
382
|
+
<a name="l00360"></a>00360 }
|
383
|
+
<a name="l00361"></a>00361 <span class="comment"></span>
|
384
|
+
<a name="l00362"></a>00362 <span class="comment"> /**</span>
|
385
|
+
<a name="l00363"></a>00363 <span class="comment"> * Checks whether this parser is still capable of accepting input (that</span>
|
386
|
+
<a name="l00364"></a>00364 <span class="comment"> * is, that this parser is not in a final state).</span>
|
387
|
+
<a name="l00365"></a>00365 <span class="comment"> */</span>
|
388
|
+
<a name="l00366"></a><a class="code" href="classPassenger_1_1ScgiRequestParser.html#5368ed40bf7c896f68d7ffaae164a092">00366</a> <span class="keywordtype">bool</span> <a class="code" href="classPassenger_1_1ScgiRequestParser.html#5368ed40bf7c896f68d7ffaae164a092" title="Checks whether this parser is still capable of accepting input (that is, that this...">acceptingInput</a>()<span class="keyword"> const </span>{
|
389
|
+
<a name="l00367"></a>00367 <span class="keywordflow">return</span> state != DONE && state != ERROR;
|
390
|
+
<a name="l00368"></a>00368 }
|
391
|
+
<a name="l00369"></a>00369 };
|
392
|
+
<a name="l00370"></a>00370
|
393
|
+
<a name="l00371"></a>00371 } <span class="comment">// namespace Passenger</span>
|
394
|
+
</pre></div></div>
|
395
|
+
<hr size="1"><address style="text-align: right;"><small>Generated on Sun Feb 21 12:22:46 2010 for Passenger by
|
396
|
+
<a href="http://www.doxygen.org/index.html">
|
397
|
+
<img src="doxygen.png" alt="doxygen" align="middle" border="0"></a> 1.5.8 </small></address>
|
398
|
+
</body>
|
399
|
+
</html>
|
@@ -0,0 +1,356 @@
|
|
1
|
+
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
|
2
|
+
<html><head><meta http-equiv="Content-Type" content="text/html;charset=UTF-8">
|
3
|
+
<title>Passenger: ServerInstanceDir.h Source File</title>
|
4
|
+
<link href="tabs.css" rel="stylesheet" type="text/css">
|
5
|
+
<link href="doxygen.css" rel="stylesheet" type="text/css">
|
6
|
+
</head><body>
|
7
|
+
<!-- Generated by Doxygen 1.5.8 -->
|
8
|
+
<div class="navigation" id="top">
|
9
|
+
<div class="tabs">
|
10
|
+
<ul>
|
11
|
+
<li><a href="main.html"><span>Main Page</span></a></li>
|
12
|
+
<li><a href="modules.html"><span>Modules</span></a></li>
|
13
|
+
<li><a href="namespaces.html"><span>Namespaces</span></a></li>
|
14
|
+
<li><a href="annotated.html"><span>Classes</span></a></li>
|
15
|
+
<li class="current"><a href="files.html"><span>Files</span></a></li>
|
16
|
+
</ul>
|
17
|
+
</div>
|
18
|
+
<div class="tabs">
|
19
|
+
<ul>
|
20
|
+
<li><a href="files.html"><span>File List</span></a></li>
|
21
|
+
</ul>
|
22
|
+
</div>
|
23
|
+
<h1>ServerInstanceDir.h</h1><div class="fragment"><pre class="fragment"><a name="l00001"></a>00001 <span class="comment">/*</span>
|
24
|
+
<a name="l00002"></a>00002 <span class="comment"> * Phusion Passenger - http://www.modrails.com/</span>
|
25
|
+
<a name="l00003"></a>00003 <span class="comment"> * Copyright (c) 2010 Phusion</span>
|
26
|
+
<a name="l00004"></a>00004 <span class="comment"> *</span>
|
27
|
+
<a name="l00005"></a>00005 <span class="comment"> * "Phusion Passenger" is a trademark of Hongli Lai & Ninh Bui.</span>
|
28
|
+
<a name="l00006"></a>00006 <span class="comment"> *</span>
|
29
|
+
<a name="l00007"></a>00007 <span class="comment"> * Permission is hereby granted, free of charge, to any person obtaining a copy</span>
|
30
|
+
<a name="l00008"></a>00008 <span class="comment"> * of this software and associated documentation files (the "Software"), to deal</span>
|
31
|
+
<a name="l00009"></a>00009 <span class="comment"> * in the Software without restriction, including without limitation the rights</span>
|
32
|
+
<a name="l00010"></a>00010 <span class="comment"> * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell</span>
|
33
|
+
<a name="l00011"></a>00011 <span class="comment"> * copies of the Software, and to permit persons to whom the Software is</span>
|
34
|
+
<a name="l00012"></a>00012 <span class="comment"> * furnished to do so, subject to the following conditions:</span>
|
35
|
+
<a name="l00013"></a>00013 <span class="comment"> *</span>
|
36
|
+
<a name="l00014"></a>00014 <span class="comment"> * The above copyright notice and this permission notice shall be included in</span>
|
37
|
+
<a name="l00015"></a>00015 <span class="comment"> * all copies or substantial portions of the Software.</span>
|
38
|
+
<a name="l00016"></a>00016 <span class="comment"> *</span>
|
39
|
+
<a name="l00017"></a>00017 <span class="comment"> * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR</span>
|
40
|
+
<a name="l00018"></a>00018 <span class="comment"> * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,</span>
|
41
|
+
<a name="l00019"></a>00019 <span class="comment"> * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE</span>
|
42
|
+
<a name="l00020"></a>00020 <span class="comment"> * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER</span>
|
43
|
+
<a name="l00021"></a>00021 <span class="comment"> * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,</span>
|
44
|
+
<a name="l00022"></a>00022 <span class="comment"> * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN</span>
|
45
|
+
<a name="l00023"></a>00023 <span class="comment"> * THE SOFTWARE.</span>
|
46
|
+
<a name="l00024"></a>00024 <span class="comment"> */</span>
|
47
|
+
<a name="l00025"></a>00025 <span class="preprocessor">#ifndef _PASSENGER_SERVER_INSTANCE_DIR_H_</span>
|
48
|
+
<a name="l00026"></a>00026 <span class="preprocessor"></span><span class="preprocessor">#define _PASSENGER_SERVER_INSTANCE_DIR_H_</span>
|
49
|
+
<a name="l00027"></a>00027 <span class="preprocessor"></span>
|
50
|
+
<a name="l00028"></a>00028 <span class="preprocessor">#include <boost/noncopyable.hpp></span>
|
51
|
+
<a name="l00029"></a>00029 <span class="preprocessor">#include <boost/shared_ptr.hpp></span>
|
52
|
+
<a name="l00030"></a>00030
|
53
|
+
<a name="l00031"></a>00031 <span class="preprocessor">#include <sys/types.h></span>
|
54
|
+
<a name="l00032"></a>00032 <span class="preprocessor">#include <dirent.h></span>
|
55
|
+
<a name="l00033"></a>00033 <span class="preprocessor">#include <unistd.h></span>
|
56
|
+
<a name="l00034"></a>00034 <span class="preprocessor">#include <pwd.h></span>
|
57
|
+
<a name="l00035"></a>00035 <span class="preprocessor">#include <grp.h></span>
|
58
|
+
<a name="l00036"></a>00036 <span class="preprocessor">#include <cstdlib></span>
|
59
|
+
<a name="l00037"></a>00037 <span class="preprocessor">#include <cstring></span>
|
60
|
+
<a name="l00038"></a>00038 <span class="preprocessor">#include <string></span>
|
61
|
+
<a name="l00039"></a>00039
|
62
|
+
<a name="l00040"></a>00040 <span class="preprocessor">#include "Exceptions.h"</span>
|
63
|
+
<a name="l00041"></a>00041 <span class="preprocessor">#include "Utils.h"</span>
|
64
|
+
<a name="l00042"></a>00042
|
65
|
+
<a name="l00043"></a>00043 <span class="keyword">namespace </span>Passenger {
|
66
|
+
<a name="l00044"></a>00044
|
67
|
+
<a name="l00045"></a>00045 <span class="keyword">using namespace </span>std;
|
68
|
+
<a name="l00046"></a>00046 <span class="keyword">using namespace </span>boost;
|
69
|
+
<a name="l00047"></a>00047
|
70
|
+
<a name="l00048"></a>00048 <span class="keyword">class </span>ServerInstanceDir: <span class="keyword">public</span> noncopyable {
|
71
|
+
<a name="l00049"></a>00049 <span class="keyword">public</span>:
|
72
|
+
<a name="l00050"></a>00050 <span class="comment">// Don't forget to update lib/phusion_passenger/admin_tools/server_instance.rb too.</span>
|
73
|
+
<a name="l00051"></a>00051 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> DIR_STRUCTURE_MAJOR_VERSION = 1;
|
74
|
+
<a name="l00052"></a>00052 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> DIR_STRUCTURE_MINOR_VERSION = 0;
|
75
|
+
<a name="l00053"></a>00053 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> GENERATION_STRUCTURE_MAJOR_VERSION = 1;
|
76
|
+
<a name="l00054"></a>00054 <span class="keyword">static</span> <span class="keyword">const</span> <span class="keywordtype">int</span> GENERATION_STRUCTURE_MINOR_VERSION = 0;
|
77
|
+
<a name="l00055"></a>00055
|
78
|
+
<a name="l00056"></a>00056 <span class="keyword">class </span>Generation: <span class="keyword">public</span> noncopyable {
|
79
|
+
<a name="l00057"></a>00057 <span class="keyword">private</span>:
|
80
|
+
<a name="l00058"></a>00058 <span class="keyword">friend</span> <span class="keyword">class </span>ServerInstanceDir;
|
81
|
+
<a name="l00059"></a>00059
|
82
|
+
<a name="l00060"></a>00060 <span class="keywordtype">string</span> path;
|
83
|
+
<a name="l00061"></a>00061 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> number;
|
84
|
+
<a name="l00062"></a>00062 <span class="keywordtype">bool</span> owner;
|
85
|
+
<a name="l00063"></a>00063
|
86
|
+
<a name="l00064"></a>00064 Generation(<span class="keyword">const</span> <span class="keywordtype">string</span> &serverInstanceDir, <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> number) {
|
87
|
+
<a name="l00065"></a>00065 path = serverInstanceDir + <span class="stringliteral">"/generation-"</span> + <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(number);
|
88
|
+
<a name="l00066"></a>00066 this->number = number;
|
89
|
+
<a name="l00067"></a>00067 owner = <span class="keyword">false</span>;
|
90
|
+
<a name="l00068"></a>00068 }
|
91
|
+
<a name="l00069"></a>00069
|
92
|
+
<a name="l00070"></a>00070 <span class="keywordtype">void</span> create(<span class="keywordtype">bool</span> userSwitching, <span class="keyword">const</span> <span class="keywordtype">string</span> &defaultUser,
|
93
|
+
<a name="l00071"></a>00071 <span class="keyword">const</span> <span class="keywordtype">string</span> &defaultGroup, uid_t webServerWorkerUid,
|
94
|
+
<a name="l00072"></a>00072 gid_t webServerWorkerGid)
|
95
|
+
<a name="l00073"></a>00073 {
|
96
|
+
<a name="l00074"></a>00074 <span class="keywordtype">bool</span> runningAsRoot = geteuid() == 0;
|
97
|
+
<a name="l00075"></a>00075 <span class="keyword">struct </span>passwd *defaultUserEntry;
|
98
|
+
<a name="l00076"></a>00076 <span class="keyword">struct </span>group *defaultGroupEntry;
|
99
|
+
<a name="l00077"></a>00077 uid_t defaultUid;
|
100
|
+
<a name="l00078"></a>00078 gid_t defaultGid;
|
101
|
+
<a name="l00079"></a>00079
|
102
|
+
<a name="l00080"></a>00080 defaultUserEntry = getpwnam(defaultUser.c_str());
|
103
|
+
<a name="l00081"></a>00081 <span class="keywordflow">if</span> (defaultUserEntry == NULL) {
|
104
|
+
<a name="l00082"></a>00082 <span class="keywordflow">throw</span> NonExistentUserException(<span class="stringliteral">"Default user '"</span> + defaultUser +
|
105
|
+
<a name="l00083"></a>00083 <span class="stringliteral">"' does not exist."</span>);
|
106
|
+
<a name="l00084"></a>00084 }
|
107
|
+
<a name="l00085"></a>00085 defaultUid = defaultUserEntry->pw_uid;
|
108
|
+
<a name="l00086"></a>00086 defaultGroupEntry = getgrnam(defaultGroup.c_str());
|
109
|
+
<a name="l00087"></a>00087 <span class="keywordflow">if</span> (defaultGroupEntry == NULL) {
|
110
|
+
<a name="l00088"></a>00088 <span class="keywordflow">throw</span> NonExistentGroupException(<span class="stringliteral">"Default group '"</span> + defaultGroup +
|
111
|
+
<a name="l00089"></a>00089 <span class="stringliteral">"' does not exist."</span>);
|
112
|
+
<a name="l00090"></a>00090 }
|
113
|
+
<a name="l00091"></a>00091 defaultGid = defaultGroupEntry->gr_gid;
|
114
|
+
<a name="l00092"></a>00092
|
115
|
+
<a name="l00093"></a>00093 <span class="comment">/* We set a very tight permission here: no read or write access for</span>
|
116
|
+
<a name="l00094"></a>00094 <span class="comment"> * anybody except the owner. The individual files and subdirectories</span>
|
117
|
+
<a name="l00095"></a>00095 <span class="comment"> * decide for themselves whether they're readable by anybody.</span>
|
118
|
+
<a name="l00096"></a>00096 <span class="comment"> */</span>
|
119
|
+
<a name="l00097"></a>00097 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path, <span class="stringliteral">"u=rwxs,g=x,o=x"</span>);
|
120
|
+
<a name="l00098"></a>00098
|
121
|
+
<a name="l00099"></a>00099 <span class="comment">/* Write structure version file. */</span>
|
122
|
+
<a name="l00100"></a>00100 <span class="keywordtype">string</span> structureVersionFile = path + <span class="stringliteral">"/structure_version.txt"</span>;
|
123
|
+
<a name="l00101"></a>00101 <a class="code" href="group__Support.html#g6f414070373465da2e12947463f69953" title="Create the given file with the given contents, permissions and ownership.">createFile</a>(structureVersionFile,
|
124
|
+
<a name="l00102"></a>00102 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(GENERATION_STRUCTURE_MAJOR_VERSION) + <span class="stringliteral">"."</span> +
|
125
|
+
<a name="l00103"></a>00103 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(GENERATION_STRUCTURE_MINOR_VERSION),
|
126
|
+
<a name="l00104"></a>00104 S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH);
|
127
|
+
<a name="l00105"></a>00105
|
128
|
+
<a name="l00106"></a>00106
|
129
|
+
<a name="l00107"></a>00107 <span class="comment">/* We want the upload buffer directory to be only writable by the web</span>
|
130
|
+
<a name="l00108"></a>00108 <span class="comment"> * server's worker processs. Other users may not have any access to this</span>
|
131
|
+
<a name="l00109"></a>00109 <span class="comment"> * directory.</span>
|
132
|
+
<a name="l00110"></a>00110 <span class="comment"> */</span>
|
133
|
+
<a name="l00111"></a>00111 <span class="keywordflow">if</span> (runningAsRoot) {
|
134
|
+
<a name="l00112"></a>00112 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/buffered_uploads"</span>, <span class="stringliteral">"u=rwxs,g=,o="</span>,
|
135
|
+
<a name="l00113"></a>00113 webServerWorkerUid, webServerWorkerGid);
|
136
|
+
<a name="l00114"></a>00114 } <span class="keywordflow">else</span> {
|
137
|
+
<a name="l00115"></a>00115 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/buffered_uploads"</span>, <span class="stringliteral">"u=rwxs,g=,o="</span>);
|
138
|
+
<a name="l00116"></a>00116 }
|
139
|
+
<a name="l00117"></a>00117
|
140
|
+
<a name="l00118"></a>00118 <span class="comment">/* The web server must be able to directly connect to a backend. */</span>
|
141
|
+
<a name="l00119"></a>00119 <span class="keywordflow">if</span> (runningAsRoot) {
|
142
|
+
<a name="l00120"></a>00120 <span class="keywordflow">if</span> (userSwitching) {
|
143
|
+
<a name="l00121"></a>00121 <span class="comment">/* Each backend process may be running as a different user,</span>
|
144
|
+
<a name="l00122"></a>00122 <span class="comment"> * so the backends subdirectory must be world-writable.</span>
|
145
|
+
<a name="l00123"></a>00123 <span class="comment"> * However we don't want everybody to be able to know the</span>
|
146
|
+
<a name="l00124"></a>00124 <span class="comment"> * sockets' filenames, so the directory is not readable.</span>
|
147
|
+
<a name="l00125"></a>00125 <span class="comment"> */</span>
|
148
|
+
<a name="l00126"></a>00126 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/backends"</span>, <span class="stringliteral">"u=rwxs,g=wx,o=wx"</span>);
|
149
|
+
<a name="l00127"></a>00127 } <span class="keywordflow">else</span> {
|
150
|
+
<a name="l00128"></a>00128 <span class="comment">/* All backend processes are running as defaultUser/defaultGroup,</span>
|
151
|
+
<a name="l00129"></a>00129 <span class="comment"> * so make defaultUser/defaultGroup the owner and group of the</span>
|
152
|
+
<a name="l00130"></a>00130 <span class="comment"> * subdirecory.</span>
|
153
|
+
<a name="l00131"></a>00131 <span class="comment"> *</span>
|
154
|
+
<a name="l00132"></a>00132 <span class="comment"> * The directory is not readable as a security precaution:</span>
|
155
|
+
<a name="l00133"></a>00133 <span class="comment"> * nobody should be able to know the sockets' filenames without</span>
|
156
|
+
<a name="l00134"></a>00134 <span class="comment"> * having access to the application pool.</span>
|
157
|
+
<a name="l00135"></a>00135 <span class="comment"> */</span>
|
158
|
+
<a name="l00136"></a>00136 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/backends"</span>, <span class="stringliteral">"u=rwxs,g=x,o=x"</span>, defaultUid, defaultGid);
|
159
|
+
<a name="l00137"></a>00137 }
|
160
|
+
<a name="l00138"></a>00138 } <span class="keywordflow">else</span> {
|
161
|
+
<a name="l00139"></a>00139 <span class="comment">/* All backend processes are running as the same user as the web server,</span>
|
162
|
+
<a name="l00140"></a>00140 <span class="comment"> * so only allow access for this user.</span>
|
163
|
+
<a name="l00141"></a>00141 <span class="comment"> */</span>
|
164
|
+
<a name="l00142"></a>00142 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/backends"</span>, <span class="stringliteral">"u=rwxs,g=,o="</span>);
|
165
|
+
<a name="l00143"></a>00143 }
|
166
|
+
<a name="l00144"></a>00144
|
167
|
+
<a name="l00145"></a>00145 <span class="comment">/* The helper server (containing the application pool) must be able to access</span>
|
168
|
+
<a name="l00146"></a>00146 <span class="comment"> * the spawn server's socket.</span>
|
169
|
+
<a name="l00147"></a>00147 <span class="comment"> */</span>
|
170
|
+
<a name="l00148"></a>00148 <span class="keywordflow">if</span> (runningAsRoot) {
|
171
|
+
<a name="l00149"></a>00149 <span class="keywordflow">if</span> (userSwitching) {
|
172
|
+
<a name="l00150"></a>00150 <span class="comment">/* Both the helper server and the spawn server are</span>
|
173
|
+
<a name="l00151"></a>00151 <span class="comment"> * running as root.</span>
|
174
|
+
<a name="l00152"></a>00152 <span class="comment"> */</span>
|
175
|
+
<a name="l00153"></a>00153 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/spawn-server"</span>, <span class="stringliteral">"u=rwxs,g=,o="</span>);
|
176
|
+
<a name="l00154"></a>00154 } <span class="keywordflow">else</span> {
|
177
|
+
<a name="l00155"></a>00155 <span class="comment">/* Both the helper server and the spawn server are</span>
|
178
|
+
<a name="l00156"></a>00156 <span class="comment"> * running as defaultUser/defaultGroup.</span>
|
179
|
+
<a name="l00157"></a>00157 <span class="comment"> */</span>
|
180
|
+
<a name="l00158"></a>00158 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/spawn-server"</span>, <span class="stringliteral">"u=rwxs,g=,o="</span>,
|
181
|
+
<a name="l00159"></a>00159 defaultUid, defaultGid);
|
182
|
+
<a name="l00160"></a>00160 }
|
183
|
+
<a name="l00161"></a>00161 } <span class="keywordflow">else</span> {
|
184
|
+
<a name="l00162"></a>00162 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path + <span class="stringliteral">"/spawn-server"</span>, <span class="stringliteral">"u=rwxs,g=,o="</span>);
|
185
|
+
<a name="l00163"></a>00163 }
|
186
|
+
<a name="l00164"></a>00164
|
187
|
+
<a name="l00165"></a>00165 owner = <span class="keyword">true</span>;
|
188
|
+
<a name="l00166"></a>00166 }
|
189
|
+
<a name="l00167"></a>00167
|
190
|
+
<a name="l00168"></a>00168 <span class="keyword">public</span>:
|
191
|
+
<a name="l00169"></a>00169 ~Generation() {
|
192
|
+
<a name="l00170"></a>00170 <span class="keywordflow">if</span> (owner) {
|
193
|
+
<a name="l00171"></a>00171 <a class="code" href="namespacePassenger.html#8422d210cbd2f62d21254415b5a7f747" title="Remove an entire directory tree recursively.">removeDirTree</a>(path);
|
194
|
+
<a name="l00172"></a>00172 }
|
195
|
+
<a name="l00173"></a>00173 }
|
196
|
+
<a name="l00174"></a>00174
|
197
|
+
<a name="l00175"></a>00175 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> getNumber()<span class="keyword"> const </span>{
|
198
|
+
<a name="l00176"></a>00176 <span class="keywordflow">return</span> number;
|
199
|
+
<a name="l00177"></a>00177 }
|
200
|
+
<a name="l00178"></a>00178
|
201
|
+
<a name="l00179"></a>00179 <span class="keywordtype">string</span> getPath()<span class="keyword"> const </span>{
|
202
|
+
<a name="l00180"></a>00180 <span class="keywordflow">return</span> path;
|
203
|
+
<a name="l00181"></a>00181 }
|
204
|
+
<a name="l00182"></a>00182
|
205
|
+
<a name="l00183"></a>00183 <span class="keywordtype">void</span> detach() {
|
206
|
+
<a name="l00184"></a>00184 owner = <span class="keyword">false</span>;
|
207
|
+
<a name="l00185"></a>00185 }
|
208
|
+
<a name="l00186"></a>00186 };
|
209
|
+
<a name="l00187"></a>00187
|
210
|
+
<a name="l00188"></a>00188 <span class="keyword">typedef</span> shared_ptr<Generation> GenerationPtr;
|
211
|
+
<a name="l00189"></a>00189
|
212
|
+
<a name="l00190"></a>00190 <span class="keyword">private</span>:
|
213
|
+
<a name="l00191"></a>00191 <span class="keywordtype">string</span> path;
|
214
|
+
<a name="l00192"></a>00192 <span class="keywordtype">bool</span> owner;
|
215
|
+
<a name="l00193"></a>00193
|
216
|
+
<a name="l00194"></a>00194 <span class="keyword">friend</span> <span class="keyword">class </span>Generation;
|
217
|
+
<a name="l00195"></a>00195
|
218
|
+
<a name="l00196"></a>00196 <span class="keywordtype">void</span> initialize(<span class="keyword">const</span> <span class="keywordtype">string</span> &path, <span class="keywordtype">bool</span> owner) {
|
219
|
+
<a name="l00197"></a>00197 this->path = path;
|
220
|
+
<a name="l00198"></a>00198 this->owner = owner;
|
221
|
+
<a name="l00199"></a>00199
|
222
|
+
<a name="l00200"></a>00200 <span class="comment">/* Create the server instance directory. We only need to write to this</span>
|
223
|
+
<a name="l00201"></a>00201 <span class="comment"> * directory for these reasons:</span>
|
224
|
+
<a name="l00202"></a>00202 <span class="comment"> * 1. Initial population of structure files (structure_version.txt, instance.pid).</span>
|
225
|
+
<a name="l00203"></a>00203 <span class="comment"> * 2. Creating/removing a generation directory.</span>
|
226
|
+
<a name="l00204"></a>00204 <span class="comment"> * 3. Removing the entire server instance directory (after all</span>
|
227
|
+
<a name="l00205"></a>00205 <span class="comment"> * generations are removed).</span>
|
228
|
+
<a name="l00206"></a>00206 <span class="comment"> *</span>
|
229
|
+
<a name="l00207"></a>00207 <span class="comment"> * 1 and 2 are done by the helper server during initialization and before lowering</span>
|
230
|
+
<a name="l00208"></a>00208 <span class="comment"> * privilege. 3 is done during helper server shutdown by a cleanup process that's</span>
|
231
|
+
<a name="l00209"></a>00209 <span class="comment"> * running as the same user the helper server was running as before privilege</span>
|
232
|
+
<a name="l00210"></a>00210 <span class="comment"> * lowering.</span>
|
233
|
+
<a name="l00211"></a>00211 <span class="comment"> * Therefore, we make the directory only writable by the user the helper server</span>
|
234
|
+
<a name="l00212"></a>00212 <span class="comment"> * was running as before privilege is lowered. Everybody else has read and execute</span>
|
235
|
+
<a name="l00213"></a>00213 <span class="comment"> * rights though, because we want admin tools to be able to list the available</span>
|
236
|
+
<a name="l00214"></a>00214 <span class="comment"> * generations no matter what user they're running as.</span>
|
237
|
+
<a name="l00215"></a>00215 <span class="comment"> */</span>
|
238
|
+
<a name="l00216"></a>00216 <a class="code" href="namespacePassenger.html#448f25263af0b37b91caee3d294ec4aa" title="Create the directory at the given path, creating intermediate directories if necessary...">makeDirTree</a>(path, <span class="stringliteral">"u=rwxs,g=rx,o=rx"</span>);
|
239
|
+
<a name="l00217"></a>00217 }
|
240
|
+
<a name="l00218"></a>00218
|
241
|
+
<a name="l00219"></a>00219 <span class="keyword">public</span>:
|
242
|
+
<a name="l00220"></a>00220 ServerInstanceDir(pid_t webServerPid, <span class="keyword">const</span> <span class="keywordtype">string</span> &parentDir = <span class="stringliteral">""</span>, <span class="keywordtype">bool</span> owner = <span class="keyword">true</span>) {
|
243
|
+
<a name="l00221"></a>00221 <span class="keywordtype">string</span> theParentDir;
|
244
|
+
<a name="l00222"></a>00222
|
245
|
+
<a name="l00223"></a>00223 <span class="keywordflow">if</span> (parentDir.empty()) {
|
246
|
+
<a name="l00224"></a>00224 theParentDir = <a class="code" href="group__Support.html#g68f4105c5a8e510520b5ea3eecb66213" title="Return the path name for the directory in which the system stores general temporary...">getSystemTempDir</a>();
|
247
|
+
<a name="l00225"></a>00225 } <span class="keywordflow">else</span> {
|
248
|
+
<a name="l00226"></a>00226 theParentDir = parentDir;
|
249
|
+
<a name="l00227"></a>00227 }
|
250
|
+
<a name="l00228"></a>00228
|
251
|
+
<a name="l00229"></a>00229 <span class="comment">/* We embed the super structure version in the server instance directory name</span>
|
252
|
+
<a name="l00230"></a>00230 <span class="comment"> * because it's possible to upgrade Phusion Passenger without changing the</span>
|
253
|
+
<a name="l00231"></a>00231 <span class="comment"> * web server's PID. This way each incompatible upgrade will use its own</span>
|
254
|
+
<a name="l00232"></a>00232 <span class="comment"> * server instance directory.</span>
|
255
|
+
<a name="l00233"></a>00233 <span class="comment"> */</span>
|
256
|
+
<a name="l00234"></a>00234 initialize(theParentDir + <span class="stringliteral">"/passenger."</span> +
|
257
|
+
<a name="l00235"></a>00235 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(DIR_STRUCTURE_MAJOR_VERSION) + <span class="stringliteral">"."</span> +
|
258
|
+
<a name="l00236"></a>00236 <a class="code" href="group__Support.html#gf6fab368d70c18fdf16bf5a24f630407" title="Convert anything to a string.">toString</a>(DIR_STRUCTURE_MINOR_VERSION) + <span class="stringliteral">"."</span> +
|
259
|
+
<a name="l00237"></a>00237 toString<unsigned long long>(webServerPid),
|
260
|
+
<a name="l00238"></a>00238 owner);
|
261
|
+
<a name="l00239"></a>00239
|
262
|
+
<a name="l00240"></a>00240 }
|
263
|
+
<a name="l00241"></a>00241
|
264
|
+
<a name="l00242"></a>00242 ServerInstanceDir(<span class="keyword">const</span> <span class="keywordtype">string</span> &path, <span class="keywordtype">bool</span> owner = <span class="keyword">true</span>) {
|
265
|
+
<a name="l00243"></a>00243 initialize(path, owner);
|
266
|
+
<a name="l00244"></a>00244 }
|
267
|
+
<a name="l00245"></a>00245
|
268
|
+
<a name="l00246"></a>00246 ~ServerInstanceDir() {
|
269
|
+
<a name="l00247"></a>00247 <span class="keywordflow">if</span> (owner) {
|
270
|
+
<a name="l00248"></a>00248 GenerationPtr newestGeneration;
|
271
|
+
<a name="l00249"></a>00249 <span class="keywordflow">try</span> {
|
272
|
+
<a name="l00250"></a>00250 newestGeneration = getNewestGeneration();
|
273
|
+
<a name="l00251"></a>00251 } <span class="keywordflow">catch</span> (<span class="keyword">const</span> FileSystemException &e) {
|
274
|
+
<a name="l00252"></a>00252 <span class="keywordflow">if</span> (e.code() == ENOENT) {
|
275
|
+
<a name="l00253"></a>00253 <span class="keywordflow">return</span>;
|
276
|
+
<a name="l00254"></a>00254 } <span class="keywordflow">else</span> {
|
277
|
+
<a name="l00255"></a>00255 <span class="keywordflow">throw</span>;
|
278
|
+
<a name="l00256"></a>00256 }
|
279
|
+
<a name="l00257"></a>00257 }
|
280
|
+
<a name="l00258"></a>00258 <span class="keywordflow">if</span> (newestGeneration == NULL) {
|
281
|
+
<a name="l00259"></a>00259 <a class="code" href="namespacePassenger.html#8422d210cbd2f62d21254415b5a7f747" title="Remove an entire directory tree recursively.">removeDirTree</a>(path);
|
282
|
+
<a name="l00260"></a>00260 }
|
283
|
+
<a name="l00261"></a>00261 }
|
284
|
+
<a name="l00262"></a>00262 }
|
285
|
+
<a name="l00263"></a>00263
|
286
|
+
<a name="l00264"></a>00264 <span class="keywordtype">string</span> getPath()<span class="keyword"> const </span>{
|
287
|
+
<a name="l00265"></a>00265 <span class="keywordflow">return</span> path;
|
288
|
+
<a name="l00266"></a>00266 }
|
289
|
+
<a name="l00267"></a>00267
|
290
|
+
<a name="l00268"></a>00268 <span class="keywordtype">void</span> detach() {
|
291
|
+
<a name="l00269"></a>00269 owner = <span class="keyword">false</span>;
|
292
|
+
<a name="l00270"></a>00270 }
|
293
|
+
<a name="l00271"></a>00271
|
294
|
+
<a name="l00272"></a>00272 GenerationPtr newGeneration(<span class="keywordtype">bool</span> userSwitching, <span class="keyword">const</span> <span class="keywordtype">string</span> &defaultUser,
|
295
|
+
<a name="l00273"></a>00273 <span class="keyword">const</span> <span class="keywordtype">string</span> &defaultGroup, uid_t webServerWorkerUid,
|
296
|
+
<a name="l00274"></a>00274 gid_t webServerWorkerGid)
|
297
|
+
<a name="l00275"></a>00275 {
|
298
|
+
<a name="l00276"></a>00276 GenerationPtr newestGeneration = getNewestGeneration();
|
299
|
+
<a name="l00277"></a>00277 <span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> newNumber;
|
300
|
+
<a name="l00278"></a>00278 <span class="keywordflow">if</span> (newestGeneration != NULL) {
|
301
|
+
<a name="l00279"></a>00279 newNumber = newestGeneration->getNumber() + 1;
|
302
|
+
<a name="l00280"></a>00280 } <span class="keywordflow">else</span> {
|
303
|
+
<a name="l00281"></a>00281 newNumber = 0;
|
304
|
+
<a name="l00282"></a>00282 }
|
305
|
+
<a name="l00283"></a>00283
|
306
|
+
<a name="l00284"></a>00284 GenerationPtr generation(<span class="keyword">new</span> Generation(path, newNumber));
|
307
|
+
<a name="l00285"></a>00285 generation->create(userSwitching, defaultUser, defaultGroup,
|
308
|
+
<a name="l00286"></a>00286 webServerWorkerUid, webServerWorkerGid);
|
309
|
+
<a name="l00287"></a>00287 <span class="keywordflow">return</span> generation;
|
310
|
+
<a name="l00288"></a>00288 }
|
311
|
+
<a name="l00289"></a>00289
|
312
|
+
<a name="l00290"></a>00290 GenerationPtr getGeneration(<span class="keywordtype">unsigned</span> <span class="keywordtype">int</span> number)<span class="keyword"> const </span>{
|
313
|
+
<a name="l00291"></a>00291 <span class="keywordflow">return</span> <a class="code" href="group__Support.html#g41b6c4a82fed72531a147de0505a8396" title="Convenience shortcut for creating a shared_ptr.">ptr</a>(<span class="keyword">new</span> Generation(path, number));
|
314
|
+
<a name="l00292"></a>00292 }
|
315
|
+
<a name="l00293"></a>00293
|
316
|
+
<a name="l00294"></a>00294 GenerationPtr getNewestGeneration()<span class="keyword"> const </span>{
|
317
|
+
<a name="l00295"></a>00295 DIR *dir = opendir(path.c_str());
|
318
|
+
<a name="l00296"></a>00296 <span class="keyword">struct </span>dirent *entry;
|
319
|
+
<a name="l00297"></a>00297 <span class="keywordtype">int</span> result = -1;
|
320
|
+
<a name="l00298"></a>00298
|
321
|
+
<a name="l00299"></a>00299 <span class="keywordflow">if</span> (dir == NULL) {
|
322
|
+
<a name="l00300"></a>00300 <span class="keywordtype">int</span> e = errno;
|
323
|
+
<a name="l00301"></a>00301 <span class="keywordflow">throw</span> FileSystemException(<span class="stringliteral">"Cannot open directory "</span> + path,
|
324
|
+
<a name="l00302"></a>00302 e, path);
|
325
|
+
<a name="l00303"></a>00303 }
|
326
|
+
<a name="l00304"></a>00304 <span class="keywordflow">while</span> ((entry = readdir(dir)) != NULL) {
|
327
|
+
<a name="l00305"></a>00305 <span class="keywordflow">if</span> (entry->d_type == DT_DIR
|
328
|
+
<a name="l00306"></a>00306 && strncmp(entry->d_name, <span class="stringliteral">"generation-"</span>, <span class="keyword">sizeof</span>(<span class="stringliteral">"generation-"</span>) - 1) == 0) {
|
329
|
+
<a name="l00307"></a>00307 <span class="keyword">const</span> <span class="keywordtype">char</span> *numberString = entry->d_name + <span class="keyword">sizeof</span>(<span class="stringliteral">"generation-"</span>) - 1;
|
330
|
+
<a name="l00308"></a>00308 <span class="keywordtype">int</span> number = <a class="code" href="group__Support.html#g73b17b509ee44938a56bf16cbf82fb48" title="Converts the given string to an integer.">atoi</a>(numberString);
|
331
|
+
<a name="l00309"></a>00309 <span class="keywordflow">if</span> (number >= 0 && number > result) {
|
332
|
+
<a name="l00310"></a>00310 result = number;
|
333
|
+
<a name="l00311"></a>00311 }
|
334
|
+
<a name="l00312"></a>00312 }
|
335
|
+
<a name="l00313"></a>00313 }
|
336
|
+
<a name="l00314"></a>00314 closedir(dir);
|
337
|
+
<a name="l00315"></a>00315
|
338
|
+
<a name="l00316"></a>00316 <span class="keywordflow">if</span> (result == -1) {
|
339
|
+
<a name="l00317"></a>00317 <span class="keywordflow">return</span> GenerationPtr();
|
340
|
+
<a name="l00318"></a>00318 } <span class="keywordflow">else</span> {
|
341
|
+
<a name="l00319"></a>00319 <span class="keywordflow">return</span> getGeneration(result);
|
342
|
+
<a name="l00320"></a>00320 }
|
343
|
+
<a name="l00321"></a>00321 }
|
344
|
+
<a name="l00322"></a>00322 };
|
345
|
+
<a name="l00323"></a>00323
|
346
|
+
<a name="l00324"></a>00324 <span class="keyword">typedef</span> shared_ptr<ServerInstanceDir> ServerInstanceDirPtr;
|
347
|
+
<a name="l00325"></a>00325
|
348
|
+
<a name="l00326"></a>00326 } <span class="comment">// namespace Passenger</span>
|
349
|
+
<a name="l00327"></a>00327
|
350
|
+
<a name="l00328"></a>00328 <span class="preprocessor">#endif </span><span class="comment">/* _PASSENGER_SERVER_INSTANCE_DIR_H_ */</span>
|
351
|
+
</pre></div></div>
|
352
|
+
<hr size="1"><address style="text-align: right;"><small>Generated on Sun Feb 21 12:22:46 2010 for Passenger by
|
353
|
+
<a href="http://www.doxygen.org/index.html">
|
354
|
+
<img src="doxygen.png" alt="doxygen" align="middle" border="0"></a> 1.5.8 </small></address>
|
355
|
+
</body>
|
356
|
+
</html>
|