firenxis-god 0.11.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (123) hide show
  1. data/Announce.txt +135 -0
  2. data/History.txt +393 -0
  3. data/README.txt +59 -0
  4. data/Rakefile +142 -0
  5. data/bin/god +132 -0
  6. data/ext/god/.gitignore +5 -0
  7. data/ext/god/extconf.rb +55 -0
  8. data/ext/god/kqueue_handler.c +125 -0
  9. data/ext/god/netlink_handler.c +168 -0
  10. data/god.gemspec +164 -0
  11. data/lib/god.rb +701 -0
  12. data/lib/god/behavior.rb +52 -0
  13. data/lib/god/behaviors/clean_pid_file.rb +21 -0
  14. data/lib/god/behaviors/clean_unix_socket.rb +21 -0
  15. data/lib/god/behaviors/notify_when_flapping.rb +51 -0
  16. data/lib/god/cli/command.rb +256 -0
  17. data/lib/god/cli/run.rb +172 -0
  18. data/lib/god/cli/version.rb +23 -0
  19. data/lib/god/compat19.rb +36 -0
  20. data/lib/god/condition.rb +96 -0
  21. data/lib/god/conditions/always.rb +23 -0
  22. data/lib/god/conditions/complex.rb +86 -0
  23. data/lib/god/conditions/cpu_usage.rb +80 -0
  24. data/lib/god/conditions/degrading_lambda.rb +52 -0
  25. data/lib/god/conditions/disk_usage.rb +32 -0
  26. data/lib/god/conditions/file_mtime.rb +28 -0
  27. data/lib/god/conditions/flapping.rb +128 -0
  28. data/lib/god/conditions/http_response_code.rb +168 -0
  29. data/lib/god/conditions/lambda.rb +25 -0
  30. data/lib/god/conditions/memory_usage.rb +82 -0
  31. data/lib/god/conditions/process_exits.rb +72 -0
  32. data/lib/god/conditions/process_running.rb +74 -0
  33. data/lib/god/conditions/tries.rb +44 -0
  34. data/lib/god/configurable.rb +57 -0
  35. data/lib/god/contact.rb +114 -0
  36. data/lib/god/contacts/campfire.rb +121 -0
  37. data/lib/god/contacts/email.rb +136 -0
  38. data/lib/god/contacts/jabber.rb +75 -0
  39. data/lib/god/contacts/prowl.rb +57 -0
  40. data/lib/god/contacts/scout.rb +55 -0
  41. data/lib/god/contacts/twitter.rb +51 -0
  42. data/lib/god/contacts/webhook.rb +73 -0
  43. data/lib/god/dependency_graph.rb +41 -0
  44. data/lib/god/diagnostics.rb +37 -0
  45. data/lib/god/driver.rb +206 -0
  46. data/lib/god/errors.rb +24 -0
  47. data/lib/god/event_handler.rb +108 -0
  48. data/lib/god/event_handlers/dummy_handler.rb +13 -0
  49. data/lib/god/event_handlers/kqueue_handler.rb +17 -0
  50. data/lib/god/event_handlers/netlink_handler.rb +13 -0
  51. data/lib/god/logger.rb +109 -0
  52. data/lib/god/metric.rb +59 -0
  53. data/lib/god/process.rb +363 -0
  54. data/lib/god/registry.rb +32 -0
  55. data/lib/god/simple_logger.rb +59 -0
  56. data/lib/god/socket.rb +107 -0
  57. data/lib/god/sugar.rb +47 -0
  58. data/lib/god/sys_logger.rb +45 -0
  59. data/lib/god/system/portable_poller.rb +42 -0
  60. data/lib/god/system/process.rb +50 -0
  61. data/lib/god/system/slash_proc_poller.rb +92 -0
  62. data/lib/god/task.rb +503 -0
  63. data/lib/god/timeline.rb +25 -0
  64. data/lib/god/trigger.rb +43 -0
  65. data/lib/god/watch.rb +188 -0
  66. data/test/configs/child_events/child_events.god +44 -0
  67. data/test/configs/child_events/simple_server.rb +3 -0
  68. data/test/configs/child_polls/child_polls.god +37 -0
  69. data/test/configs/child_polls/simple_server.rb +12 -0
  70. data/test/configs/complex/complex.god +59 -0
  71. data/test/configs/complex/simple_server.rb +3 -0
  72. data/test/configs/contact/contact.god +108 -0
  73. data/test/configs/contact/simple_server.rb +3 -0
  74. data/test/configs/daemon_events/daemon_events.god +37 -0
  75. data/test/configs/daemon_events/simple_server.rb +8 -0
  76. data/test/configs/daemon_events/simple_server_stop.rb +11 -0
  77. data/test/configs/daemon_polls/daemon_polls.god +17 -0
  78. data/test/configs/daemon_polls/simple_server.rb +6 -0
  79. data/test/configs/degrading_lambda/degrading_lambda.god +31 -0
  80. data/test/configs/degrading_lambda/tcp_server.rb +15 -0
  81. data/test/configs/lifecycle/lifecycle.god +25 -0
  82. data/test/configs/matias/matias.god +50 -0
  83. data/test/configs/real.rb +59 -0
  84. data/test/configs/running_load/running_load.god +16 -0
  85. data/test/configs/stop_options/simple_server.rb +12 -0
  86. data/test/configs/stop_options/stop_options.god +39 -0
  87. data/test/configs/stress/simple_server.rb +3 -0
  88. data/test/configs/stress/stress.god +15 -0
  89. data/test/configs/task/logs/.placeholder +0 -0
  90. data/test/configs/task/task.god +26 -0
  91. data/test/configs/test.rb +61 -0
  92. data/test/helper.rb +141 -0
  93. data/test/suite.rb +6 -0
  94. data/test/test_behavior.rb +18 -0
  95. data/test/test_campfire.rb +23 -0
  96. data/test/test_condition.rb +50 -0
  97. data/test/test_conditions_disk_usage.rb +50 -0
  98. data/test/test_conditions_http_response_code.rb +109 -0
  99. data/test/test_conditions_process_running.rb +40 -0
  100. data/test/test_conditions_tries.rb +67 -0
  101. data/test/test_contact.rb +109 -0
  102. data/test/test_dependency_graph.rb +62 -0
  103. data/test/test_driver.rb +11 -0
  104. data/test/test_email.rb +34 -0
  105. data/test/test_event_handler.rb +80 -0
  106. data/test/test_god.rb +570 -0
  107. data/test/test_handlers_kqueue_handler.rb +16 -0
  108. data/test/test_jabber.rb +29 -0
  109. data/test/test_logger.rb +55 -0
  110. data/test/test_metric.rb +72 -0
  111. data/test/test_process.rb +247 -0
  112. data/test/test_prowl.rb +15 -0
  113. data/test/test_registry.rb +15 -0
  114. data/test/test_socket.rb +34 -0
  115. data/test/test_sugar.rb +42 -0
  116. data/test/test_system_portable_poller.rb +17 -0
  117. data/test/test_system_process.rb +30 -0
  118. data/test/test_task.rb +246 -0
  119. data/test/test_timeline.rb +37 -0
  120. data/test/test_trigger.rb +59 -0
  121. data/test/test_watch.rb +279 -0
  122. data/test/test_webhook.rb +15 -0
  123. metadata +362 -0
data/test/helper.rb ADDED
@@ -0,0 +1,141 @@
1
+ require 'rubygems'
2
+ require File.join(File.dirname(__FILE__), *%w[.. lib god sys_logger])
3
+ require File.join(File.dirname(__FILE__), *%w[.. lib god])
4
+ God::EventHandler.load
5
+
6
+ require 'test/unit'
7
+ require 'set'
8
+
9
+ include God
10
+
11
+ if Process.uid != 0
12
+ abort <<-EOF
13
+ \n
14
+ *********************************************************************
15
+ * *
16
+ * You need to run these tests as root *
17
+ * chroot and netlink (linux only) require it *
18
+ * *
19
+ *********************************************************************
20
+ EOF
21
+ end
22
+
23
+ begin
24
+ require 'mocha'
25
+ rescue LoadError
26
+ unless gems ||= false
27
+ require 'rubygems'
28
+ gems = true
29
+ retry
30
+ else
31
+ abort "=> You need the Mocha gem to run these tests."
32
+ end
33
+ end
34
+
35
+ module God
36
+ module Conditions
37
+ class FakeCondition < Condition
38
+ def test
39
+ true
40
+ end
41
+ end
42
+
43
+ class FakePollCondition < PollCondition
44
+ def test
45
+ true
46
+ end
47
+ end
48
+
49
+ class FakeEventCondition < EventCondition
50
+ def register
51
+ end
52
+ def deregister
53
+ end
54
+ end
55
+ end
56
+
57
+ module Behaviors
58
+ class FakeBehavior < Behavior
59
+ def before_start
60
+ 'foo'
61
+ end
62
+ def after_start
63
+ 'bar'
64
+ end
65
+ end
66
+ end
67
+
68
+ module Contacts
69
+ class FakeContact < Contact
70
+ end
71
+
72
+ class InvalidContact
73
+ end
74
+ end
75
+
76
+ def self.reset
77
+ self.watches = nil
78
+ self.groups = nil
79
+ self.server = nil
80
+ self.inited = nil
81
+ self.host = nil
82
+ self.port = nil
83
+ self.pid_file_directory = nil
84
+ self.registry.reset
85
+ end
86
+ end
87
+
88
+ def silence_warnings
89
+ old_verbose, $VERBOSE = $VERBOSE, nil
90
+ yield
91
+ ensure
92
+ $VERBOSE = old_verbose
93
+ end
94
+
95
+ LOG.instance_variable_set(:@io, StringIO.new('/dev/null'))
96
+
97
+ module Kernel
98
+ def abort(text)
99
+ raise SystemExit
100
+ end
101
+ def exit(code)
102
+ raise SystemExit
103
+ end
104
+ end
105
+
106
+ module Test::Unit::Assertions
107
+ def assert_abort
108
+ assert_raise SystemExit do
109
+ yield
110
+ end
111
+ end
112
+ end
113
+
114
+ # This allows you to be a good OOP citizen and honor encapsulation, but
115
+ # still make calls to private methods (for testing) by doing
116
+ #
117
+ # obj.bypass.private_thingie(arg1, arg2)
118
+ #
119
+ # Which is easier on the eye than
120
+ #
121
+ # obj.send(:private_thingie, arg1, arg2)
122
+ #
123
+ class Object
124
+ class Bypass
125
+ instance_methods.each do |m|
126
+ undef_method m unless m =~ /^(__|object_id)/
127
+ end
128
+
129
+ def initialize(ref)
130
+ @ref = ref
131
+ end
132
+
133
+ def method_missing(sym, *args)
134
+ @ref.__send__(sym, *args)
135
+ end
136
+ end
137
+
138
+ def bypass
139
+ Bypass.new(self)
140
+ end
141
+ end
data/test/suite.rb ADDED
@@ -0,0 +1,6 @@
1
+ require 'test/unit'
2
+
3
+ tests = Dir["#{File.dirname(__FILE__)}/test_*.rb"]
4
+ tests.each do |file|
5
+ require file
6
+ end
@@ -0,0 +1,18 @@
1
+ require File.dirname(__FILE__) + '/helper'
2
+
3
+ class TestBehavior < Test::Unit::TestCase
4
+ def test_generate_should_return_an_object_corresponding_to_the_given_type
5
+ assert_equal Behaviors::FakeBehavior, Behavior.generate(:fake_behavior, nil).class
6
+ end
7
+
8
+ def test_generate_should_raise_on_invalid_type
9
+ assert_raise NoSuchBehaviorError do
10
+ Behavior.generate(:foo, nil)
11
+ end
12
+ end
13
+
14
+ def test_complain
15
+ SysLogger.expects(:log).with(:error, 'foo')
16
+ assert !Behavior.allocate.bypass.complain('foo')
17
+ end
18
+ end
@@ -0,0 +1,23 @@
1
+ require File.dirname(__FILE__) + '/helper'
2
+ require 'tinder'
3
+
4
+ class TestCampfire < Test::Unit::TestCase
5
+ def setup
6
+ @campfire = God::Contacts::Campfire.new
7
+ end
8
+
9
+ def test_exists
10
+ God::Contacts::Campfire
11
+ end
12
+
13
+ def test_notify
14
+ @campfire.subdomain = 'github'
15
+ @campfire.token = 'abc'
16
+ @campfire.room = 'danger'
17
+
18
+ time = Time.now
19
+ body = "[#{time.strftime('%H:%M:%S')}] host - msg"
20
+ Marshmallow::Connection.any_instance.expects(:speak).with('danger', body)
21
+ @campfire.notify('msg', time, 'prio', 'cat', 'host')
22
+ end
23
+ end
@@ -0,0 +1,50 @@
1
+ require File.dirname(__FILE__) + '/helper'
2
+
3
+ class BadlyImplementedCondition < PollCondition
4
+ end
5
+
6
+ class TestCondition < Test::Unit::TestCase
7
+
8
+ # generate
9
+
10
+ def test_generate_should_return_an_object_corresponding_to_the_given_type
11
+ assert_equal Conditions::ProcessRunning, Condition.generate(:process_running, nil).class
12
+ end
13
+
14
+ def test_generate_should_raise_on_invalid_type
15
+ assert_raise NoSuchConditionError do
16
+ Condition.generate(:foo, nil)
17
+ end
18
+ end
19
+
20
+ def test_generate_should_abort_on_event_condition_without_loaded_event_system
21
+ God::EventHandler.stubs(:operational?).returns(false)
22
+ assert_abort do
23
+ God::EventHandler.start
24
+ Condition.generate(:process_exits, nil).class
25
+ end
26
+ end
27
+
28
+ def test_generate_should_return_a_good_error_message_for_invalid_types
29
+ emsg = "No Condition found with the class name God::Conditions::FooBar"
30
+ rmsg = nil
31
+
32
+ begin
33
+ Condition.generate(:foo_bar, nil)
34
+ rescue => e
35
+ rmsg = e.message
36
+ end
37
+
38
+ assert_equal emsg, rmsg
39
+ end
40
+
41
+ # test
42
+
43
+ def test_test_should_raise_if_not_defined_in_subclass
44
+ c = BadlyImplementedCondition.new
45
+
46
+ assert_raise AbstractMethodNotOverriddenError do
47
+ c.test
48
+ end
49
+ end
50
+ end
@@ -0,0 +1,50 @@
1
+ require File.dirname(__FILE__) + '/helper'
2
+
3
+ class TestConditionsDiskUsage < Test::Unit::TestCase
4
+ # valid?
5
+
6
+ def test_valid_should_return_false_if_no_above_given
7
+ c = Conditions::DiskUsage.new
8
+ c.mount_point = '/'
9
+ c.watch = stub(:name => 'foo')
10
+ assert_equal false, c.valid?
11
+ end
12
+
13
+ def test_valid_should_return_false_if_no_mount_point_given
14
+ c = Conditions::DiskUsage.new
15
+ c.above = 90
16
+ c.watch = stub(:name => 'foo')
17
+ assert_equal false, c.valid?
18
+ end
19
+
20
+ def test_valid_should_return_true_if_required_options_all_set
21
+ c = Conditions::DiskUsage.new
22
+ c.above = 90
23
+ c.mount_point = '/'
24
+ c.watch = stub(:name => 'foo')
25
+
26
+ assert_equal true, c.valid?
27
+ end
28
+
29
+ # test
30
+
31
+ def test_test_should_return_true_if_above_limit
32
+ c = Conditions::DiskUsage.new
33
+ c.above = 90
34
+ c.mount_point = '/'
35
+
36
+ c.expects(:`).returns('91')
37
+
38
+ assert_equal true, c.test
39
+ end
40
+
41
+ def test_test_should_return_false_if_below_limit
42
+ c = Conditions::DiskUsage.new
43
+ c.above = 90
44
+ c.mount_point = '/'
45
+
46
+ c.expects(:`).returns('90')
47
+
48
+ assert_equal false, c.test
49
+ end
50
+ end
@@ -0,0 +1,109 @@
1
+ require File.dirname(__FILE__) + '/helper'
2
+
3
+ class TestHttpResponseCode < Test::Unit::TestCase
4
+ def valid_condition
5
+ c = Conditions::HttpResponseCode.new()
6
+ c.watch = stub(:name => 'foo')
7
+ c.host = 'localhost'
8
+ c.port = 8080
9
+ c.path = '/'
10
+ c.timeout = 10
11
+ c.code_is = 200
12
+ c.times = 1
13
+ yield(c) if block_given?
14
+ c.prepare
15
+ c
16
+ end
17
+
18
+ # valid?
19
+
20
+ def test_valid_condition_is_valid
21
+ c = valid_condition
22
+ assert c.valid?
23
+ end
24
+
25
+ def test_valid_should_return_false_if_both_code_is_and_code_is_not_are_set
26
+ c = valid_condition do |cc|
27
+ cc.code_is_not = 500
28
+ end
29
+ assert !c.valid?
30
+ end
31
+
32
+ def test_valid_should_return_false_if_no_host_set
33
+ c = valid_condition do |cc|
34
+ cc.host = nil
35
+ end
36
+ assert !c.valid?
37
+ end
38
+
39
+ # test
40
+
41
+ def test_test_should_return_false_if_code_is_is_set_to_200_but_response_is_500
42
+ c = valid_condition
43
+ Net::HTTP.expects(:start).yields(stub(:read_timeout= => nil, :get => stub(:code => 500)))
44
+ assert_equal false, c.test
45
+ end
46
+
47
+ def test_test_should_return_false_if_code_is_not_is_set_to_200_and_response_is_200
48
+ c = valid_condition do |cc|
49
+ cc.code_is = nil
50
+ cc.code_is_not = [200]
51
+ end
52
+ Net::HTTP.expects(:start).yields(stub(:read_timeout= => nil, :get => stub(:code => 200)))
53
+ assert_equal false, c.test
54
+ end
55
+
56
+ def test_test_should_return_true_if_code_is_is_set_to_200_and_response_is_200
57
+ c = valid_condition
58
+ Net::HTTP.expects(:start).yields(stub(:read_timeout= => nil, :get => stub(:code => 200)))
59
+ assert_equal true, c.test
60
+ end
61
+
62
+ def test_test_should_return_false_if_code_is_not_is_set_to_200_but_response_is_500
63
+ c = valid_condition do |cc|
64
+ cc.code_is = nil
65
+ cc.code_is_not = [200]
66
+ end
67
+ Net::HTTP.expects(:start).yields(stub(:read_timeout= => nil, :get => stub(:code => 500)))
68
+ assert_equal true, c.test
69
+ end
70
+
71
+ def test_test_should_return_false_if_code_is_is_set_to_200_but_response_times_out
72
+ c = valid_condition
73
+ Net::HTTP.expects(:start).raises(Timeout::Error, '')
74
+ assert_equal false, c.test
75
+ end
76
+
77
+ def test_test_should_return_true_if_code_is_not_is_set_to_200_and_response_times_out
78
+ c = valid_condition do |cc|
79
+ cc.code_is = nil
80
+ cc.code_is_not = [200]
81
+ end
82
+ Net::HTTP.expects(:start).raises(Timeout::Error, '')
83
+ assert_equal true, c.test
84
+ end
85
+
86
+ def test_test_should_return_false_if_code_is_is_set_to_200_but_cant_connect
87
+ c = valid_condition
88
+ Net::HTTP.expects(:start).raises(Errno::ECONNREFUSED, '')
89
+ assert_equal false, c.test
90
+ end
91
+
92
+ def test_test_should_return_true_if_code_is_not_is_set_to_200_and_cant_connect
93
+ c = valid_condition do |cc|
94
+ cc.code_is = nil
95
+ cc.code_is_not = [200]
96
+ end
97
+ Net::HTTP.expects(:start).raises(Errno::ECONNREFUSED, '')
98
+ assert_equal true, c.test
99
+ end
100
+
101
+ def test_test_should_return_true_if_code_is_is_set_to_200_and_response_is_200_twice_for_times_two_of_two
102
+ c = valid_condition do |cc|
103
+ cc.times = [2, 2]
104
+ end
105
+ Net::HTTP.expects(:start).yields(stub(:read_timeout= => nil, :get => stub(:code => 200))).times(2)
106
+ assert_equal false, c.test
107
+ assert_equal true, c.test
108
+ end
109
+ end
@@ -0,0 +1,40 @@
1
+ require File.dirname(__FILE__) + '/helper'
2
+
3
+ class TestConditionsProcessRunning < Test::Unit::TestCase
4
+ def test_missing_pid_file_returns_opposite
5
+ [true, false].each do |r|
6
+ c = Conditions::ProcessRunning.new
7
+ c.running = r
8
+ c.stubs(:watch).returns(stub(:pid => 99999999, :name => 'foo'))
9
+ assert_equal !r, c.test
10
+ end
11
+ end
12
+
13
+ def test_not_running_returns_opposite
14
+ [true, false].each do |r|
15
+ c = Conditions::ProcessRunning.new
16
+ c.running = r
17
+
18
+ File.stubs(:exist?).returns(true)
19
+ c.stubs(:watch).returns(stub(:pid => 123))
20
+ File.stubs(:read).returns('5')
21
+ System::Process.any_instance.stubs(:exists?).returns(false)
22
+
23
+ assert_equal !r, c.test
24
+ end
25
+ end
26
+
27
+ def test_running_returns_same
28
+ [true, false].each do |r|
29
+ c = Conditions::ProcessRunning.new
30
+ c.running = r
31
+
32
+ File.stubs(:exist?).returns(true)
33
+ c.stubs(:watch).returns(stub(:pid => 123))
34
+ File.stubs(:read).returns('5')
35
+ System::Process.any_instance.stubs(:exists?).returns(true)
36
+
37
+ assert_equal r, c.test
38
+ end
39
+ end
40
+ end