Class: WEBrick::GenericServer
- Inherits:
-
Object
- Object
- WEBrick::GenericServer
- Defined in:
- lib/webrick/ssl.rb,
lib/webrick/server.rb
Overview
Base TCP server class. You must subclass GenericServer and provide a #run method.
Direct Known Subclasses
Instance Attribute Summary collapse
-
#config ⇒ Object
readonly
The server configuration.
-
#listeners ⇒ Object
readonly
Sockets listening for connections.
-
#logger ⇒ Object
readonly
The server logger.
-
#status ⇒ Object
readonly
The server status.
-
#tokens ⇒ Object
readonly
Tokens control the number of outstanding clients.
Instance Method Summary collapse
-
#[](key) ⇒ Object
Retrieves
key
from the configuration. -
#initialize(config = {}, default = Config::General) ⇒ GenericServer
constructor
Creates a new generic server from
config
. -
#listen(address, port) ⇒ Object
Adds listeners from
address
andport
to the server. -
#run(sock) ⇒ Object
You must subclass GenericServer and implement #run which accepts a TCP client socket.
-
#setup_ssl_context(config) ⇒ Object
Sets up an SSL context for
config
. -
#shutdown ⇒ Object
Shuts down the server and all listening sockets.
-
#ssl_context ⇒ Object
SSL context for the server when run in SSL mode.
-
#ssl_servername_callback(sslsocket, hostname = nil) ⇒ Object
ServerNameIndication callback.
-
#start(&block) ⇒ Object
Starts the server and runs the
block
for each connection. -
#stop ⇒ Object
Stops the server from accepting new connections.
Constructor Details
#initialize(config = {}, default = Config::General) ⇒ GenericServer
Creates a new generic server from config
. The default configuration comes from default
.
88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 |
# File 'lib/webrick/server.rb', line 88 def initialize(config={}, default=Config::General) @config = default.dup.update(config) @status = :Stop @config[:Logger] ||= Log::new @logger = @config[:Logger] @tokens = Thread::SizedQueue.new(@config[:MaxClients]) @config[:MaxClients].times{ @tokens.push(nil) } webrickv = WEBrick::VERSION rubyv = "#{RUBY_VERSION} (#{RUBY_RELEASE_DATE}) [#{RUBY_PLATFORM}]" @logger.info("WEBrick #{webrickv}") @logger.info("ruby #{rubyv}") @listeners = [] @shutdown_pipe = nil unless @config[:DoNotListen] if @config[:Listen] warn(":Listen option is deprecated; use GenericServer#listen", uplevel: 1) end listen(@config[:BindAddress], @config[:Port]) if @config[:Port] == 0 @config[:Port] = @listeners[0].addr[1] end end end |
Instance Attribute Details
#config ⇒ Object (readonly)
The server configuration
66 67 68 |
# File 'lib/webrick/server.rb', line 66 def config @config end |
#listeners ⇒ Object (readonly)
Sockets listening for connections.
82 83 84 |
# File 'lib/webrick/server.rb', line 82 def listeners @listeners end |
#logger ⇒ Object (readonly)
The server logger. This is independent from the HTTP access log.
71 72 73 |
# File 'lib/webrick/server.rb', line 71 def logger @logger end |
#status ⇒ Object (readonly)
The server status. One of :Stop, :Running or :Shutdown
61 62 63 |
# File 'lib/webrick/server.rb', line 61 def status @status end |
#tokens ⇒ Object (readonly)
Tokens control the number of outstanding clients. The :MaxClients
configuration sets this.
77 78 79 |
# File 'lib/webrick/server.rb', line 77 def tokens @tokens end |
Instance Method Details
#[](key) ⇒ Object
Retrieves key
from the configuration
118 119 120 |
# File 'lib/webrick/server.rb', line 118 def [](key) @config[key] end |
#listen(address, port) ⇒ Object
Adds listeners from address
and port
to the server. See WEBrick::Utils::create_listeners for details.
164 165 166 167 168 169 170 171 172 173 174 175 |
# File 'lib/webrick/ssl.rb', line 164 def listen(address, port) # :nodoc: listeners = Utils::create_listeners(address, port) if @config[:SSLEnable] listeners.collect!{|svr| ssvr = ::OpenSSL::SSL::SSLServer.new(svr, ssl_context) ssvr.start_immediately = @config[:SSLStartImmediately] ssvr } end @listeners += listeners setup_shutdown_pipe end |
#run(sock) ⇒ Object
You must subclass GenericServer and implement #run which accepts a TCP client socket
241 242 243 |
# File 'lib/webrick/server.rb', line 241 def run(sock) @logger.fatal "run() must be provided by user." end |
#setup_ssl_context(config) ⇒ Object
Sets up an SSL context for config
180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 |
# File 'lib/webrick/ssl.rb', line 180 def setup_ssl_context(config) # :nodoc: unless config[:SSLCertificate] cn = config[:SSLCertName] comment = config[:SSLCertComment] cert, key = Utils::create_self_signed_cert(2048, cn, comment) config[:SSLCertificate] = cert config[:SSLPrivateKey] = key end ctx = OpenSSL::SSL::SSLContext.new ctx.key = config[:SSLPrivateKey] ctx.cert = config[:SSLCertificate] ctx.client_ca = config[:SSLClientCA] ctx.extra_chain_cert = config[:SSLExtraChainCert] ctx.ca_file = config[:SSLCACertificateFile] ctx.ca_path = config[:SSLCACertificatePath] ctx.cert_store = config[:SSLCertificateStore] ctx.tmp_dh_callback = config[:SSLTmpDhCallback] ctx.verify_mode = config[:SSLVerifyClient] ctx.verify_depth = config[:SSLVerifyDepth] ctx.verify_callback = config[:SSLVerifyCallback] ctx.servername_cb = config[:SSLServerNameCallback] || proc { |args| ssl_servername_callback(*args) } ctx.timeout = config[:SSLTimeout] ctx. = config[:SSLOptions] ctx.ciphers = config[:SSLCiphers] ctx end |
#shutdown ⇒ Object
Shuts down the server and all listening sockets. New listeners must be provided to restart the server.
231 232 233 234 235 |
# File 'lib/webrick/server.rb', line 231 def shutdown stop alarm_shutdown_pipe(&:close) end |
#ssl_context ⇒ Object
SSL context for the server when run in SSL mode
149 150 151 152 153 154 155 156 157 |
# File 'lib/webrick/ssl.rb', line 149 def ssl_context # :nodoc: @ssl_context ||= begin if @config[:SSLEnable] ssl_context = setup_ssl_context(@config) @logger.info("\n" + @config[:SSLCertificate].to_text) ssl_context end end end |
#ssl_servername_callback(sslsocket, hostname = nil) ⇒ Object
ServerNameIndication callback
210 211 212 |
# File 'lib/webrick/ssl.rb', line 210 def ssl_servername_callback(sslsocket, hostname = nil) # default end |
#start(&block) ⇒ Object
Starts the server and runs the block
for each connection. This method does not return until the server is stopped from a signal handler or another thread using #stop or #shutdown.
If the block raises a subclass of StandardError the exception is logged and ignored. If an IOError or Errno::EBADF exception is raised the exception is ignored. If an Exception subclass is raised the exception is logged and re-raised which stops the server.
To completely shut down a server call #shutdown from ensure:
server = WEBrick::GenericServer.new
# or WEBrick::HTTPServer.new
begin
server.start
ensure
server.shutdown
end
151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 |
# File 'lib/webrick/server.rb', line 151 def start(&block) raise ServerError, "already started." if @status != :Stop server_type = @config[:ServerType] || SimpleServer setup_shutdown_pipe server_type.start{ @logger.info \ "#{self.class}#start: pid=#{$$} port=#{@config[:Port]}" @status = :Running call_callback(:StartCallback) shutdown_pipe = @shutdown_pipe thgroup = ThreadGroup.new begin while @status == :Running begin sp = shutdown_pipe[0] if svrs = IO.select([sp, *@listeners]) if svrs[0].include? sp # swallow shutdown pipe buf = String.new nil while String === sp.read_nonblock([sp.nread, 8].max, buf, exception: false) break end svrs[0].each{|svr| @tokens.pop # blocks while no token is there. if sock = accept_client(svr) unless config[:DoNotReverseLookup].nil? sock.do_not_reverse_lookup = !!config[:DoNotReverseLookup] end th = start_thread(sock, &block) th[:WEBrickThread] = true thgroup.add(th) else @tokens.push(nil) end } end rescue Errno::EBADF, Errno::ENOTSOCK, IOError => ex # if the listening socket was closed in GenericServer#shutdown, # IO::select raise it. rescue StandardError => ex msg = "#{ex.class}: #{ex.}\n\t#{ex.backtrace[0]}" @logger.error msg rescue Exception => ex @logger.fatal ex raise end end ensure cleanup_shutdown_pipe(shutdown_pipe) cleanup_listener @status = :Shutdown @logger.info "going to shutdown ..." thgroup.list.each{|th| th.join if th[:WEBrickThread] } call_callback(:StopCallback) @logger.info "#{self.class}#start done." @status = :Stop end } end |
#stop ⇒ Object
Stops the server from accepting new connections.
219 220 221 222 223 224 225 |
# File 'lib/webrick/server.rb', line 219 def stop if @status == :Running @status = :Shutdown end alarm_shutdown_pipe {|f| f.write_nonblock("\0")} end |