sticky_blox 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
data/LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright (c) 2009 Jason Rogers
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,19 @@
1
+ = sticky_blox
2
+
3
+ StickyBlox is a late binding traits-approach to Ruby meta-programming. It allows you to define
4
+ functionality within a class or object and apply that functionality to instances of other objects
5
+ without having to know anything about how those other objects are created.
6
+
7
+ == Note on Patches/Pull Requests
8
+
9
+ * Fork the project.
10
+ * Make your feature addition or bug fix.
11
+ * Add tests for it. This is important so I don't break it in a
12
+ future version unintentionally.
13
+ * Commit, do not mess with rakefile, version, or history.
14
+ (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)
15
+ * Send me a pull request. Bonus points for topic branches.
16
+
17
+ == Copyright
18
+
19
+ Copyright (c) 2010 Jason Rogers. See LICENSE for details.
data/README.textile ADDED
@@ -0,0 +1,91 @@
1
+ h2. What is StickyBlox?
2
+
3
+ StickyBlox is a late binding traits-approach to Ruby meta-programming. It allows you to define
4
+ functionality within a class or object and apply that functionality to instances of other objects
5
+ without having to know anything about how those other objects are created.
6
+
7
+ h2. Why?
8
+
9
+ If you're thinking to yourself, "I could just open a class and directly add the methods I want, and
10
+ get the same behavior, or I could create a module and extend or include it where ever I want..." you
11
+ would be right. So, why use StickyBlox? It's really a matter of taste. Sometimes, I don't like opening
12
+ other classes to add my application specific behavior, and I would rather have somewhere else in which
13
+ to encapsulate that behavior. I would also like to be able to reuse that behavior with any object
14
+ without having to add the methods to every class in the system. StickyBlox allows you have the behavior
15
+ shoe-horned into classes such that the functionalities (sticks) become instance methods or just have
16
+ the stick available to be bound to anything, at anytime.
17
+
18
+ h2. Example
19
+
20
+ With the later releases of Sinatra, your code is not reload-able in development. You either have to
21
+ restart your server (which can be very burndensome, especially if you have a long start up time) or
22
+ you have to run your application through some Shotgun or something similar.
23
+
24
+ Here we define a reloader for your route implementations. With this you can pull the logic or your
25
+ routes out to well-defined components and then reload those components at run-time to get new behavior.
26
+
27
+ h3. app.rb
28
+
29
+ <pre><code>
30
+ require 'reloading'
31
+
32
+ ReloadableRoutes.stick_to Sinatra::Application
33
+
34
+ get "/reload/:file/?" do
35
+ reload
36
+ end
37
+
38
+ get "/touch_current_user/?" do
39
+ touch_current_user
40
+ end
41
+
42
+ put "/run_as_someone_else/?" do
43
+ # is the current user allowed to masquerade?
44
+ current_user = session[:current_user]
45
+ raise "unauthorized" unless current_user.allowed_to_masquerade_as_someone_else?
46
+
47
+ # let's track the fact that our current user is masquerading
48
+ touch_current_user
49
+
50
+ # do some other marvelous stuff, but let's not argue about the use-cases of such a route
51
+
52
+ # finally let's reset the application state for the current user
53
+ session[:current_user] = current_user
54
+ end
55
+
56
+ ['table1', 'table2'].each do |table_name|
57
+ get table_name do
58
+ load_records_for(table_name)
59
+ end
60
+ end
61
+ </code></pre>
62
+
63
+ h3. reloading.rb
64
+
65
+ <pre><code>
66
+ require 'sticky_blox'
67
+
68
+ class ReloadableRoutes
69
+ include StickyBlox
70
+
71
+ stick :reload do
72
+ # params comes from the instance of the Sinatra application into
73
+ # which this stick is bound
74
+ raise "no file specified" unless params[:file]
75
+ load params[:file]
76
+ "Success"
77
+ end
78
+
79
+ stick :touch_current_user do
80
+ # use the instance variables typically available to an instance of a
81
+ # Sinatra::Application, or subclass, to update some aspect of the system
82
+ session[:current_user].touch
83
+ end
84
+
85
+ stick :load_records_for {|some_table_name|
86
+ # again, params comes from the instance of the Sinatra application
87
+ # into which this stick is bound
88
+ model_for(some_table_name).filter(params[some_table_name]).all
89
+ }
90
+ end
91
+ </code></pre>
data/TODO ADDED
@@ -0,0 +1 @@
1
+ Anything?
data/VERSION ADDED
@@ -0,0 +1 @@
1
+ 1.0.0
@@ -0,0 +1,34 @@
1
+ require File.join( File.dirname(__FILE__), 'proc_extension' )
2
+
3
+ module StickyBlox
4
+ def self.included(base)
5
+ base.extend ClassMethods
6
+ end
7
+
8
+ module ClassMethods
9
+ def stick_to binding
10
+ (class << self; self; end).instance_eval do
11
+ clz = self
12
+ @stuck_methods.each do |route_name|
13
+ binding.class_eval do
14
+ define_method route_name do | *args |
15
+ clz.send(route_name).bind(self).call(*args)
16
+ end
17
+ end
18
+ end
19
+ end
20
+ end
21
+
22
+ def stick name, &block
23
+ (class << self; self; end).instance_eval do
24
+ @stuck_methods ||= []
25
+ @stuck_methods << name unless @stuck_methods.include?(name)
26
+ end
27
+ self.class.class_eval do
28
+ define_method name do
29
+ block
30
+ end
31
+ end
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,19 @@
1
+ begin
2
+ Proc.instance_method(:bind)
3
+ rescue NameError
4
+ # Proc instance don't respond to :bind, so let's define it
5
+ class Proc
6
+ # this was lifted from the facets project (facets.rubyforge.org)
7
+ # which in turn was lifted from Rails
8
+ def bind(object)
9
+ block, time = self, Time.now
10
+ (class << object; self; end).class_eval do
11
+ method_name = "__bind_#{time.to_i}_#{time.usec}"
12
+ define_method(method_name, &block)
13
+ method = instance_method(method_name)
14
+ remove_method(method_name)
15
+ method
16
+ end.bind(object)
17
+ end
18
+ end
19
+ end
@@ -0,0 +1 @@
1
+ require File.join( File.dirname(__FILE__), 'sticky_blox', 'behavior' )
data/spec/helper.rb ADDED
@@ -0,0 +1,7 @@
1
+ require 'rubygems'
2
+ require 'spec'
3
+ require 'spearmint'
4
+
5
+ $LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
6
+ $LOAD_PATH.unshift(File.dirname(__FILE__))
7
+ require 'sticky_blox'
data/spec/spearmint.rb ADDED
@@ -0,0 +1,19 @@
1
+ require File.join(File.dirname(__FILE__), '..', 'lib', 'sticky_blox')
2
+
3
+ class Spearmint
4
+ include StickyBlox
5
+
6
+ stick :chew do
7
+ raise "ah, ah, ah ... I ain't no reverse-able object. I'm a #{self.class}" unless self.respond_to?(:reverse)
8
+ self.reverse
9
+ end
10
+
11
+ stick :chew_it_up_and_spit_it_out do
12
+ raise "ah, ah, ah ... I ain't no reverse-able object. I'm a #{self.class}" unless self.respond_to?(:reverse!)
13
+ self.reverse!
14
+ end
15
+
16
+ stick :open do |how_many_pieces|
17
+ (1..(how_many_pieces || self).to_i).collect
18
+ end
19
+ end
@@ -0,0 +1,48 @@
1
+ require 'helper'
2
+
3
+ describe "Instance-based approaches" do
4
+ class Doublemint < Spearmint
5
+ end
6
+
7
+ it "Doublemint shouldn't have sticks to use" do
8
+ proc{Doublemint.stick_to Object}.should raise_error(NameError)
9
+ end
10
+
11
+ it 'should be bound to any string' do
12
+ Spearmint.stick_to String
13
+ s = "abc"
14
+ s1 = s.chew
15
+ s1.should == s.reverse
16
+
17
+ s = "Spearmint"
18
+ s2 = "tnimraepS"
19
+ s1 = s.chew_it_up_and_spit_it_out
20
+ s1.should == s2
21
+ s1.should === s
22
+ end
23
+
24
+ it "should turn the String into an Integer and return an array of numbers" do
25
+ thing = "2"
26
+ thing.open.should == [1, 2]
27
+ thing.open(5).should == [1, 2, 3, 4, 5]
28
+ end
29
+ end
30
+
31
+ describe "Class-based approaches" do
32
+ it 'should be able to bind to the instance of any object' do
33
+ s = %w{a b c}
34
+ s1 = Spearmint.chew.bind(s).call
35
+ s1.should == %w{c b a}
36
+
37
+ s = %w{a b c}
38
+ s2 = %w{c b a}
39
+ s1 = Spearmint.chew_it_up_and_spit_it_out.bind(s).call
40
+ s1.should == s2
41
+ s1.should == s
42
+ end
43
+
44
+ it "should throw an error when binding to something other than a String" do
45
+ thing = Object.new
46
+ proc {Spearmint.chew.bind(thing).call}.should raise_error
47
+ end
48
+ end
@@ -0,0 +1,59 @@
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{sticky_blox}
8
+ s.version = "1.0.0"
9
+
10
+ s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
11
+ s.authors = ["Jason Rogers"]
12
+ s.date = %q{2010-04-12}
13
+ s.description = %q{see http://github.com/jacaetevha/sticky_blox for more description. I'm lazy}
14
+ s.email = %q{jacaetevha@gmail.com}
15
+ s.extra_rdoc_files = [
16
+ "LICENSE",
17
+ "README.rdoc",
18
+ "README.textile",
19
+ "TODO"
20
+ ]
21
+ s.files = [
22
+ "LICENSE",
23
+ "README.rdoc",
24
+ "README.textile",
25
+ "TODO",
26
+ "VERSION",
27
+ "lib/sticky_blox.rb",
28
+ "lib/sticky_blox/behavior.rb",
29
+ "lib/sticky_blox/proc_extension.rb",
30
+ "spec/helper.rb",
31
+ "spec/spearmint.rb",
32
+ "spec/sticky_blox_spec.rb",
33
+ "sticky_blox.gemspec"
34
+ ]
35
+ s.homepage = %q{http://github.com/jacaetevha/sticky_blox}
36
+ s.rdoc_options = ["--charset=UTF-8"]
37
+ s.require_paths = ["lib"]
38
+ s.rubygems_version = %q{1.3.6}
39
+ s.summary = %q{Late-binding traits for Ruby}
40
+ s.test_files = [
41
+ "spec/helper.rb",
42
+ "spec/spearmint.rb",
43
+ "spec/sticky_blox_spec.rb"
44
+ ]
45
+
46
+ if s.respond_to? :specification_version then
47
+ current_version = Gem::Specification::CURRENT_SPECIFICATION_VERSION
48
+ s.specification_version = 3
49
+
50
+ if Gem::Version.new(Gem::RubyGemsVersion) >= Gem::Version.new('1.2.0') then
51
+ s.add_development_dependency(%q<rspec>, [">= 1.3.0"])
52
+ else
53
+ s.add_dependency(%q<rspec>, [">= 1.3.0"])
54
+ end
55
+ else
56
+ s.add_dependency(%q<rspec>, [">= 1.3.0"])
57
+ end
58
+ end
59
+
metadata ADDED
@@ -0,0 +1,91 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: sticky_blox
3
+ version: !ruby/object:Gem::Version
4
+ prerelease: false
5
+ segments:
6
+ - 1
7
+ - 0
8
+ - 0
9
+ version: 1.0.0
10
+ platform: ruby
11
+ authors:
12
+ - Jason Rogers
13
+ autorequire:
14
+ bindir: bin
15
+ cert_chain: []
16
+
17
+ date: 2010-04-12 00:00:00 -04:00
18
+ default_executable:
19
+ dependencies:
20
+ - !ruby/object:Gem::Dependency
21
+ name: rspec
22
+ prerelease: false
23
+ requirement: &id001 !ruby/object:Gem::Requirement
24
+ requirements:
25
+ - - ">="
26
+ - !ruby/object:Gem::Version
27
+ segments:
28
+ - 1
29
+ - 3
30
+ - 0
31
+ version: 1.3.0
32
+ type: :development
33
+ version_requirements: *id001
34
+ description: see http://github.com/jacaetevha/sticky_blox for more description. I'm lazy
35
+ email: jacaetevha@gmail.com
36
+ executables: []
37
+
38
+ extensions: []
39
+
40
+ extra_rdoc_files:
41
+ - LICENSE
42
+ - README.rdoc
43
+ - README.textile
44
+ - TODO
45
+ files:
46
+ - LICENSE
47
+ - README.rdoc
48
+ - README.textile
49
+ - TODO
50
+ - VERSION
51
+ - lib/sticky_blox.rb
52
+ - lib/sticky_blox/behavior.rb
53
+ - lib/sticky_blox/proc_extension.rb
54
+ - spec/helper.rb
55
+ - spec/spearmint.rb
56
+ - spec/sticky_blox_spec.rb
57
+ - sticky_blox.gemspec
58
+ has_rdoc: true
59
+ homepage: http://github.com/jacaetevha/sticky_blox
60
+ licenses: []
61
+
62
+ post_install_message:
63
+ rdoc_options:
64
+ - --charset=UTF-8
65
+ require_paths:
66
+ - lib
67
+ required_ruby_version: !ruby/object:Gem::Requirement
68
+ requirements:
69
+ - - ">="
70
+ - !ruby/object:Gem::Version
71
+ segments:
72
+ - 0
73
+ version: "0"
74
+ required_rubygems_version: !ruby/object:Gem::Requirement
75
+ requirements:
76
+ - - ">="
77
+ - !ruby/object:Gem::Version
78
+ segments:
79
+ - 0
80
+ version: "0"
81
+ requirements: []
82
+
83
+ rubyforge_project:
84
+ rubygems_version: 1.3.6
85
+ signing_key:
86
+ specification_version: 3
87
+ summary: Late-binding traits for Ruby
88
+ test_files:
89
+ - spec/helper.rb
90
+ - spec/spearmint.rb
91
+ - spec/sticky_blox_spec.rb