xbar 0.4.1 → 0.4.2

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 (86) hide show
  1. data/.gitignore +1 -0
  2. data/.rspec +2 -0
  3. data/README.mkdn +1 -1
  4. data/Rakefile +5 -2
  5. data/dummy/.gitignore +15 -0
  6. data/dummy/Gemfile +38 -0
  7. data/dummy/README.rdoc +261 -0
  8. data/dummy/Rakefile +7 -0
  9. data/dummy/app/assets/images/rails.png +0 -0
  10. data/dummy/app/assets/javascripts/application.js +15 -0
  11. data/dummy/app/assets/stylesheets/application.css +13 -0
  12. data/dummy/app/controllers/application_controller.rb +3 -0
  13. data/dummy/app/helpers/application_helper.rb +2 -0
  14. data/dummy/app/mailers/.gitkeep +0 -0
  15. data/dummy/app/models/.gitkeep +0 -0
  16. data/dummy/app/views/layouts/application.html.erb +14 -0
  17. data/dummy/config.ru +4 -0
  18. data/dummy/config/application.rb +59 -0
  19. data/dummy/config/boot.rb +6 -0
  20. data/dummy/config/database.yml +25 -0
  21. data/dummy/config/environment.rb +5 -0
  22. data/dummy/config/environments/development.rb +37 -0
  23. data/dummy/config/environments/production.rb +67 -0
  24. data/dummy/config/environments/test.rb +37 -0
  25. data/dummy/config/initializers/backtrace_silencers.rb +7 -0
  26. data/dummy/config/initializers/inflections.rb +15 -0
  27. data/dummy/config/initializers/mime_types.rb +5 -0
  28. data/dummy/config/initializers/secret_token.rb +7 -0
  29. data/dummy/config/initializers/session_store.rb +8 -0
  30. data/dummy/config/initializers/wrap_parameters.rb +14 -0
  31. data/dummy/config/locales/en.yml +5 -0
  32. data/dummy/config/routes.rb +58 -0
  33. data/dummy/db/schema.rb +16 -0
  34. data/dummy/db/seeds.rb +7 -0
  35. data/dummy/doc/README_FOR_APP +2 -0
  36. data/dummy/lib/assets/.gitkeep +0 -0
  37. data/dummy/lib/generators/initializer/USAGE +8 -0
  38. data/dummy/lib/generators/initializer/initializer_generator.rb +8 -0
  39. data/dummy/lib/generators/initializer/templates/initializer.rb +1 -0
  40. data/dummy/lib/tasks/.gitkeep +0 -0
  41. data/dummy/log/.gitkeep +0 -0
  42. data/dummy/public/404.html +26 -0
  43. data/dummy/public/422.html +26 -0
  44. data/dummy/public/500.html +25 -0
  45. data/dummy/public/favicon.ico +0 -0
  46. data/dummy/public/index.html +241 -0
  47. data/dummy/public/robots.txt +5 -0
  48. data/dummy/script/rails +6 -0
  49. data/dummy/test/fixtures/.gitkeep +0 -0
  50. data/dummy/test/functional/.gitkeep +0 -0
  51. data/dummy/test/integration/.gitkeep +0 -0
  52. data/dummy/test/performance/browsing_test.rb +12 -0
  53. data/dummy/test/test_helper.rb +13 -0
  54. data/dummy/test/unit/.gitkeep +0 -0
  55. data/dummy/vendor/assets/javascripts/.gitkeep +0 -0
  56. data/dummy/vendor/assets/stylesheets/.gitkeep +0 -0
  57. data/dummy/vendor/plugins/.gitkeep +0 -0
  58. data/examples/README +3 -1
  59. data/examples/basic.rb +34 -0
  60. data/examples/config/canada.json +17 -0
  61. data/examples/config/canada2.json +17 -0
  62. data/examples/config/connection.rb +2 -0
  63. data/examples/config/simple.json +10 -10
  64. data/examples/lib/helpers.rb +90 -0
  65. data/examples/{setup.rb → lib/setup.rb} +14 -6
  66. data/examples/migrations/1_create_users.rb +1 -1
  67. data/examples/multithread.rb +47 -0
  68. data/examples/pause.rb +67 -0
  69. data/examples/pause_switch.rb +61 -0
  70. data/examples/switch.rb +42 -0
  71. data/gemfiles/rails32.gemfile.lock +3 -3
  72. data/lib/xbar.rb +1 -0
  73. data/lib/xbar/mapper.rb +61 -14
  74. data/lib/xbar/model.rb +1 -10
  75. data/lib/xbar/proxy.rb +104 -33
  76. data/lib/xbar/shard.rb +17 -1
  77. data/lib/xbar/version.rb +1 -1
  78. data/run +15 -7
  79. data/spec/console.rb +0 -1
  80. data/spec/spec_helper.rb +5 -2
  81. data/spec/xbar/association_spec.rb +1 -0
  82. data/spec/xbar/mapper_spec.rb +2 -4
  83. data/spec/xbar/model_spec.rb +11 -8
  84. data/spec/xbar/proxy_spec.rb +1 -1
  85. metadata +90 -28
  86. data/examples/example1.rb +0 -34
File without changes
@@ -0,0 +1,241 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Ruby on Rails: Welcome aboard</title>
5
+ <style type="text/css" media="screen">
6
+ body {
7
+ margin: 0;
8
+ margin-bottom: 25px;
9
+ padding: 0;
10
+ background-color: #f0f0f0;
11
+ font-family: "Lucida Grande", "Bitstream Vera Sans", "Verdana";
12
+ font-size: 13px;
13
+ color: #333;
14
+ }
15
+
16
+ h1 {
17
+ font-size: 28px;
18
+ color: #000;
19
+ }
20
+
21
+ a {color: #03c}
22
+ a:hover {
23
+ background-color: #03c;
24
+ color: white;
25
+ text-decoration: none;
26
+ }
27
+
28
+
29
+ #page {
30
+ background-color: #f0f0f0;
31
+ width: 750px;
32
+ margin: 0;
33
+ margin-left: auto;
34
+ margin-right: auto;
35
+ }
36
+
37
+ #content {
38
+ float: left;
39
+ background-color: white;
40
+ border: 3px solid #aaa;
41
+ border-top: none;
42
+ padding: 25px;
43
+ width: 500px;
44
+ }
45
+
46
+ #sidebar {
47
+ float: right;
48
+ width: 175px;
49
+ }
50
+
51
+ #footer {
52
+ clear: both;
53
+ }
54
+
55
+ #header, #about, #getting-started {
56
+ padding-left: 75px;
57
+ padding-right: 30px;
58
+ }
59
+
60
+
61
+ #header {
62
+ background-image: url("assets/rails.png");
63
+ background-repeat: no-repeat;
64
+ background-position: top left;
65
+ height: 64px;
66
+ }
67
+ #header h1, #header h2 {margin: 0}
68
+ #header h2 {
69
+ color: #888;
70
+ font-weight: normal;
71
+ font-size: 16px;
72
+ }
73
+
74
+
75
+ #about h3 {
76
+ margin: 0;
77
+ margin-bottom: 10px;
78
+ font-size: 14px;
79
+ }
80
+
81
+ #about-content {
82
+ background-color: #ffd;
83
+ border: 1px solid #fc0;
84
+ margin-left: -55px;
85
+ margin-right: -10px;
86
+ }
87
+ #about-content table {
88
+ margin-top: 10px;
89
+ margin-bottom: 10px;
90
+ font-size: 11px;
91
+ border-collapse: collapse;
92
+ }
93
+ #about-content td {
94
+ padding: 10px;
95
+ padding-top: 3px;
96
+ padding-bottom: 3px;
97
+ }
98
+ #about-content td.name {color: #555}
99
+ #about-content td.value {color: #000}
100
+
101
+ #about-content ul {
102
+ padding: 0;
103
+ list-style-type: none;
104
+ }
105
+
106
+ #about-content.failure {
107
+ background-color: #fcc;
108
+ border: 1px solid #f00;
109
+ }
110
+ #about-content.failure p {
111
+ margin: 0;
112
+ padding: 10px;
113
+ }
114
+
115
+
116
+ #getting-started {
117
+ border-top: 1px solid #ccc;
118
+ margin-top: 25px;
119
+ padding-top: 15px;
120
+ }
121
+ #getting-started h1 {
122
+ margin: 0;
123
+ font-size: 20px;
124
+ }
125
+ #getting-started h2 {
126
+ margin: 0;
127
+ font-size: 14px;
128
+ font-weight: normal;
129
+ color: #333;
130
+ margin-bottom: 25px;
131
+ }
132
+ #getting-started ol {
133
+ margin-left: 0;
134
+ padding-left: 0;
135
+ }
136
+ #getting-started li {
137
+ font-size: 18px;
138
+ color: #888;
139
+ margin-bottom: 25px;
140
+ }
141
+ #getting-started li h2 {
142
+ margin: 0;
143
+ font-weight: normal;
144
+ font-size: 18px;
145
+ color: #333;
146
+ }
147
+ #getting-started li p {
148
+ color: #555;
149
+ font-size: 13px;
150
+ }
151
+
152
+
153
+ #sidebar ul {
154
+ margin-left: 0;
155
+ padding-left: 0;
156
+ }
157
+ #sidebar ul h3 {
158
+ margin-top: 25px;
159
+ font-size: 16px;
160
+ padding-bottom: 10px;
161
+ border-bottom: 1px solid #ccc;
162
+ }
163
+ #sidebar li {
164
+ list-style-type: none;
165
+ }
166
+ #sidebar ul.links li {
167
+ margin-bottom: 5px;
168
+ }
169
+
170
+ .filename {
171
+ font-style: italic;
172
+ }
173
+ </style>
174
+ <script type="text/javascript">
175
+ function about() {
176
+ info = document.getElementById('about-content');
177
+ if (window.XMLHttpRequest)
178
+ { xhr = new XMLHttpRequest(); }
179
+ else
180
+ { xhr = new ActiveXObject("Microsoft.XMLHTTP"); }
181
+ xhr.open("GET","rails/info/properties",false);
182
+ xhr.send("");
183
+ info.innerHTML = xhr.responseText;
184
+ info.style.display = 'block'
185
+ }
186
+ </script>
187
+ </head>
188
+ <body>
189
+ <div id="page">
190
+ <div id="sidebar">
191
+ <ul id="sidebar-items">
192
+ <li>
193
+ <h3>Browse the documentation</h3>
194
+ <ul class="links">
195
+ <li><a href="http://guides.rubyonrails.org/">Rails Guides</a></li>
196
+ <li><a href="http://api.rubyonrails.org/">Rails API</a></li>
197
+ <li><a href="http://www.ruby-doc.org/core/">Ruby core</a></li>
198
+ <li><a href="http://www.ruby-doc.org/stdlib/">Ruby standard library</a></li>
199
+ </ul>
200
+ </li>
201
+ </ul>
202
+ </div>
203
+
204
+ <div id="content">
205
+ <div id="header">
206
+ <h1>Welcome aboard</h1>
207
+ <h2>You&rsquo;re riding Ruby on Rails!</h2>
208
+ </div>
209
+
210
+ <div id="about">
211
+ <h3><a href="rails/info/properties" onclick="about(); return false">About your application&rsquo;s environment</a></h3>
212
+ <div id="about-content" style="display: none"></div>
213
+ </div>
214
+
215
+ <div id="getting-started">
216
+ <h1>Getting started</h1>
217
+ <h2>Here&rsquo;s how to get rolling:</h2>
218
+
219
+ <ol>
220
+ <li>
221
+ <h2>Use <code>rails generate</code> to create your models and controllers</h2>
222
+ <p>To see all available options, run it without parameters.</p>
223
+ </li>
224
+
225
+ <li>
226
+ <h2>Set up a default route and remove <span class="filename">public/index.html</span></h2>
227
+ <p>Routes are set up in <span class="filename">config/routes.rb</span>.</p>
228
+ </li>
229
+
230
+ <li>
231
+ <h2>Create your database</h2>
232
+ <p>Run <code>rake db:create</code> to create your database. If you're not using SQLite (the default), edit <span class="filename">config/database.yml</span> with your username and password.</p>
233
+ </li>
234
+ </ol>
235
+ </div>
236
+ </div>
237
+
238
+ <div id="footer">&nbsp;</div>
239
+ </div>
240
+ </body>
241
+ </html>
@@ -0,0 +1,5 @@
1
+ # See http://www.robotstxt.org/wc/norobots.html for documentation on how to use the robots.txt file
2
+ #
3
+ # To ban all spiders from the entire site uncomment the next two lines:
4
+ # User-Agent: *
5
+ # Disallow: /
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env ruby
2
+ # This command will automatically be run when you run "rails" with Rails 3 gems installed from the root of your application.
3
+
4
+ APP_PATH = File.expand_path('../../config/application', __FILE__)
5
+ require File.expand_path('../../config/boot', __FILE__)
6
+ require 'rails/commands'
File without changes
File without changes
File without changes
@@ -0,0 +1,12 @@
1
+ require 'test_helper'
2
+ require 'rails/performance_test_help'
3
+
4
+ class BrowsingTest < ActionDispatch::PerformanceTest
5
+ # Refer to the documentation for all available options
6
+ # self.profile_options = { :runs => 5, :metrics => [:wall_time, :memory]
7
+ # :output => 'tmp/performance', :formats => [:flat] }
8
+
9
+ def test_homepage
10
+ get '/'
11
+ end
12
+ end
@@ -0,0 +1,13 @@
1
+ ENV["RAILS_ENV"] = "test"
2
+ require File.expand_path('../../config/environment', __FILE__)
3
+ require 'rails/test_help'
4
+
5
+ class ActiveSupport::TestCase
6
+ # Setup all fixtures in test/fixtures/*.(yml|csv) for all tests in alphabetical order.
7
+ #
8
+ # Note: You'll currently still have to declare fixtures explicitly in integration tests
9
+ # -- they do not yet inherit this setting
10
+ fixtures :all
11
+
12
+ # Add more helper methods to be used by all tests here...
13
+ end
File without changes
File without changes
File without changes
File without changes
data/examples/README CHANGED
@@ -1,5 +1,7 @@
1
1
  Set your environment variable BUNDLE_GEMFILE to one of the
2
2
  gemfiles in ../gemfiles, then do
3
3
 
4
- bundle exec ruby example1.rb
4
+ 'bundle exec ruby <file>'
5
+
6
+ where file is 'basic.rb' or 'multithread.rb', etc.
5
7
 
data/examples/basic.rb ADDED
@@ -0,0 +1,34 @@
1
+ require File.expand_path(File.join(File.dirname(__FILE__), 'lib/setup'))
2
+
3
+ module Examples
4
+ Setup.start('simple', 'test', 1)
5
+
6
+ # Define the model to let us access the 'users' table through ActiveRecord.
7
+ class User < ActiveRecord::Base; end
8
+
9
+ # Everything is now set up. Remember that the three store shards don't
10
+ # really replicate. XBar is set up to *think* that they do. We know
11
+ # better, and can check that reads and writes go to the shards that
12
+ # we think that they should. The curious results below wouldn't happen
13
+ # if replication is really taking place (modulo the eventual-consistency
14
+ # problem).
15
+
16
+ User.using(:bakery).create!([{name: "mudpie"}, {name: "hohos"}])
17
+ XBar.using(:deli) do
18
+ User.create!([{name: "pastrami"}, {name: "potato salad"}, {name: "pizza"}])
19
+ end
20
+ User.using(:store).create!(name: "safeway")
21
+
22
+ s1 = User.using_any(:store).all.size # reads 'bakery', gets 2
23
+ s2 = User.using_any(:store).all.size # reads 'deli', gets 3
24
+ s3 = User.using_any(:store).all.size # reads 'bakery', gets 2
25
+ s4 = User.using_any(:store).all.size # reads 'deli', gets 2
26
+ p1 = User.using(:produce).all.size # 1
27
+ b1 = User.using(:bakery).all.size # 2
28
+ d1 = User.using(:deli).all.size # 3
29
+
30
+ puts [s1, s2, s3, s4, p1, b1, d1].to_s # [2, 3, 2, 3, 1, 2, 3]
31
+
32
+ Setup.stop(1)
33
+ end
34
+
@@ -0,0 +1,17 @@
1
+ {
2
+ "connections": {
3
+ "canada_1": "mysql2://root:graceling@deimos.thirdmode.com:3307/canada",
4
+ "canada_2": "mysql2://root:graceling@deimos.thirdmode.com:3308/canada",
5
+ "canada_3": "mysql2://root:graceling@deimos.thirdmode.com:3309/canada"
6
+ },
7
+ "environments": {
8
+ "test": {
9
+ "shards": {
10
+ "canada": ["canada_1", "canada_2", "canada_3"],
11
+ "canada_east": "canada_1",
12
+ "canada_central": "canada_2",
13
+ "canada_west": "canada_3"
14
+ }
15
+ }
16
+ }
17
+ }
@@ -0,0 +1,17 @@
1
+ {
2
+ "connections": {
3
+ "canada_1": "mysql2://root:graceling@deimos.thirdmode.com:3307/canada",
4
+ "canada_2": "mysql2://root:graceling@deimos.thirdmode.com:3308/canada",
5
+ "canada_3": "mysql2://root:graceling@deimos.thirdmode.com:3309/canada"
6
+ },
7
+ "environments": {
8
+ "test": {
9
+ "shards": {
10
+ "canada": ["canada_2", "canada_1", "canada_3"],
11
+ "canada_east": "canada_1",
12
+ "canada_central": "canada_2",
13
+ "canada_west": "canada_3"
14
+ }
15
+ }
16
+ }
17
+ }
@@ -0,0 +1,2 @@
1
+ ActiveRecord::Base.establish_connection(:adapter => "mysql2", :database => "master",
2
+ :host => "localhost", :port => 3306, :username => "root", :password => "")
@@ -1,21 +1,21 @@
1
1
  {
2
- "__COMMENT": "Simple SQLite French Environment",
2
+ "__COMMENT": "Simple SQLite Grocery Environment",
3
3
 
4
4
  "connections": {
5
- "paris_m": { "adapter": "sqlite3", "database": "/tmp/paris.sqlite3"},
6
- "france_1": { "adapter": "sqlite3", "database": "/tmp/france_1.sqlite3"},
7
- "france_2": { "adapter": "sqlite3", "database": "/tmp/france_2.sqlite3"},
8
- "france_3": { "adapter": "sqlite3", "database": "/tmp/france_3.sqlite3"}
5
+ "warehouse": { "adapter": "sqlite3", "database": "/tmp/warehouse.sqlite3"},
6
+ "produce": { "adapter": "sqlite3", "database": "/tmp/produce.sqlite3"},
7
+ "bakery": { "adapter": "sqlite3", "database": "/tmp/bakery.sqlite3"},
8
+ "deli": { "adapter": "sqlite3", "database": "/tmp/deli.sqlite3"}
9
9
  },
10
10
 
11
11
  "environments": {
12
12
  "test": {
13
13
  "shards": {
14
- "master": "paris_m",
15
- "france": ["france_1", "france_2", "france_3"],
16
- "france_nord": "france_1",
17
- "france_central": "france_2",
18
- "france_sud": "france_3"
14
+ "master": "warehouse",
15
+ "store": ["produce", "bakery", "deli"],
16
+ "produce": "produce",
17
+ "bakery": "bakery",
18
+ "deli": "deli"
19
19
  }
20
20
  }
21
21
  }
@@ -0,0 +1,90 @@
1
+ require 'active_support'
2
+ require 'active_record'
3
+ require 'xbar'
4
+
5
+ module XBar
6
+ module Example
7
+ module Helpers
8
+
9
+ def do_work(num_threads, iterations, shard)
10
+ @threads = []
11
+ num_threads.times do |i|
12
+ @threads << Thread.new(i) do
13
+ XBar.using(shard) do
14
+ iterations.times do |j|
15
+ name = "Thread_#{i}_#{j}"
16
+ User.create!(:name => name)
17
+ User.using_any.all # allow read from slave
18
+ end
19
+ end
20
+ end
21
+ end
22
+ end
23
+
24
+ def join_workers
25
+ @threads.each(&:join)
26
+ end
27
+
28
+ def shard_master_config(shard)
29
+ XBar::Mapper.shards[shard][0].spec.config
30
+ end
31
+
32
+ def model_config(klass)
33
+ #klass.connnection.shard_list.size
34
+ ActiveRecord::Base.connection_handler.retrieve_connection_pool(klass)
35
+ nil
36
+ end
37
+
38
+ def empty_users_table(shard)
39
+ config = shard_master_config(shard)
40
+ if config[:adapter] == "mysql2"
41
+ client = Mysql2::Client.new(config)
42
+ client.query("DELETE FROM users")
43
+ end
44
+ end
45
+
46
+ def query_users_table(shard)
47
+ config = shard_master_config(shard)
48
+ if config[:adapter] == "mysql2"
49
+ client = Mysql2::Client.new(config)
50
+ results = client.query("SELECT COUNT(*) AS count FROM users")
51
+ results.first["count"]
52
+ end
53
+ end
54
+
55
+ # Request all proxies pause themselves.
56
+ def request_pause
57
+ XBar::Mapper.proxies.values.each do |proxy|
58
+ proxy.request_pause
59
+ end
60
+ end
61
+
62
+ # Wait until all proxies are paused.
63
+ def wait_for_pause
64
+ loop do
65
+ count = 0
66
+ XBar::Mapper.proxies.values.each do |proxy|
67
+ count += 1 if proxy.paused?
68
+ end
69
+ break if count == XBar::Mapper.proxies.size
70
+ end
71
+ end
72
+
73
+ # Unpause all proxies.
74
+ def unpause
75
+ XBar::Mapper.proxies.values.each do |proxy|
76
+ proxy.unpause
77
+ end
78
+ end
79
+
80
+ def cleanup_exited_threads
81
+ @threads.each do |t|
82
+ XBar::Mapper.unregister(t)
83
+ end
84
+ @threads = []
85
+ XBar::Mapper.disconnect_all!
86
+ end
87
+
88
+ end
89
+ end
90
+ end