acts_as_messenger 0.0.1

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 ADDED
@@ -0,0 +1,5 @@
1
+ README.rdoc
2
+ lib/**/*.rb
3
+ bin/*
4
+ features/**/*.feature
5
+ LICENSE
data/.gitignore ADDED
@@ -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 Mike Nelson
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,65 @@
1
+ = acts_as_messenger
2
+
3
+ Standard facebook-style messaging system. Not bound to any friendship structure. Integrates directly into your User model. Allows for users to comment on a thread, leave a thread, and be updated when a thread is commented on.
4
+
5
+
6
+ == Installation
7
+
8
+ Install the gem
9
+
10
+ gem install acts_as_messenger
11
+
12
+ Generate the necessary classes and migrations
13
+
14
+ script/generate acts_as_messenger
15
+
16
+ Migrate the DB
17
+
18
+ rake db:migrate
19
+
20
+ Add acts_as_messenger to your user model
21
+
22
+ class User < ActiveRecord::Base
23
+ ...
24
+ acts_as_messenger
25
+ ...
26
+ end
27
+
28
+ == Usage
29
+
30
+ With acts_as_messenger now incorporated into your User model, you have access to helper methods for creating messages and recipients. It's as simple as calling the send_message(title, body, recipients) method:
31
+
32
+ mike = User.find 4
33
+ john = User.find 3
34
+ sky = User.find 2
35
+
36
+ thread = mike.send_message('Message Title', 'Message Body', [john, sky])
37
+ # OR
38
+ thread = mike.send_message('Message Title', 'Message Body', [3, 2])
39
+
40
+ The recipients of the message can be an array of AR objects or can be Fixnum's (or any combination). In the case of a Fixnum, the class type is assumed to be the same as the sending class.
41
+ ---
42
+ From the thread object you can access the participants of the message directly. Participants are considered as the author of the message and anyone who's commented on the thread.
43
+
44
+ thread.participants
45
+ => [mike]
46
+
47
+ john.comment_on(thread, 'Some comment')
48
+
49
+ thread.participants
50
+ => [mike, john]
51
+
52
+ === Privacy
53
+
54
+ All threads have a private_thread field. This allows the can_be_viewed_by? method to be utilized. The can_be_viewed_by? method checks to see if the object is either the author or one of the recipient objects:
55
+
56
+ random_user = User.find 490
57
+ thread.can_be_viewed_by?(random_user)
58
+ => false
59
+
60
+ thread.can_be_viewed_by?(john)
61
+ => true
62
+
63
+ == Copyright
64
+
65
+ Copyright (c) 2010 Mike Nelson. See LICENSE for details.
data/Rakefile ADDED
@@ -0,0 +1,46 @@
1
+ require 'rubygems'
2
+ require 'rake'
3
+
4
+ begin
5
+ require 'jeweler'
6
+ Jeweler::Tasks.new do |gem|
7
+ gem.name = "acts_as_messenger"
8
+ gem.summary = "Standard facebook-style messaging system. Not bound to any friendship structure. Integrates directly into your User model."
9
+ gem.description = "Standard facebook-style messaging system. Not bound to any friendship structure. Integrates directly into your User model. Allows for users to comment on a thread, leave a thread, and be updated when a thread is commented on."
10
+ gem.email = "mdnelson30@gmail.com"
11
+ gem.homepage = "http://github.com/mnelson/acts_as_messenger"
12
+ gem.authors = ["Mike Nelson"]
13
+ gem.add_development_dependency "rspec", ">= 1.2.9"
14
+ gem.add_development_dependency "active_record"
15
+ # gem is a Gem::Specification... see http://www.rubygems.org/read/chapter/20 for additional settings
16
+ end
17
+ Jeweler::GemcutterTasks.new
18
+ rescue LoadError
19
+ puts "Jeweler (or a dependency) not available. Install it with: gem install jeweler"
20
+ end
21
+
22
+ require 'spec/rake/spectask'
23
+ Spec::Rake::SpecTask.new(:spec) do |spec|
24
+ spec.libs << 'lib' << 'spec'
25
+ spec.spec_files = FileList['spec/**/*_spec.rb']
26
+ end
27
+
28
+ Spec::Rake::SpecTask.new(:rcov) do |spec|
29
+ spec.libs << 'lib' << 'spec'
30
+ spec.pattern = 'spec/**/*_spec.rb'
31
+ spec.rcov = true
32
+ end
33
+
34
+ task :spec => :check_dependencies
35
+
36
+ task :default => :spec
37
+
38
+ require 'rake/rdoctask'
39
+ Rake::RDocTask.new do |rdoc|
40
+ version = File.exist?('VERSION') ? File.read('VERSION') : ""
41
+
42
+ rdoc.rdoc_dir = 'rdoc'
43
+ rdoc.title = "acts_as_messenger #{version}"
44
+ rdoc.rdoc_files.include('README*')
45
+ rdoc.rdoc_files.include('lib/**/*.rb')
46
+ end
data/VERSION ADDED
@@ -0,0 +1 @@
1
+ 0.0.1
@@ -0,0 +1,72 @@
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{acts_as_messenger}
8
+ s.version = "0.0.1"
9
+
10
+ s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
11
+ s.authors = ["Mike Nelson"]
12
+ s.date = %q{2010-02-02}
13
+ s.description = %q{Standard facebook-style messaging system. Not bound to any friendship structure. Integrates directly into your User model. Allows for users to comment on a thread, leave a thread, and be updated when a thread is commented on.}
14
+ s.email = %q{mdnelson30@gmail.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
+ "acts_as_messenger.gemspec",
27
+ "generators/acts_as_messenger/acts_as_messenger_generator.rb",
28
+ "generators/acts_as_messenger/templates/migrations/create_comments.rb",
29
+ "generators/acts_as_messenger/templates/migrations/create_message_threads.rb",
30
+ "generators/acts_as_messenger/templates/migrations/create_recipients.rb",
31
+ "generators/acts_as_messenger/templates/models/comment.rb",
32
+ "generators/acts_as_messenger/templates/models/message_thread.rb",
33
+ "generators/acts_as_messenger/templates/models/recipient.rb",
34
+ "lib/acts_as_messenger.rb",
35
+ "lib/acts_as_messenger/thread.rb",
36
+ "spec/acts_as_messenger_spec.rb",
37
+ "spec/database.yml",
38
+ "spec/debug.log",
39
+ "spec/models/user.rb",
40
+ "spec/schema.rb",
41
+ "spec/spec.opts",
42
+ "spec/spec_helper.rb"
43
+ ]
44
+ s.homepage = %q{http://github.com/mnelson/acts_as_messenger}
45
+ s.rdoc_options = ["--charset=UTF-8"]
46
+ s.require_paths = ["lib"]
47
+ s.rubygems_version = %q{1.3.5}
48
+ s.summary = %q{Standard facebook-style messaging system. Not bound to any friendship structure. Integrates directly into your User model.}
49
+ s.test_files = [
50
+ "spec/acts_as_messenger_spec.rb",
51
+ "spec/models/user.rb",
52
+ "spec/schema.rb",
53
+ "spec/spec_helper.rb"
54
+ ]
55
+
56
+ if s.respond_to? :specification_version then
57
+ current_version = Gem::Specification::CURRENT_SPECIFICATION_VERSION
58
+ s.specification_version = 3
59
+
60
+ if Gem::Version.new(Gem::RubyGemsVersion) >= Gem::Version.new('1.2.0') then
61
+ s.add_development_dependency(%q<rspec>, [">= 1.2.9"])
62
+ s.add_development_dependency(%q<active_record>, [">= 0"])
63
+ else
64
+ s.add_dependency(%q<rspec>, [">= 1.2.9"])
65
+ s.add_dependency(%q<active_record>, [">= 0"])
66
+ end
67
+ else
68
+ s.add_dependency(%q<rspec>, [">= 1.2.9"])
69
+ s.add_dependency(%q<active_record>, [">= 0"])
70
+ end
71
+ end
72
+
@@ -0,0 +1,17 @@
1
+ class ActsAsMessengerGenerator < Rails::Generator::Base
2
+
3
+ def manifest
4
+ record do |m|
5
+
6
+ %w(comment message_thread recipient).each do |model|
7
+ m.file "models/#{model}.rb", "app/models/#{model}.rb"
8
+ m.migration_template "migrations/create_#{model.pluralize}.rb", "db/migrate"
9
+ end
10
+
11
+ end
12
+ end
13
+
14
+ def file_name
15
+ "acts_as_messenger"
16
+ end
17
+ end
@@ -0,0 +1,19 @@
1
+ class CreateComments < ActiveRecord::Migration
2
+
3
+ def self.up
4
+ create_table :comments do |t|
5
+ t.references :commentable, :polymorphic => true
6
+ t.integer :author_id
7
+ t.text :body
8
+ t.timestamps
9
+ end
10
+
11
+ add_index :comments, :author_id
12
+ add_index :comments, [:commentable_id, :commentable_type]
13
+ end
14
+
15
+ def self.down
16
+ drop_table :comments
17
+ end
18
+
19
+ end
@@ -0,0 +1,19 @@
1
+ class CreateMessageThreads < ActiveRecord::Migration
2
+
3
+ def self.up
4
+ create_table :message_threads do |t|
5
+ t.integer :author_id
6
+ t.string :title
7
+ t.text :body
8
+ t.boolean :private_thread, :default => true
9
+ t.timestamps
10
+ end
11
+
12
+ add_index :message_threads, :author_id
13
+ end
14
+
15
+ def self.down
16
+ drop_table :message_threads
17
+ end
18
+
19
+ end
@@ -0,0 +1,20 @@
1
+ class CreateRecipients < ActiveRecord::Migration
2
+
3
+ def self.up
4
+ create_table :recipients do |t|
5
+ t.references :receiver, :polymorphic => true
6
+ t.references :message_thread
7
+ t.boolean :has_read
8
+ t.timestamps
9
+ end
10
+
11
+ add_index :recipients, :message_thread_id
12
+ add_index :recipients, [:receiver_id, :receiver_type]
13
+
14
+ end
15
+
16
+ def self.down
17
+
18
+ end
19
+
20
+ end
@@ -0,0 +1,8 @@
1
+ class Comment < ActiveRecord::Base
2
+
3
+ default_scope :order => 'created_at asc'
4
+
5
+ belongs_to :author, :class_name => 'User'
6
+ belongs_to :commentable, :polymorphic => true
7
+
8
+ end
@@ -0,0 +1,21 @@
1
+ class MessageThread < ActiveRecord::Base
2
+ include SocialButterfly::Thread
3
+
4
+ acts_as_thread
5
+ has_many :recipients, :dependent => :destroy
6
+
7
+ def recipient_by_user(u)
8
+ self.recipients.by_user(u).first
9
+ end
10
+
11
+
12
+ def can_be_viewed_by?(user)
13
+ !!(user && (!self.private_thread || self.author == user || self.recipient_objects.include?(user)))
14
+ end
15
+
16
+ protected
17
+
18
+ def recipient_objects
19
+ self.recipients.find(:all, :include => :receiver).collect{|r| r.receiver}
20
+ end
21
+ end
@@ -0,0 +1,22 @@
1
+ class Recipient < ActiveRecord::Base
2
+
3
+
4
+ belongs_to :receiver, :polymorphic => true
5
+ belongs_to :message_thread
6
+
7
+ default_scope :order => 'has_read ASC, created_at DESC'
8
+
9
+ named_scope :by_user, lambda {|u| {:conditions => ['receiver_type = ? and receiver_id = ?', 'User', u.id]}}
10
+ named_scope :read, :conditions => ['has_read = ?', true]
11
+ named_scope :unread, :conditions => ['has_read = ?', false]
12
+
13
+
14
+ def read!
15
+ self.update_attribute(:has_read, true)
16
+ end
17
+
18
+ def unread!
19
+ self.update_attribute(:has_read, false)
20
+ end
21
+
22
+ end
@@ -0,0 +1,36 @@
1
+ module SocialButterfly
2
+ module Thread
3
+
4
+ def self.included(model)
5
+ model.extend(SocialButterfly::Thread::ClassMethods)
6
+
7
+ model.send(:include, SocialButterfly::Thread::InstanceMethods)
8
+ end
9
+
10
+ module ClassMethods
11
+ def acts_as_thread
12
+
13
+ validates_presence_of :author
14
+ validates_inclusion_of :private_thread, :in => [true, false]
15
+
16
+ belongs_to :author, :class_name => 'User'
17
+ has_many :comments, :as => :commentable, :dependent => :destroy
18
+ has_many :comment_participants, :through => :comments, :source => :author, :uniq => true
19
+
20
+ named_scope :by_user, lambda {|u| u && {:conditions => ['author_id = ?', u.id]} || {}}
21
+
22
+ end
23
+ end
24
+
25
+ module InstanceMethods
26
+ def last_post
27
+ self.comments.last || self
28
+ end
29
+
30
+ def participants
31
+ (self.comment_participants + [self.author].compact).uniq
32
+ end
33
+
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,50 @@
1
+ module SocialButterfly
2
+ module Message
3
+ def self.included(model)
4
+ model.extend(SocialButterfly::Message::ClassMethods)
5
+ model.send(:include, SocialButterfly::Message::InstanceMethods)
6
+ end
7
+
8
+ module ClassMethods
9
+ def acts_as_messenger
10
+ has_many :recipients, :as => :receiver
11
+ has_many :unread_recipients, :source => :recipients, :class_name => 'Recipient', :as => :receiver, :conditions => {:read => false}
12
+ has_many :read_recipients, :source => :recipients, :class_name => 'Recipient', :as => :receiver, :conditions => {:read => true}
13
+
14
+ has_many :sent_messages, :class_name => 'MessageThread', :foreign_key => :author_id
15
+ has_many :unread_messages, :through => :unread_recipients, :source => :message_thread
16
+ has_many :read_messages, :through => :read_recipients, :source => :message_thread
17
+ end
18
+ end
19
+
20
+ module InstanceMethods
21
+ def send_message(title, body, recips)
22
+ thread = ::MessageThread.create(:author => self, :title => title, :body => body, :private_thread => true)
23
+ thread.recipients = [recips].flatten.uniq.collect{|r| rdata = recipient_data(r); thread.recipients.create(:receiver_id => rdata[0], :receiver_type => rdata[1])}
24
+ thread
25
+ end
26
+ def recipient_for(message)
27
+ self.recipients.find(:first, :conditions => ['message_thread_id = ?', message.id])
28
+ end
29
+
30
+ def comment_on(message, body)
31
+ if message && body
32
+ message.comments.create(:author => self, :body => body)
33
+ recip = recipient_for(message)
34
+ (message.recipients - [recip].compact).each{|r| r.unread!}
35
+ true
36
+ else
37
+ false
38
+ end
39
+ end
40
+
41
+ private
42
+
43
+ def recipient_data(given)
44
+ given.is_a?(Fixnum) && [given, self.class.name] || [given.id, given.class.name]
45
+ end
46
+ end
47
+ end
48
+ end
49
+
50
+ ::ActiveRecord::Base.send :include, SocialButterfly::Message
@@ -0,0 +1,101 @@
1
+ require File.expand_path(File.dirname(__FILE__) + '/spec_helper')
2
+
3
+ describe "ActsAsMessenger" do
4
+
5
+ before(:each) do
6
+ @mike = User.create(:name => 'Mike')
7
+ @john = User.create(:name => 'John')
8
+ @sky = User.create(:name => 'Sky')
9
+ end
10
+
11
+ it "Should integrate flawlessly" do
12
+ @mike.methods.include?('send_message').should be_true
13
+ m = @mike.method('send_message')
14
+ m.arity.should eql(3)
15
+ end
16
+
17
+ it "Should generate a valid thread" do
18
+ thread = @mike.send_message('Title', 'Body', @john)
19
+ thread.valid?.should be_true
20
+ end
21
+
22
+ it "Should generate recipients correctly" do
23
+ thread = @mike.send_message('Title 1', 'Body 1', @john)
24
+
25
+ thread.recipients.size.should eql(1)
26
+ @john.recipients.size.should eql(1)
27
+ @mike.recipients.size.should eql(0)
28
+
29
+ thread = @john.send_message('Title 2', 'Body 2', [@mike.id, @sky.id])
30
+
31
+ thread.recipients.size.should eql(2)
32
+
33
+ @mike.reload
34
+ @john.reload
35
+ @sky.reload
36
+
37
+ @mike.recipients.size.should eql(1)
38
+ @john.recipients.size.should eql(1)
39
+ @sky.recipients.size.should eql(1)
40
+
41
+ end
42
+
43
+ it "Should allow for removal of recipient from thread" do
44
+ thread = @mike.send_message('Title 3', 'Body 3', [@john, @sky])
45
+
46
+ @john.recipients.first.read!
47
+ @john.recipients.read.size.should eql(1)
48
+ @john.recipients.unread.size.should eql(0)
49
+
50
+ @sky.comment_on(thread, 'This is totally awesome')
51
+
52
+ @john.recipients.unread.size.should eql(1)
53
+ @john.recipients.read.size.should eql(0)
54
+
55
+ @john.recipients.first.destroy
56
+
57
+ @sky.comment_on(thread, 'Seriously, i love this')
58
+
59
+ @john.recipients.size.should eql(0)
60
+
61
+ end
62
+
63
+ it "Should allow destroy all recipients on thread destruction" do
64
+ thread = @mike.send_message('Title 4', 'Body 4', [@john, @sky, @john, @sky])
65
+
66
+ @john.recipients.size.should eql(1)
67
+
68
+ thread.destroy
69
+
70
+ @john.recipients.size.should eql(0)
71
+ @sky.recipients.size.should eql(0)
72
+
73
+ end
74
+
75
+ it "Should access participants properly" do
76
+ thread = @mike.send_message('Title 5', 'Body 5', [@john, @sky])
77
+
78
+ thread.participants.sort{|a,b| a.id <=> b.id}.should eql([@mike])
79
+
80
+ @john.comment_on(thread, 'Test')
81
+
82
+ thread.reload
83
+ thread.participants.sort{|a,b| a.id <=> b.id}.should eql([@mike, @john])
84
+
85
+ @sky.comment_on(thread, 'Tester')
86
+
87
+ thread.reload
88
+ thread.participants.sort{|a,b| a.id <=> b.id}.should eql([@mike, @john, @sky])
89
+ end
90
+
91
+ it "Should restrict access to the right objects" do
92
+ thread = @mike.send_message('Title 6', 'Body 6', [@john, @sky])
93
+
94
+ frank = User.create
95
+ thread.can_be_viewed_by?(frank).should be_false
96
+
97
+ thread.can_be_viewed_by?(@sky).should be_true
98
+
99
+ end
100
+
101
+ end
data/spec/database.yml ADDED
@@ -0,0 +1,7 @@
1
+ mysql:
2
+ adapter: mysql
3
+ encoding: utf8
4
+ database: gem_test
5
+ username: root
6
+ password:
7
+ pool: 5