timecube 0.0.0
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.
- data/.document +5 -0
- data/.gitignore +21 -0
- data/LICENSE +20 -0
- data/README.rdoc +17 -0
- data/Rakefile +53 -0
- data/VERSION +1 -0
- data/lib/timecube.rb +186 -0
- data/test/helper.rb +10 -0
- data/test/test_timecube.rb +7 -0
- data/timecube.gemspec +54 -0
- metadata +84 -0
data/.document
ADDED
data/.gitignore
ADDED
data/LICENSE
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2009 Jason Morrison
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.rdoc
ADDED
@@ -0,0 +1,17 @@
|
|
1
|
+
= timecube
|
2
|
+
|
3
|
+
Description goes here.
|
4
|
+
|
5
|
+
== Note on Patches/Pull Requests
|
6
|
+
|
7
|
+
* Fork the project.
|
8
|
+
* Make your feature addition or bug fix.
|
9
|
+
* Add tests for it. This is important so I don't break it in a
|
10
|
+
future version unintentionally.
|
11
|
+
* Commit, do not mess with rakefile, version, or history.
|
12
|
+
(if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull)
|
13
|
+
* Send me a pull request. Bonus points for topic branches.
|
14
|
+
|
15
|
+
== Copyright
|
16
|
+
|
17
|
+
Copyright (c) 2010 Jason Morrison. See LICENSE for details.
|
data/Rakefile
ADDED
@@ -0,0 +1,53 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'rake'
|
3
|
+
|
4
|
+
begin
|
5
|
+
require 'jeweler'
|
6
|
+
Jeweler::Tasks.new do |gem|
|
7
|
+
gem.name = "timecube"
|
8
|
+
gem.summary = %Q{EventMachine proxy for MySQL date/time functions}
|
9
|
+
gem.description = %Q{Nature's harmonic simultaneous 4-day time cube}
|
10
|
+
gem.email = "jmorrison@thoughtbot.com"
|
11
|
+
gem.homepage = "http://github.com/jasonm/timecube"
|
12
|
+
gem.authors = ["Jason Morrison"]
|
13
|
+
gem.add_development_dependency "thoughtbot-shoulda", ">= 0"
|
14
|
+
# gem is a Gem::Specification... see http://www.rubygems.org/read/chapter/20 for additional settings
|
15
|
+
end
|
16
|
+
Jeweler::GemcutterTasks.new
|
17
|
+
rescue LoadError
|
18
|
+
puts "Jeweler (or a dependency) not available. Install it with: gem install jeweler"
|
19
|
+
end
|
20
|
+
|
21
|
+
require 'rake/testtask'
|
22
|
+
Rake::TestTask.new(:test) do |test|
|
23
|
+
test.libs << 'lib' << 'test'
|
24
|
+
test.pattern = 'test/**/test_*.rb'
|
25
|
+
test.verbose = true
|
26
|
+
end
|
27
|
+
|
28
|
+
begin
|
29
|
+
require 'rcov/rcovtask'
|
30
|
+
Rcov::RcovTask.new do |test|
|
31
|
+
test.libs << 'test'
|
32
|
+
test.pattern = 'test/**/test_*.rb'
|
33
|
+
test.verbose = true
|
34
|
+
end
|
35
|
+
rescue LoadError
|
36
|
+
task :rcov do
|
37
|
+
abort "RCov is not available. In order to run rcov, you must: sudo gem install spicycode-rcov"
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
task :test => :check_dependencies
|
42
|
+
|
43
|
+
task :default => :test
|
44
|
+
|
45
|
+
require 'rake/rdoctask'
|
46
|
+
Rake::RDocTask.new do |rdoc|
|
47
|
+
version = File.exist?('VERSION') ? File.read('VERSION') : ""
|
48
|
+
|
49
|
+
rdoc.rdoc_dir = 'rdoc'
|
50
|
+
rdoc.title = "timecube #{version}"
|
51
|
+
rdoc.rdoc_files.include('README*')
|
52
|
+
rdoc.rdoc_files.include('lib/**/*.rb')
|
53
|
+
end
|
data/VERSION
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
0.0.0
|
data/lib/timecube.rb
ADDED
@@ -0,0 +1,186 @@
|
|
1
|
+
require "rubygems"
|
2
|
+
gem "em-proxy"
|
3
|
+
gem "em-mysql"
|
4
|
+
|
5
|
+
require "em-proxy"
|
6
|
+
require "em-mysql"
|
7
|
+
require "stringio"
|
8
|
+
require "fiber"
|
9
|
+
|
10
|
+
Proxy.start(:host => "0.0.0.0", :port => 3307) do |conn|
|
11
|
+
conn.server :mysql, :host => "127.0.0.1", :port => 3306, :relay_server => true
|
12
|
+
|
13
|
+
QUERY_CMD = 3
|
14
|
+
MAX_PACKET_LENGTH = 2**24-1
|
15
|
+
|
16
|
+
# open a direct connection to MySQL for the schema-free coordination logic
|
17
|
+
@mysql = EventMachine::MySQL.new(:host => 'localhost', :database => 'noschema')
|
18
|
+
|
19
|
+
conn.on_data do |data|
|
20
|
+
fiber = Fiber.new {
|
21
|
+
p [:original_request, data]
|
22
|
+
|
23
|
+
overhead, chunks, seq = data[0,4].unpack("CvC")
|
24
|
+
type, sql = data[4, data.size].unpack("Ca*")
|
25
|
+
|
26
|
+
p [:request, [overhead, chunks, seq], [type, sql]]
|
27
|
+
|
28
|
+
if type == QUERY_CMD
|
29
|
+
query = sql.downcase.split
|
30
|
+
p [:query, query]
|
31
|
+
|
32
|
+
# TODO: can probably switch to http://github.com/omghax/sql
|
33
|
+
# for AST query parsing & mods.
|
34
|
+
|
35
|
+
case query.first
|
36
|
+
when "create" then
|
37
|
+
# Allow schemaless table creation, ex: 'create table posts'
|
38
|
+
# By creating a table with a single id for key storage, aka
|
39
|
+
# rewrite to: 'create table posts (id varchar(255))'. All
|
40
|
+
# future attribute tables will be created on demand at
|
41
|
+
# insert time of a new record
|
42
|
+
overload = "(id varchar(255), UNIQUE(id));"
|
43
|
+
query += [overload]
|
44
|
+
overhead += overload.size + 1
|
45
|
+
|
46
|
+
p [:create_new_schema_free_table, query, data]
|
47
|
+
|
48
|
+
when "insert" then
|
49
|
+
# Overload the INSERT syntax to allow for nested parameters
|
50
|
+
# inside the statement. ex:
|
51
|
+
# INSERT INTO posts (id, author, nickname, ...) VALUES (
|
52
|
+
# 'ilya', 'Ilya Grigorik', 'igrigorik'
|
53
|
+
# )
|
54
|
+
#
|
55
|
+
# The following query will be mapped into 3 distinct tables:
|
56
|
+
# => 'posts' table will store the key
|
57
|
+
# => 'posts_author' will store key, value
|
58
|
+
# => 'posts_nickname' will store key, value
|
59
|
+
#
|
60
|
+
# or, in SQL..
|
61
|
+
#
|
62
|
+
# => insert into posts values("ilya");
|
63
|
+
# => create table posts_author (id varchar(40), value varchar(255), UNIQUE(id));
|
64
|
+
# => insert into posts_author values("ilya", "Ilya Grigorik");
|
65
|
+
# => ... repeat for every attribute
|
66
|
+
#
|
67
|
+
# If the table post_value has not been seen before, it will
|
68
|
+
# be created on the fly. Hence allowing us to add and remove
|
69
|
+
# keys and values at will. :-)
|
70
|
+
#
|
71
|
+
# P.S. There is probably cleaner syntax for this, but hey...
|
72
|
+
|
73
|
+
|
74
|
+
if insert = sql.match(/\((.*?)\).*?\((.*?)\)/)
|
75
|
+
data = {}
|
76
|
+
table = query[2]
|
77
|
+
keys = insert[1].split(',').map!{|s| s.strip}
|
78
|
+
values = insert[2].scan(/([^\'|\"]+)/).flatten.reject {|s| s.strip == ','}
|
79
|
+
keys.each_with_index {|k,i| data[k] = values[i]}
|
80
|
+
|
81
|
+
data.each do |key, value|
|
82
|
+
next if key == 'id'
|
83
|
+
attr_sql = "insert into #{table}_#{key} values('#{data['id']}', '#{value}')"
|
84
|
+
|
85
|
+
q = @mysql.query(attr_sql)
|
86
|
+
q.errback { |res|
|
87
|
+
# if the attribute table for this model does not yet exist then create it!
|
88
|
+
# - yes, there is a race condition here, add fiber logic later
|
89
|
+
if res.is_a?(Mysql::Error) and res.message =~ /Table.*doesn\'t exist/
|
90
|
+
|
91
|
+
table_sql = "create table #{table}_#{key} (id varchar(255), value varchar(255), UNIQUE(id))"
|
92
|
+
tc = @mysql.query(table_sql)
|
93
|
+
tc.callback { @mysql.query(attr_sql) }
|
94
|
+
end
|
95
|
+
}
|
96
|
+
|
97
|
+
p [:inserted_attr, table, key, value]
|
98
|
+
end
|
99
|
+
|
100
|
+
# override the query to insert the key into posts table
|
101
|
+
query = query[0,3] + ["VALUES('#{data['id']}')"]
|
102
|
+
overhead = query.join(" ").size + 1
|
103
|
+
|
104
|
+
p [:insert, query]
|
105
|
+
end
|
106
|
+
|
107
|
+
when "select" then
|
108
|
+
# Overload the select call to perform a multi-join in the background
|
109
|
+
# and rewrite the attribute names to fool the client into thinking it
|
110
|
+
# all came from the same table.
|
111
|
+
#
|
112
|
+
# To figure out which tables we need to join on, do the simple / dumb
|
113
|
+
# approach and issue a 'show tables like key_%' to do 'runtime
|
114
|
+
# introspection'. Could easily cache this, but that's for later.
|
115
|
+
#
|
116
|
+
# Ex, a 'select * from posts' query with one value (author) would be
|
117
|
+
# rewritten into the following query:
|
118
|
+
#
|
119
|
+
# SELECT posts.id as id, posts_author.value as author FROM posts
|
120
|
+
# LEFT OUTER JOIN posts_author ON posts_author.id = posts.id
|
121
|
+
# WHERE posts.id = "ilya";
|
122
|
+
|
123
|
+
select = sql.match(/select(.*?)from\s([^\s]+)/)
|
124
|
+
where = sql.match(/where\s([^=]+)\s?=\s?'?"?([^\s'"]+)'?"?/)
|
125
|
+
attrs, table = select[1].strip.split(','), select[2] if select
|
126
|
+
key = where[2] if where
|
127
|
+
|
128
|
+
if select
|
129
|
+
p [:select, select, attrs, where]
|
130
|
+
|
131
|
+
tables = @mysql.query("show tables like '#{table}_%'")
|
132
|
+
tables.callback { |res|
|
133
|
+
fiber.resume(res.all_hashes.collect(&:values).flatten.collect{ |c|
|
134
|
+
c.split('_').last
|
135
|
+
})
|
136
|
+
}
|
137
|
+
tables = Fiber.yield
|
138
|
+
|
139
|
+
p [:select_tables, tables]
|
140
|
+
|
141
|
+
# build the select statements, hide the tables behind each attribute
|
142
|
+
join = "select #{table}.id as id "
|
143
|
+
tables.each do |column|
|
144
|
+
join += " , #{table}_#{column}.value as #{column} "
|
145
|
+
end
|
146
|
+
|
147
|
+
# add the joins to stich it all together
|
148
|
+
join += " FROM #{table} "
|
149
|
+
tables.each do |column|
|
150
|
+
join += " LEFT OUTER JOIN #{table}_#{column} ON #{table}_#{column}.id = #{table}.id "
|
151
|
+
end
|
152
|
+
|
153
|
+
join += " WHERE #{table}.id = '#{key}' " if key
|
154
|
+
|
155
|
+
query = [join]
|
156
|
+
overhead = join.size + 1
|
157
|
+
|
158
|
+
p [:join_query, join]
|
159
|
+
end
|
160
|
+
end
|
161
|
+
|
162
|
+
# repack the query data and forward to server
|
163
|
+
# - have to split message on packet boundaries
|
164
|
+
|
165
|
+
seq, data = 0, []
|
166
|
+
query = StringIO.new([type, query.join(" ")].pack("Ca*"))
|
167
|
+
while q = query.read(MAX_PACKET_LENGTH)
|
168
|
+
data.push [q.length % 256, q.length / 256, seq].pack("CvC") + q
|
169
|
+
seq = (seq + 1) % 256
|
170
|
+
end
|
171
|
+
|
172
|
+
p [:final_query, data, chunks, overhead]
|
173
|
+
puts "-" * 100
|
174
|
+
end
|
175
|
+
|
176
|
+
[data].flatten.each do |chunk|
|
177
|
+
conn.relay_to_servers(chunk)
|
178
|
+
end
|
179
|
+
|
180
|
+
:async # we will render results later
|
181
|
+
}
|
182
|
+
|
183
|
+
fiber.resume
|
184
|
+
end
|
185
|
+
end
|
186
|
+
|
data/test/helper.rb
ADDED
data/timecube.gemspec
ADDED
@@ -0,0 +1,54 @@
|
|
1
|
+
# Generated by jeweler
|
2
|
+
# DO NOT EDIT THIS FILE DIRECTLY
|
3
|
+
# Instead, edit Jeweler::Tasks in Rakefile, and run the gemspec command
|
4
|
+
# -*- encoding: utf-8 -*-
|
5
|
+
|
6
|
+
Gem::Specification.new do |s|
|
7
|
+
s.name = %q{timecube}
|
8
|
+
s.version = "0.0.0"
|
9
|
+
|
10
|
+
s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
|
11
|
+
s.authors = ["Jason Morrison"]
|
12
|
+
s.date = %q{2010-07-02}
|
13
|
+
s.description = %q{Nature's harmonic simultaneous 4-day time cube}
|
14
|
+
s.email = %q{jmorrison@thoughtbot.com}
|
15
|
+
s.extra_rdoc_files = [
|
16
|
+
"LICENSE",
|
17
|
+
"README.rdoc"
|
18
|
+
]
|
19
|
+
s.files = [
|
20
|
+
".document",
|
21
|
+
".gitignore",
|
22
|
+
"LICENSE",
|
23
|
+
"README.rdoc",
|
24
|
+
"Rakefile",
|
25
|
+
"VERSION",
|
26
|
+
"lib/timecube.rb",
|
27
|
+
"test/helper.rb",
|
28
|
+
"test/test_timecube.rb",
|
29
|
+
"timecube.gemspec"
|
30
|
+
]
|
31
|
+
s.homepage = %q{http://github.com/jasonm/timecube}
|
32
|
+
s.rdoc_options = ["--charset=UTF-8"]
|
33
|
+
s.require_paths = ["lib"]
|
34
|
+
s.rubygems_version = %q{1.3.6}
|
35
|
+
s.summary = %q{EventMachine proxy for MySQL date/time functions}
|
36
|
+
s.test_files = [
|
37
|
+
"test/helper.rb",
|
38
|
+
"test/test_timecube.rb"
|
39
|
+
]
|
40
|
+
|
41
|
+
if s.respond_to? :specification_version then
|
42
|
+
current_version = Gem::Specification::CURRENT_SPECIFICATION_VERSION
|
43
|
+
s.specification_version = 3
|
44
|
+
|
45
|
+
if Gem::Version.new(Gem::RubyGemsVersion) >= Gem::Version.new('1.2.0') then
|
46
|
+
s.add_development_dependency(%q<thoughtbot-shoulda>, [">= 0"])
|
47
|
+
else
|
48
|
+
s.add_dependency(%q<thoughtbot-shoulda>, [">= 0"])
|
49
|
+
end
|
50
|
+
else
|
51
|
+
s.add_dependency(%q<thoughtbot-shoulda>, [">= 0"])
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
metadata
ADDED
@@ -0,0 +1,84 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: timecube
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
prerelease: false
|
5
|
+
segments:
|
6
|
+
- 0
|
7
|
+
- 0
|
8
|
+
- 0
|
9
|
+
version: 0.0.0
|
10
|
+
platform: ruby
|
11
|
+
authors:
|
12
|
+
- Jason Morrison
|
13
|
+
autorequire:
|
14
|
+
bindir: bin
|
15
|
+
cert_chain: []
|
16
|
+
|
17
|
+
date: 2010-07-02 00:00:00 -04:00
|
18
|
+
default_executable:
|
19
|
+
dependencies:
|
20
|
+
- !ruby/object:Gem::Dependency
|
21
|
+
name: thoughtbot-shoulda
|
22
|
+
prerelease: false
|
23
|
+
requirement: &id001 !ruby/object:Gem::Requirement
|
24
|
+
requirements:
|
25
|
+
- - ">="
|
26
|
+
- !ruby/object:Gem::Version
|
27
|
+
segments:
|
28
|
+
- 0
|
29
|
+
version: "0"
|
30
|
+
type: :development
|
31
|
+
version_requirements: *id001
|
32
|
+
description: Nature's harmonic simultaneous 4-day time cube
|
33
|
+
email: jmorrison@thoughtbot.com
|
34
|
+
executables: []
|
35
|
+
|
36
|
+
extensions: []
|
37
|
+
|
38
|
+
extra_rdoc_files:
|
39
|
+
- LICENSE
|
40
|
+
- README.rdoc
|
41
|
+
files:
|
42
|
+
- .document
|
43
|
+
- .gitignore
|
44
|
+
- LICENSE
|
45
|
+
- README.rdoc
|
46
|
+
- Rakefile
|
47
|
+
- VERSION
|
48
|
+
- lib/timecube.rb
|
49
|
+
- test/helper.rb
|
50
|
+
- test/test_timecube.rb
|
51
|
+
- timecube.gemspec
|
52
|
+
has_rdoc: true
|
53
|
+
homepage: http://github.com/jasonm/timecube
|
54
|
+
licenses: []
|
55
|
+
|
56
|
+
post_install_message:
|
57
|
+
rdoc_options:
|
58
|
+
- --charset=UTF-8
|
59
|
+
require_paths:
|
60
|
+
- lib
|
61
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
62
|
+
requirements:
|
63
|
+
- - ">="
|
64
|
+
- !ruby/object:Gem::Version
|
65
|
+
segments:
|
66
|
+
- 0
|
67
|
+
version: "0"
|
68
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
69
|
+
requirements:
|
70
|
+
- - ">="
|
71
|
+
- !ruby/object:Gem::Version
|
72
|
+
segments:
|
73
|
+
- 0
|
74
|
+
version: "0"
|
75
|
+
requirements: []
|
76
|
+
|
77
|
+
rubyforge_project:
|
78
|
+
rubygems_version: 1.3.6
|
79
|
+
signing_key:
|
80
|
+
specification_version: 3
|
81
|
+
summary: EventMachine proxy for MySQL date/time functions
|
82
|
+
test_files:
|
83
|
+
- test/helper.rb
|
84
|
+
- test/test_timecube.rb
|