proxymgr 0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +8 -0
  3. data/.rspec +2 -0
  4. data/.rubocop.yml +11 -0
  5. data/Gemfile +8 -0
  6. data/Gemfile.lock +52 -0
  7. data/README.md +99 -0
  8. data/Rakefile +5 -0
  9. data/bin/proxymgr +74 -0
  10. data/etc/haproxy.cfg.erb +11 -0
  11. data/examples/config.yml +5 -0
  12. data/lib/proxymgr.rb +20 -0
  13. data/lib/proxymgr/callbacks.rb +17 -0
  14. data/lib/proxymgr/config.rb +130 -0
  15. data/lib/proxymgr/haproxy.rb +51 -0
  16. data/lib/proxymgr/haproxy/control.rb +46 -0
  17. data/lib/proxymgr/haproxy/process.rb +107 -0
  18. data/lib/proxymgr/haproxy/server.rb +24 -0
  19. data/lib/proxymgr/haproxy/socket.rb +67 -0
  20. data/lib/proxymgr/haproxy/socket_manager.rb +62 -0
  21. data/lib/proxymgr/haproxy/state.rb +124 -0
  22. data/lib/proxymgr/haproxy/updater.rb +74 -0
  23. data/lib/proxymgr/logging.rb +26 -0
  24. data/lib/proxymgr/platform.rb +16 -0
  25. data/lib/proxymgr/platform/linux.rb +9 -0
  26. data/lib/proxymgr/process_manager.rb +101 -0
  27. data/lib/proxymgr/process_manager/signal_handler.rb +44 -0
  28. data/lib/proxymgr/service_config.rb +12 -0
  29. data/lib/proxymgr/service_config/base.rb +16 -0
  30. data/lib/proxymgr/service_config/zookeeper.rb +33 -0
  31. data/lib/proxymgr/service_manager.rb +53 -0
  32. data/lib/proxymgr/sink.rb +100 -0
  33. data/lib/proxymgr/watcher.rb +9 -0
  34. data/lib/proxymgr/watcher/base.rb +75 -0
  35. data/lib/proxymgr/watcher/campanja_zk.rb +20 -0
  36. data/lib/proxymgr/watcher/dns.rb +36 -0
  37. data/lib/proxymgr/watcher/file.rb +45 -0
  38. data/lib/proxymgr/watcher/zookeeper.rb +61 -0
  39. data/packaging/profile.sh +1 -0
  40. data/packaging/recipe.rb +35 -0
  41. data/proxymgr.gemspec +20 -0
  42. data/spec/spec_helper.rb +23 -0
  43. data/spec/support/dummy_watcher.rb +21 -0
  44. data/spec/support/fake_proxy.rb +15 -0
  45. data/spec/support/fake_zookeeper.rb +170 -0
  46. data/spec/support/mock_servers.rb +7 -0
  47. data/spec/unit/haproxy/socket_manager_spec.rb +40 -0
  48. data/spec/unit/haproxy/updater_spec.rb +123 -0
  49. data/spec/unit/service_manager_spec.rb +49 -0
  50. data/spec/unit/sink_spec.rb +41 -0
  51. data/spec/unit/watcher/base_spec.rb +27 -0
  52. metadata +188 -0
@@ -0,0 +1,7 @@
1
+ module MockServers
2
+ def new_mock_server(backend, name, status = nil)
3
+ ProxyMgr::Haproxy::Server.new(nil, {'pxname' => backend,
4
+ 'svname' => name,
5
+ 'status' => status})
6
+ end
7
+ end
@@ -0,0 +1,40 @@
1
+ require 'spec_helper'
2
+
3
+ describe ProxyMgr::Haproxy::SocketManager, '#update' do
4
+ before(:each) do
5
+ @sm = ProxyMgr::Haproxy::SocketManager.new
6
+ Socket.any_instance.stub(:bind) { true }
7
+ end
8
+
9
+ it 'creates new sockets when none exist' do
10
+ watcher = double(ProxyMgr::Watcher::Base)
11
+ watcher.stub(:port) { 9090 }
12
+ @sm.update({"test" => watcher})
13
+ @sm.sockets[9090].should be_kind_of(Socket)
14
+ end
15
+
16
+ it 'deletes sockets which are no longer part of the state' do
17
+ ports = [9090, 9091]
18
+ watchers = Hash[ports.map do |port|
19
+ watcher = double(ProxyMgr::Watcher::Base)
20
+ watcher.stub(:port) { port }
21
+ ["service_" + port.to_s, watcher]
22
+ end]
23
+ @sm.update(watchers)
24
+ ports.each do |port|
25
+ @sm.sockets[port].should be_kind_of(Socket)
26
+ end
27
+ watchers.delete("service_9090")
28
+ Socket.any_instance.should_receive(:close)
29
+ @sm.update(watchers)
30
+ @sm.sockets[9091].should be_kind_of(Socket)
31
+ @sm.sockets[9090].should == nil
32
+ end
33
+
34
+ it 'sets SO_REUSEADDR on sockets' do
35
+ watcher = double(ProxyMgr::Watcher::Base)
36
+ watcher.stub(:port) { 9090 }
37
+ Socket.any_instance.should_receive(:setsockopt).with(::Socket::SOL_SOCKET, ::Socket::SO_REUSEADDR, 1)
38
+ @sm.update("test" => watcher)
39
+ end
40
+ end
@@ -0,0 +1,123 @@
1
+ require 'spec_helper'
2
+
3
+ describe ProxyMgr::Haproxy::Updater, '#produce_changeset' do
4
+ before do
5
+ @socket = double(ProxyMgr::Haproxy::Socket)
6
+ @sm = double(ProxyMgr::ServiceManager)
7
+ @updater = ProxyMgr::Haproxy::Updater.new(@socket)
8
+ end
9
+
10
+ context 'stats socket not available' do
11
+ before do
12
+ @socket.stub(:connected?) { false }
13
+ end
14
+
15
+ it 'should require restarts if no stats socket is available' do
16
+ @socket.stub(:connected?) { false }
17
+ changeset = @updater.produce_changeset(nil)
18
+ changeset.restart_needed?.should == true
19
+ end
20
+ end
21
+
22
+ context 'stats socket available' do
23
+ before do
24
+ @socket.stub(:connected?) { true }
25
+ end
26
+
27
+ it 'requires restarts if a new server is provided' do
28
+ @socket.should_receive(:servers) do
29
+ ['a', 'b'].map { |name| new_mock_server('dummy', 'name') }
30
+ end
31
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm)
32
+ watcher.servers = ['a', 'b', 'c']
33
+ changeset = @updater.produce_changeset('dummy' => watcher)
34
+ changeset.restart_needed?.should == true
35
+ end
36
+
37
+ it 'disables servers if they are already added' do
38
+ @socket.should_receive(:servers) { [] }
39
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm)
40
+ watcher.servers = ['a', 'b', 'c']
41
+ @updater.produce_changeset('dummy' => watcher)
42
+ @socket.should_receive(:servers) do
43
+ ['a', 'b', 'c'].map { |name| new_mock_server('dummy', name) }
44
+ end
45
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm)
46
+ watcher.servers = ['a', 'b']
47
+ changeset = @updater.produce_changeset('dummy' => watcher)
48
+ changeset.disable.should == {'dummy' => ['c']}
49
+ changeset.restart_needed?.should == false
50
+ end
51
+
52
+ it 'enables servers if they are down and added' do
53
+ @socket.should_receive(:servers) { [] }
54
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm)
55
+ watcher.servers = []
56
+ @updater.produce_changeset('dummy' => watcher)
57
+ @socket.should_receive(:servers) do
58
+ s = ['a', 'b'].map { |name| new_mock_server('dummy', name) }
59
+ s << new_mock_server('dummy', 'c', 'MAINT')
60
+ end
61
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm)
62
+ watcher.servers = ['a', 'b', 'c']
63
+ changeset = @updater.produce_changeset('dummy' => watcher)
64
+ changeset.enable.should == {'dummy' => ['c']}
65
+ changeset.restart_needed?.should == false
66
+ end
67
+
68
+ it 'requires restart if a backend is added' do
69
+ @socket.should_receive(:servers) { [] }
70
+ watcher = double(ProxyMgr::Watcher::Base)
71
+ watcher.should_receive(:servers) { ['a', 'b'] }
72
+ changeset = @updater.produce_changeset('dummy' => watcher)
73
+ changeset.restart_needed?.should == true
74
+ end
75
+
76
+ it 'requires a restart if a server has not already been added/is disabled' do
77
+ @socket.stub(:servers) do
78
+ ['a', 'b'].map do |name|
79
+ new_mock_server('dummy', name)
80
+ end
81
+ end
82
+ watcher = double(ProxyMgr::Watcher::Base)
83
+ watcher.should_receive(:servers) { ['a', 'b'] }
84
+ changeset = @updater.produce_changeset('dummy' => watcher)
85
+ watcher.should_receive(:servers) { ['a', 'b', 'c'] }
86
+ changeset = @updater.produce_changeset('dummy' => watcher)
87
+ changeset.restart_needed?.should == true
88
+ end
89
+
90
+ it 'requires restart if a backend is deleted' do
91
+ @socket.stub(:servers) do
92
+ ['a', 'b'].map do |name|
93
+ new_mock_server('dummy', name)
94
+ end
95
+ ['c', 'd'].map do |name|
96
+ new_mock_server('dummy2', name)
97
+ end
98
+ end
99
+ watcher1 = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm).tap do |w|
100
+ w.servers = ['a', 'b']
101
+ end
102
+ watcher2 = ProxyMgr::Watcher::Dummy.new('dummy2', {}, @sm).tap do |w|
103
+ w.servers = ['c', 'd']
104
+ end
105
+ @updater.produce_changeset('dummy' => watcher1,
106
+ 'dummy2' => watcher2)
107
+ changeset = @updater.produce_changeset('dummy' => watcher1)
108
+ changeset.restart_needed?.should == true
109
+ end
110
+
111
+ it 'requires restarting if listen_options is changed' do
112
+ @socket.should_receive(:servers) { [] }
113
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {}, @sm)
114
+ watcher.servers = ['a']
115
+ @updater.produce_changeset('dummy' => watcher)
116
+ @socket.should_receive(:servers) { [new_mock_server('dummy', 'a')] }
117
+ watcher = ProxyMgr::Watcher::Dummy.new('dummy', {'listen_options' => ['test option']}, @sm)
118
+ watcher.servers = ['a']
119
+ changeset = @updater.produce_changeset('dummy' => watcher)
120
+ changeset.restart_needed?.should == true
121
+ end
122
+ end
123
+ end
@@ -0,0 +1,49 @@
1
+ require 'spec_helper'
2
+
3
+ describe ProxyMgr::ServiceManager do
4
+ before(:each) do
5
+ @sink = double(ProxyMgr::Sink)
6
+ @manager = ProxyMgr::ServiceManager.new(@sink)
7
+ end
8
+
9
+ it 'adds new services' do
10
+ expect(ProxyMgr::Watcher::Dummy).to receive(:new).with('dummy', {'config' => 'thing'}, @manager)
11
+ @manager.update_service('dummy', {'type' => 'dummy', 'config' => 'thing'})
12
+ end
13
+
14
+ it 'starts services if valid' do
15
+ expect_any_instance_of(ProxyMgr::Watcher::Dummy).to receive(:watch)
16
+ ProxyMgr::Watcher::Dummy.any_instance.stub(:valid?) { true }
17
+ @manager.update_service('dummy', {'type' => 'dummy', 'config' => 'thing'})
18
+ end
19
+
20
+ it 'shuts down old services when replaced' do
21
+ @manager.update_service('dummy', {'type' => "dummy"})
22
+ expect_any_instance_of(ProxyMgr::Watcher::Dummy).to receive(:shutdown)
23
+ @manager.update_service('dummy', {'type' => "dummy"})
24
+ end
25
+
26
+ it 'shuts down serviecs when deleted and notifies sink' do
27
+ @sink.should_receive(:update_backends)
28
+ @manager.update_service('dummy', {'type' => "dummy"})
29
+ expect_any_instance_of(ProxyMgr::Watcher::Dummy).to receive(:shutdown)
30
+ @manager.delete_service('dummy')
31
+ end
32
+
33
+ it 'notifies sink when watchers are updated' do
34
+ @manager.update_service('dummy', {'type' => 'dummy'})
35
+ @sink.should_receive(:update_backends)
36
+ @manager.update_backends
37
+ end
38
+
39
+ it 'shuts down all services and sink when shut down' do
40
+ @manager.update_service('dummy', {'type' => 'dummy'})
41
+ expect_any_instance_of(ProxyMgr::Watcher::Dummy).to receive(:shutdown)
42
+ @sink.should_receive(:shutdown)
43
+ @manager.shutdown
44
+ end
45
+
46
+ it 'does not create services for non-existing watcher types' do
47
+ @manager.update_service('dummy', {'type' => 'NonExisting'}).should == nil
48
+ end
49
+ end
@@ -0,0 +1,41 @@
1
+ require 'spec_helper'
2
+
3
+ describe ProxyMgr::Sink do
4
+ require 'absolute_time'
5
+
6
+ before(:each) do
7
+ @manager = double(ProxyMgr::ServiceManager)
8
+ @watcher = ProxyMgr::Watcher::Dummy.new('test', {'port' => 1}, @manager)
9
+ @watcher.servers = ['a', 'b']
10
+ @mutex = Mutex.new
11
+ @cv = ConditionVariable.new
12
+ end
13
+
14
+ it 'calls update_backends after a delay' do
15
+ t1 = AbsoluteTime.now
16
+ elapsed = nil
17
+ proxy = ProxyMgr::FakeProxy.new do
18
+ elapsed = AbsoluteTime.now-t1
19
+ @mutex.synchronize { @cv.signal }
20
+ end
21
+ sink = ProxyMgr::Sink.new(proxy, :default_timeout => 2)
22
+ sink.update_backends('test' => @watcher)
23
+ @mutex.synchronize { @cv.wait(@mutex) }
24
+ elapsed.should >= 2
25
+ end
26
+
27
+ it 'calls update_backends with increasing delay if interrupted' do
28
+ t1 = AbsoluteTime.now
29
+ elapsed = nil
30
+ proxy = ProxyMgr::FakeProxy.new do
31
+ elapsed = AbsoluteTime.now-t1
32
+ @mutex.synchronize { @cv.signal }
33
+ end
34
+ sink = ProxyMgr::Sink.new(proxy, :default_timeout => 2)
35
+ sink.update_backends('test' => @watcher)
36
+ sleep(1) # this is a bit of a hack. but the semantics don't lend themselves to testing very well.
37
+ sink.update_backends('test' => @watcher, 'test2' => @watcher)
38
+ @mutex.synchronize { @cv.wait(@mutex) }
39
+ elapsed.should >= 4
40
+ end
41
+ end
@@ -0,0 +1,27 @@
1
+ require 'spec_helper'
2
+
3
+ describe ProxyMgr::Watcher::Base do
4
+ before(:each) do
5
+ @sm = double(ProxyMgr::ServiceManager)
6
+ end
7
+
8
+ it 'is valid if port is correctly specified' do
9
+ watcher = ProxyMgr::Watcher::Dummy.new('test', {'port' => 8080}, @sm)
10
+ watcher.valid?.should == true
11
+ end
12
+
13
+ it 'does not start if port is not an integer' do
14
+ watcher = ProxyMgr::Watcher::Dummy.new('test', {'port' => 'false'}, @sm)
15
+ watcher.valid?.should == false
16
+ end
17
+
18
+ it 'does not start if port is invalid' do
19
+ watcher = ProxyMgr::Watcher::Dummy.new('test', {'port' => 0}, @sm)
20
+ watcher.valid?.should == false
21
+ end
22
+
23
+ it 'is valid if listen_options specified and is an array' do
24
+ watcher = ProxyMgr::Watcher::Dummy.new('test', {'port' => 65535, 'listen_options' => ['a config option', 'another']}, @sm)
25
+ watcher.valid?.should == true
26
+ end
27
+ end
metadata ADDED
@@ -0,0 +1,188 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: proxymgr
3
+ version: !ruby/object:Gem::Version
4
+ version: '0.1'
5
+ platform: ruby
6
+ authors:
7
+ - Torbjörn Norinder
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-09-19 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: docopt
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: 0.5.0
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: 0.5.0
27
+ - !ruby/object:Gem::Dependency
28
+ name: zoology
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: absolute_time
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: yajl-ruby
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :runtime
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ">="
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: zookeeper
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :runtime
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: state_machine
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - ">="
88
+ - !ruby/object:Gem::Version
89
+ version: '0'
90
+ type: :runtime
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - ">="
95
+ - !ruby/object:Gem::Version
96
+ version: '0'
97
+ description: Manages Haproxy configuration dynamically
98
+ email:
99
+ - torbjorn@genunix.se
100
+ executables:
101
+ - proxymgr
102
+ extensions: []
103
+ extra_rdoc_files: []
104
+ files:
105
+ - ".gitignore"
106
+ - ".rspec"
107
+ - ".rubocop.yml"
108
+ - Gemfile
109
+ - Gemfile.lock
110
+ - README.md
111
+ - Rakefile
112
+ - bin/proxymgr
113
+ - etc/haproxy.cfg.erb
114
+ - examples/config.yml
115
+ - lib/proxymgr.rb
116
+ - lib/proxymgr/callbacks.rb
117
+ - lib/proxymgr/config.rb
118
+ - lib/proxymgr/haproxy.rb
119
+ - lib/proxymgr/haproxy/control.rb
120
+ - lib/proxymgr/haproxy/process.rb
121
+ - lib/proxymgr/haproxy/server.rb
122
+ - lib/proxymgr/haproxy/socket.rb
123
+ - lib/proxymgr/haproxy/socket_manager.rb
124
+ - lib/proxymgr/haproxy/state.rb
125
+ - lib/proxymgr/haproxy/updater.rb
126
+ - lib/proxymgr/logging.rb
127
+ - lib/proxymgr/platform.rb
128
+ - lib/proxymgr/platform/linux.rb
129
+ - lib/proxymgr/process_manager.rb
130
+ - lib/proxymgr/process_manager/signal_handler.rb
131
+ - lib/proxymgr/service_config.rb
132
+ - lib/proxymgr/service_config/base.rb
133
+ - lib/proxymgr/service_config/zookeeper.rb
134
+ - lib/proxymgr/service_manager.rb
135
+ - lib/proxymgr/sink.rb
136
+ - lib/proxymgr/watcher.rb
137
+ - lib/proxymgr/watcher/base.rb
138
+ - lib/proxymgr/watcher/campanja_zk.rb
139
+ - lib/proxymgr/watcher/dns.rb
140
+ - lib/proxymgr/watcher/file.rb
141
+ - lib/proxymgr/watcher/zookeeper.rb
142
+ - packaging/profile.sh
143
+ - packaging/recipe.rb
144
+ - proxymgr.gemspec
145
+ - spec/spec_helper.rb
146
+ - spec/support/dummy_watcher.rb
147
+ - spec/support/fake_proxy.rb
148
+ - spec/support/fake_zookeeper.rb
149
+ - spec/support/mock_servers.rb
150
+ - spec/unit/haproxy/socket_manager_spec.rb
151
+ - spec/unit/haproxy/updater_spec.rb
152
+ - spec/unit/service_manager_spec.rb
153
+ - spec/unit/sink_spec.rb
154
+ - spec/unit/watcher/base_spec.rb
155
+ homepage: https://github.com/campanja/proxymgr
156
+ licenses: []
157
+ metadata: {}
158
+ post_install_message:
159
+ rdoc_options: []
160
+ require_paths:
161
+ - lib
162
+ required_ruby_version: !ruby/object:Gem::Requirement
163
+ requirements:
164
+ - - ">="
165
+ - !ruby/object:Gem::Version
166
+ version: '0'
167
+ required_rubygems_version: !ruby/object:Gem::Requirement
168
+ requirements:
169
+ - - ">="
170
+ - !ruby/object:Gem::Version
171
+ version: '0'
172
+ requirements: []
173
+ rubyforge_project:
174
+ rubygems_version: 2.2.2
175
+ signing_key:
176
+ specification_version: 4
177
+ summary: Manages Haproxy configuration dynamically
178
+ test_files:
179
+ - spec/spec_helper.rb
180
+ - spec/support/dummy_watcher.rb
181
+ - spec/support/fake_proxy.rb
182
+ - spec/support/fake_zookeeper.rb
183
+ - spec/support/mock_servers.rb
184
+ - spec/unit/haproxy/socket_manager_spec.rb
185
+ - spec/unit/haproxy/updater_spec.rb
186
+ - spec/unit/service_manager_spec.rb
187
+ - spec/unit/sink_spec.rb
188
+ - spec/unit/watcher/base_spec.rb