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.
@@ -0,0 +1,5 @@
1
+ README.rdoc
2
+ lib/**/*.rb
3
+ bin/*
4
+ features/**/*.feature
5
+ LICENSE
@@ -0,0 +1,21 @@
1
+ ## MAC OS
2
+ .DS_Store
3
+
4
+ ## TEXTMATE
5
+ *.tmproj
6
+ tmtags
7
+
8
+ ## EMACS
9
+ *~
10
+ \#*
11
+ .\#*
12
+
13
+ ## VIM
14
+ *.swp
15
+
16
+ ## PROJECT::GENERAL
17
+ coverage
18
+ rdoc
19
+ pkg
20
+
21
+ ## PROJECT::SPECIFIC
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.
@@ -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.
@@ -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
@@ -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
+
@@ -0,0 +1,10 @@
1
+ require 'rubygems'
2
+ require 'test/unit'
3
+ require 'shoulda'
4
+
5
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
6
+ $LOAD_PATH.unshift(File.dirname(__FILE__))
7
+ require 'timecube'
8
+
9
+ class Test::Unit::TestCase
10
+ end
@@ -0,0 +1,7 @@
1
+ require 'helper'
2
+
3
+ class TestTimecube < Test::Unit::TestCase
4
+ should "probably rename this file and start testing for real" do
5
+ flunk "hey buddy, you should probably rename this file and start testing for real"
6
+ end
7
+ end
@@ -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