stillwater 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
data/.gitignore ADDED
@@ -0,0 +1,17 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
data/.rvmrc ADDED
@@ -0,0 +1 @@
1
+ rvm 1.8.7@balancer --create
data/.travis.yml ADDED
@@ -0,0 +1,9 @@
1
+ language: ruby
2
+ rvm:
3
+ - 1.8.7
4
+ - 1.9.2
5
+ - 1.9.3
6
+ - jruby-18mode # JRuby in 1.8 mode
7
+ - jruby-19mode # JRuby in 1.9 mode
8
+ - rbx-18mode
9
+ - rbx-19mode
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in balancer.gemspec
4
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2012 Trae Robrock
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,50 @@
1
+ # Stillwater
2
+
3
+ A simple connection pool, that allows connections to different servers (or anything else)
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'stillwater'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install stillwater
18
+
19
+ ## Usage
20
+
21
+ pool = Stillwater::ConnectionPool.new
22
+ %q{ host1.com host2.com }.each do |host|
23
+ pool.add { MyConnectionClass.new(host) }
24
+ end
25
+
26
+ # Basic connection handling
27
+ pool.with_connection do |connection|
28
+ # Do some stuff with your connection
29
+ end
30
+
31
+ # Retry connections
32
+ # This will retry your code with a new connection and mark the tried
33
+ # connection as bad. The bad connection will be put back in the pool
34
+ # at the default period of 5 minutes.
35
+ pool.retry_connection_from(ServerConnectionFailed) do |connection|
36
+ # Do some stuff with your connection
37
+ end
38
+
39
+ ## Contributing
40
+
41
+ 1. Fork it
42
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
43
+ 3. Commit your changes (`git commit -am 'Added some feature'`)
44
+ 4. Push to the branch (`git push origin my-new-feature`)
45
+ 5. Create new Pull Request
46
+
47
+ ## Contributors
48
+
49
+ * Trae Robrock ( https://github.com/trobrock )
50
+ * Julio Santos ( https://github.com/julio )
data/Rakefile ADDED
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env rake
2
+ require "bundler/gem_tasks"
3
+
4
+ require 'rspec/core/rake_task'
5
+ RSpec::Core::RakeTask.new(:spec)
6
+ task :default => :spec
data/lib/stillwater.rb ADDED
@@ -0,0 +1,5 @@
1
+ require "stillwater/version"
2
+ require "stillwater/connection_pool"
3
+
4
+ module Stillwater
5
+ end
@@ -0,0 +1,122 @@
1
+ require 'active_support/core_ext/array'
2
+
3
+ module Stillwater
4
+ class ConnectionNotAvailable < StandardError ; end
5
+
6
+ class ConnectionPool
7
+ attr_accessor :reactivate_timeout, :retry_count
8
+
9
+ def initialize
10
+ @pool = []
11
+ @reactivate_timeout = 5 * 60 # 5 minutes
12
+ @retry_count = 3
13
+ end
14
+
15
+ def add(&builder)
16
+ @pool << connection_info_from(builder)
17
+ end
18
+
19
+ def reactivate_timeout=(seconds)
20
+ running = !!@thread
21
+ @thread.kill if running
22
+ @thread = nil
23
+ @reactivate_timeout = seconds
24
+ start_polling if running
25
+ end
26
+
27
+ def with_connection(&block)
28
+ conn = checkout
29
+ result = yield conn
30
+ checkin conn
31
+
32
+ result
33
+ end
34
+
35
+ def retry_connection_from(exception_class, &block)
36
+ count = 0
37
+ conn = checkout
38
+ yield conn
39
+ rescue exception_class
40
+ deactivate conn
41
+ count += 1
42
+ raise if count >= @retry_count
43
+ retry
44
+ ensure
45
+ checkin conn
46
+ end
47
+
48
+ def checkout
49
+ connection_info = available.sample
50
+ raise ConnectionNotAvailable if connection_info.nil?
51
+ connection_info[:state] = :in_use
52
+
53
+ connection_info[:connection]
54
+ end
55
+
56
+ def checkin(conn)
57
+ connection_info = in_use.detect { |info| info[:connection] == conn }
58
+ connection_info[:state] = :available unless connection_info.nil?
59
+
60
+ true
61
+ end
62
+
63
+ def deactivate(conn)
64
+ connection_info = @pool.detect { |info| info[:connection] == conn }
65
+ connection_info[:connection] = nil
66
+ connection_info[:state] = :inactive
67
+
68
+ start_polling
69
+ true
70
+ end
71
+
72
+ def reactivate_all
73
+ inactive.each do |info|
74
+ info[:connection] = info[:builder].call
75
+ info[:state] = :available
76
+ end
77
+
78
+ true
79
+ end
80
+
81
+ def available_count
82
+ available.size
83
+ end
84
+
85
+ def inactive_count
86
+ inactive.size
87
+ end
88
+
89
+ def in_use_count
90
+ in_use.size
91
+ end
92
+
93
+ private
94
+
95
+ def start_polling
96
+ @thread ||= Thread.new do
97
+ sleep @reactivate_timeout
98
+ reactivate_all
99
+ end
100
+ end
101
+
102
+ def available
103
+ find_by_state :available
104
+ end
105
+
106
+ def in_use
107
+ find_by_state :in_use
108
+ end
109
+
110
+ def inactive
111
+ find_by_state :inactive
112
+ end
113
+
114
+ def find_by_state(state)
115
+ @pool.select { |info| info[:state] == state }
116
+ end
117
+
118
+ def connection_info_from(builder)
119
+ { :builder => builder, :state => :available, :connection => builder.call }
120
+ end
121
+ end
122
+ end
@@ -0,0 +1,3 @@
1
+ module Stillwater
2
+ VERSION = "0.0.1"
3
+ end
@@ -0,0 +1,173 @@
1
+ require 'spec_helper'
2
+
3
+ class TestConnection
4
+ attr_accessor :name
5
+ def initialize(name)
6
+ @name = name
7
+ end
8
+ end
9
+
10
+ class TestException < StandardError ; end
11
+
12
+ module Stillwater
13
+ describe ConnectionPool do
14
+ subject { ConnectionPool.new }
15
+
16
+ describe "#add" do
17
+ it "should add an available connection" do
18
+ subject.add { TestConnection.new("one") }
19
+
20
+ subject.available_count.should == 1
21
+ end
22
+ end
23
+
24
+ context "with a connection" do
25
+ before(:each) do
26
+ subject.add { TestConnection.new("one") }
27
+ end
28
+
29
+ describe "#checkout" do
30
+ it "should return a connection" do
31
+ conn = subject.checkout
32
+
33
+ conn.should be_instance_of(TestConnection)
34
+ subject.available_count.should == 0
35
+ subject.in_use_count.should == 1
36
+ end
37
+
38
+ it "should not be able to check out inactive connections" do
39
+ conn = subject.checkout
40
+ subject.deactivate conn
41
+
42
+ lambda { subject.checkout }.should raise_error(ConnectionNotAvailable)
43
+ end
44
+ end
45
+
46
+ describe "#checkin" do
47
+ it "should return the connection to the pool" do
48
+ conn = subject.checkout
49
+ subject.available_count.should == 0
50
+ subject.in_use_count.should == 1
51
+
52
+ subject.checkin(conn)
53
+ subject.available_count.should == 1
54
+ subject.in_use_count.should == 0
55
+ end
56
+ end
57
+
58
+ describe "#with_connection" do
59
+ it "should run some code with a connection" do
60
+ result = subject.with_connection do |connection|
61
+ subject.available_count.should == 0
62
+ connection.name.should == "one"
63
+
64
+ "bob"
65
+ end
66
+
67
+ result.should == "bob"
68
+ subject.available_count.should == 1
69
+ end
70
+ end
71
+
72
+ describe "#retry_connection_from" do
73
+ before(:each) do
74
+ subject.add { TestConnection.new("two") }
75
+ end
76
+
77
+ it "should retry a connection when receiving specific exception type" do
78
+ client_obj = mock("ClientObject")
79
+ client_obj.stubs(:do_something).raises(TestException).then.returns("client result")
80
+
81
+ result = subject.retry_connection_from(TestException) do |conn|
82
+ client_obj.do_something
83
+ end
84
+
85
+ subject.available_count.should == 1
86
+ subject.inactive_count.should == 1
87
+ result.should == "client result"
88
+ end
89
+
90
+ it "should respect the retry limit" do
91
+ subject.retry_count = 1
92
+
93
+ lambda {
94
+ subject.retry_connection_from(TestException) do |conn|
95
+ raise TestException
96
+ end
97
+ }.should raise_error(TestException)
98
+
99
+ subject.available_count.should == 1
100
+ subject.inactive_count.should == 1
101
+ end
102
+
103
+ it "should fail if all connections get deactivated" do
104
+ lambda {
105
+ subject.retry_connection_from(TestException) do |conn|
106
+ raise TestException
107
+ end
108
+ }.should raise_error(ConnectionNotAvailable)
109
+ end
110
+ end
111
+
112
+ describe "#deactivate" do
113
+ before(:each) do
114
+ subject.add { TestConnection.new("two") }
115
+ subject.add { TestConnection.new("three") }
116
+ end
117
+
118
+ it "should remove a deactivated connection from the pool" do
119
+ subject.available_count.should == 3
120
+ subject.inactive_count.should == 0
121
+
122
+ subject.with_connection do |connection|
123
+ subject.deactivate connection
124
+ end
125
+
126
+ subject.available_count.should == 2
127
+ subject.inactive_count.should == 1
128
+ end
129
+ end
130
+
131
+ describe "#reactivate" do
132
+ before(:each) do
133
+ subject.add { TestConnection.new("two") }
134
+ subject.add { TestConnection.new("three") }
135
+ end
136
+
137
+ it "should be able to reactivate inactive connections" do
138
+ subject.available_count.should == 3
139
+ subject.inactive_count.should == 0
140
+
141
+ conn1 = subject.checkout
142
+ conn2 = subject.checkout
143
+
144
+ subject.deactivate conn1
145
+ subject.deactivate conn2
146
+
147
+ subject.available_count.should == 1
148
+ subject.inactive_count.should == 2
149
+
150
+ subject.reactivate_all
151
+
152
+ subject.available_count.should == 3
153
+ subject.inactive_count.should == 0
154
+ end
155
+
156
+ it "should auto reactivate" do
157
+ subject.reactivate_timeout = 1
158
+
159
+ conn1 = subject.checkout
160
+ conn2 = subject.checkout
161
+
162
+ subject.deactivate conn1
163
+ subject.deactivate conn2
164
+
165
+ sleep 2
166
+
167
+ subject.available_count.should == 3
168
+ subject.inactive_count.should == 0
169
+ end
170
+ end
171
+ end
172
+ end
173
+ end
@@ -0,0 +1,14 @@
1
+ require 'stillwater'
2
+
3
+ RSpec.configure do |config|
4
+ # Use color in STDOUT
5
+ config.color_enabled = true
6
+
7
+ # Use color not only in STDOUT but also in pagers and files
8
+ config.tty = true
9
+
10
+ # Use the specified formatter
11
+ config.formatter = :documentation # :progress, :html, :textmate
12
+
13
+ config.mock_framework = :mocha
14
+ end
@@ -0,0 +1,23 @@
1
+ # -*- encoding: utf-8 -*-
2
+ require File.expand_path('../lib/stillwater/version', __FILE__)
3
+
4
+ Gem::Specification.new do |gem|
5
+ gem.authors = ["Trae Robrock", "Julio Santos"]
6
+ gem.email = ["trobrock@gmail.com"]
7
+ gem.description = %q{A simple connection pool, that allows connections to different servers (or anything else)}
8
+ gem.summary = %q{A simple connection pool, that allows connections to different servers (or anything else)}
9
+ gem.homepage = "https://github.com/trobrock/stillwater"
10
+
11
+ gem.files = `git ls-files`.split($\)
12
+ gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
13
+ gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
14
+ gem.name = "stillwater"
15
+ gem.require_paths = ["lib"]
16
+ gem.version = Stillwater::VERSION
17
+
18
+ gem.add_dependency %q<activesupport>, '~> 2.3.0'
19
+
20
+ gem.add_development_dependency %q<rake>
21
+ gem.add_development_dependency %q<rspec>
22
+ gem.add_development_dependency %q<mocha>
23
+ end
metadata ADDED
@@ -0,0 +1,137 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: stillwater
3
+ version: !ruby/object:Gem::Version
4
+ hash: 29
5
+ prerelease:
6
+ segments:
7
+ - 0
8
+ - 0
9
+ - 1
10
+ version: 0.0.1
11
+ platform: ruby
12
+ authors:
13
+ - Trae Robrock
14
+ - Julio Santos
15
+ autorequire:
16
+ bindir: bin
17
+ cert_chain: []
18
+
19
+ date: 2012-07-05 00:00:00 Z
20
+ dependencies:
21
+ - !ruby/object:Gem::Dependency
22
+ version_requirements: &id001 !ruby/object:Gem::Requirement
23
+ none: false
24
+ requirements:
25
+ - - ~>
26
+ - !ruby/object:Gem::Version
27
+ hash: 3
28
+ segments:
29
+ - 2
30
+ - 3
31
+ - 0
32
+ version: 2.3.0
33
+ name: activesupport
34
+ type: :runtime
35
+ prerelease: false
36
+ requirement: *id001
37
+ - !ruby/object:Gem::Dependency
38
+ version_requirements: &id002 !ruby/object:Gem::Requirement
39
+ none: false
40
+ requirements:
41
+ - - ">="
42
+ - !ruby/object:Gem::Version
43
+ hash: 3
44
+ segments:
45
+ - 0
46
+ version: "0"
47
+ name: rake
48
+ type: :development
49
+ prerelease: false
50
+ requirement: *id002
51
+ - !ruby/object:Gem::Dependency
52
+ version_requirements: &id003 !ruby/object:Gem::Requirement
53
+ none: false
54
+ requirements:
55
+ - - ">="
56
+ - !ruby/object:Gem::Version
57
+ hash: 3
58
+ segments:
59
+ - 0
60
+ version: "0"
61
+ name: rspec
62
+ type: :development
63
+ prerelease: false
64
+ requirement: *id003
65
+ - !ruby/object:Gem::Dependency
66
+ version_requirements: &id004 !ruby/object:Gem::Requirement
67
+ none: false
68
+ requirements:
69
+ - - ">="
70
+ - !ruby/object:Gem::Version
71
+ hash: 3
72
+ segments:
73
+ - 0
74
+ version: "0"
75
+ name: mocha
76
+ type: :development
77
+ prerelease: false
78
+ requirement: *id004
79
+ description: A simple connection pool, that allows connections to different servers (or anything else)
80
+ email:
81
+ - trobrock@gmail.com
82
+ executables: []
83
+
84
+ extensions: []
85
+
86
+ extra_rdoc_files: []
87
+
88
+ files:
89
+ - .gitignore
90
+ - .rvmrc
91
+ - .travis.yml
92
+ - Gemfile
93
+ - LICENSE
94
+ - README.md
95
+ - Rakefile
96
+ - lib/stillwater.rb
97
+ - lib/stillwater/connection_pool.rb
98
+ - lib/stillwater/version.rb
99
+ - spec/connection_pool_spec.rb
100
+ - spec/spec_helper.rb
101
+ - stillwater.gemspec
102
+ homepage: https://github.com/trobrock/stillwater
103
+ licenses: []
104
+
105
+ post_install_message:
106
+ rdoc_options: []
107
+
108
+ require_paths:
109
+ - lib
110
+ required_ruby_version: !ruby/object:Gem::Requirement
111
+ none: false
112
+ requirements:
113
+ - - ">="
114
+ - !ruby/object:Gem::Version
115
+ hash: 3
116
+ segments:
117
+ - 0
118
+ version: "0"
119
+ required_rubygems_version: !ruby/object:Gem::Requirement
120
+ none: false
121
+ requirements:
122
+ - - ">="
123
+ - !ruby/object:Gem::Version
124
+ hash: 3
125
+ segments:
126
+ - 0
127
+ version: "0"
128
+ requirements: []
129
+
130
+ rubyforge_project:
131
+ rubygems_version: 1.8.21
132
+ signing_key:
133
+ specification_version: 3
134
+ summary: A simple connection pool, that allows connections to different servers (or anything else)
135
+ test_files:
136
+ - spec/connection_pool_spec.rb
137
+ - spec/spec_helper.rb