summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authordanielsdeleo <dan@getchef.com>2015-03-27 17:46:05 -0700
committerdanielsdeleo <dan@getchef.com>2015-03-27 17:46:05 -0700
commit5b056eef1803c655782b5a2a5ce8caee67ffd07e (patch)
treed2f84dbd170e6d7275290d7781324680c06d8cfa
parentd9fe481bf29f7325a0128fa53777df57f1951359 (diff)
downloadchef-zero-5b056eef1803c655782b5a2a5ce8caee67ffd07e.tar.gz
Initial implementation of socketless requests
-rw-r--r--lib/chef_zero/rest_base.rb3
-rw-r--r--lib/chef_zero/server.rb71
2 files changed, 71 insertions, 3 deletions
diff --git a/lib/chef_zero/rest_base.rb b/lib/chef_zero/rest_base.rb
index 8156e7c..48d423a 100644
--- a/lib/chef_zero/rest_base.rb
+++ b/lib/chef_zero/rest_base.rb
@@ -195,8 +195,9 @@ module ChefZero
# Strip off /organizations/chef if we are in single org mode
if rest_path[0..1] != [ 'organizations', server.options[:single_org] ]
raise "Unexpected URL #{rest_path[0..1]} passed to build_uri in single org mode"
+ else
+ "#{base_uri}/#{rest_path[2..-1].join('/')}"
end
- "#{base_uri}/#{rest_path[2..-1].join('/')}"
else
"#{base_uri}/#{rest_path.join('/')}"
end
diff --git a/lib/chef_zero/server.rb b/lib/chef_zero/server.rb
index d78daf5..290cce8 100644
--- a/lib/chef_zero/server.rb
+++ b/lib/chef_zero/server.rb
@@ -86,7 +86,55 @@ require 'chef_zero/endpoints/not_found_endpoint'
require 'chef_zero/endpoints/version_endpoint'
module ChefZero
+ # TODO: make its own file
+ class SocketlessServerMap
+
+ def self.request(port, request_env)
+ instance.request(port, request_env)
+ end
+
+ MUTEX = Mutex.new
+
+ include Singleton
+
+ def initialize()
+ @servers_by_port = {}
+ end
+
+ def register_port(port, server)
+ MUTEX.synchronize do
+ @servers_by_port[port] = server
+ end
+ end
+
+ def register_no_listen_server(server)
+ MUTEX.synchronize do
+ 1.upto(1000) do |port|
+ unless @servers_by_port.key?(port)
+ @servers_by_port[port] = server
+ return i
+ end
+ end
+ raise "No socketless ports left to register"
+ end
+ end
+
+ def deregister(port)
+ MUTEX.synchronize do
+ @servers_by_port.delete(port)
+ end
+ end
+
+ def request(port, request_env)
+ server = @servers_by_port[port]
+ raise "No socketless chef-zero server on given port #{port.inspect}" unless server
+ server.handle_socketless_request(request_env)
+ end
+
+ end
+
class Server
+
DEFAULT_OPTIONS = {
:host => '127.0.0.1',
:port => 8889,
@@ -108,6 +156,7 @@ module ChefZero
end
@options.freeze
ChefZero::Log.level = @options[:log_level].to_sym
+ @app = nil
end
# @return [Hash]
@@ -146,6 +195,12 @@ module ChefZero
end
end
+ def local_mode_url
+ raise "Port not yet set, cannot generate URL" unless port.kind_of?(Integer)
+ "chefzero://localhost:#{port}"
+ end
+
+
#
# The data store for this server (default is in-memory).
#
@@ -224,7 +279,6 @@ module ChefZero
thread.join
end
-
#
# Start a Chef Zero server in a forked process. This method returns the PID
# to the forked process.
@@ -284,9 +338,19 @@ module ChefZero
sleep(0.01)
end
+ SocketlessServerMap.instance.register_port(@port, self)
+
@thread
end
+ def start_socketless
+ @port = SocketlessServerMap.instance.register_no_listen_server(self)
+ end
+
+ def handle_socketless_request(request_env)
+ app.call(request_env)
+ end
+
#
# Boolean method to determine if the server is currently ready to accept
# requests. This method will attempt to make an HTTP request against the
@@ -315,6 +379,7 @@ module ChefZero
if @thread
ChefZero::Log.error("Chef Zero did not stop within #{wait} seconds! Killing...")
@thread.kill
+ SocketlessServerMap.deregister(port)
end
ensure
@server = nil
@@ -545,6 +610,7 @@ module ChefZero
end
def app
+ return @app if @app
router = RestRouter.new(open_source_endpoints)
router.not_found = NotFoundEndpoint.new
@@ -553,7 +619,7 @@ module ChefZero
else
rest_base_prefix = []
end
- return proc do |env|
+ @app = proc do |env|
begin
prefix = global_endpoint?(env['PATH_INFO']) ? [] : rest_base_prefix
@@ -591,6 +657,7 @@ module ChefZero
end
end
end
+ @app
end
def dejsonize_children(hash)