journalduhacker/app/models/comment.rb
2012-08-24 11:41:36 -05:00

207 lines
5.1 KiB
Ruby

class Comment < ActiveRecord::Base
belongs_to :user
belongs_to :story
has_many :votes,
:dependent => :delete_all
belongs_to :parent_comment,
:class_name => "Comment"
attr_accessible :comment
attr_accessor :parent_comment_short_id, :current_vote, :previewing,
:indent_level, :highlighted
before_create :assign_short_id_and_upvote
after_create :assign_votes, :mark_submitter, :deliver_reply_notifications
after_destroy :unassign_votes
MAX_EDIT_MINS = 45
define_index do
indexes comment
indexes user.username, :as => :author
has "(upvotes - downvotes)", :as => :score, :type => :integer,
:sortable => true
has created_at
end
validate do
self.comment.to_s.strip == "" &&
errors.add(:comment, "cannot be blank.")
self.user_id.blank? &&
errors.add(:user_id, "cannot be blank.")
self.story_id.blank? &&
errors.add(:story_id, "cannot be blank.")
(m = self.comment.to_s.strip.match(/\A(t)his([\.!])?$\z/i)) &&
errors.add(:base, (m[1] == "T" ? "N" : "n") + "ope" + m[2].to_s)
end
def self.regenerate_markdown
Comment.all.each do |c|
c.markeddown_comment = c.generated_markeddown_comment
Comment.record_timestamps = false
c.save(:validate => false)
end
nil
end
def assign_short_id_and_upvote
10.times do |try|
if try == 10
raise "too many hash collisions"
end
self.short_id = Utils.random_str(6)
if !Comment.find_by_short_id(self.short_id)
break
end
end
self.upvotes = 1
end
def assign_votes
Vote.vote_thusly_on_story_or_comment_for_user_because(1, self.story_id,
self.id, self.user.id, nil, false)
self.story.update_comment_count!
end
def mark_submitter
Keystore.increment_value_for("user:#{self.user_id}:comments_posted")
end
def deliver_reply_notifications
begin
if self.parent_comment_id && u = self.parent_comment.try(:user)
if u.email_replies?
EmailReply.reply(self, u).deliver
end
if u.pushover_replies? && u.pushover_user_key.present?
Pushover.push(u.pushover_user_key, u.pushover_device, {
:title => "Lobsters reply from #{self.user.username} on " <<
"#{self.story.title}",
:message => self.plaintext_comment,
:url => self.url,
:url_title => "Reply to #{self.user.username}",
})
end
end
rescue
end
end
def give_upvote_or_downvote_and_recalculate_confidence!(upvote, downvote)
self.upvotes += upvote.to_i
self.downvotes += downvote.to_i
Comment.connection.execute("UPDATE #{Comment.table_name} SET " <<
"upvotes = COALESCE(upvotes, 0) + #{upvote.to_i}, " <<
"downvotes = COALESCE(downvotes, 0) + #{downvote.to_i}, " <<
"confidence = '#{self.calculated_confidence}' WHERE id = " <<
"#{self.id.to_i}")
end
# http://evanmiller.org/how-not-to-sort-by-average-rating.html
# https://github.com/reddit/reddit/blob/master/r2/r2/lib/db/_sorts.pyx
def calculated_confidence
n = (upvotes + downvotes).to_f
if n == 0.0
return 0
end
z = 1.281551565545 # 80% confidence
p = upvotes.to_f / n
left = p + (1 / ((2.0 * n) * z * z))
right = z * Math.sqrt((p * ((1.0 - p) / n)) + (z * (z / (4.0 * n * n))))
under = 1.0 + ((1.0 / n) * z * z)
return (left - right) / under
end
def unassign_votes
self.story.update_comment_count!
end
def score
self.upvotes - self.downvotes
end
def generated_markeddown_comment
Markdowner.to_html(self.comment)
end
def comment=(com)
self[:comment] = com.to_s.rstrip
self.markeddown_comment = self.generated_markeddown_comment
end
def plaintext_comment
# TODO: linkify then strip tags and convert entities back
comment
end
def flag!
Story.update_counters self.id, :flaggings => 1
end
def has_been_edited?
self.updated_at && (self.updated_at - self.created_at > 1.minute)
end
def self.ordered_for_story_or_thread_for_user(story_id, thread_id, user_id)
parents = {}
if thread_id
cs = [ "thread_id = ?", thread_id ]
else
cs = [ "story_id = ?", story_id ]
end
Comment.find(:all, :conditions => cs, :order => "confidence DESC",
:include => :user).each do |c|
(parents[c.parent_comment_id.to_i] ||= []).push c
end
# top-down list of comments, regardless of indent level
ordered = []
recursor = lambda{|comment,level|
if comment
comment.indent_level = level
ordered.push comment
end
# for each comment that is a child of this one, recurse with it
(parents[comment ? comment.id : 0] || []).each do |child|
recursor.call(child, level + 1)
end
}
recursor.call(nil, 0)
# TODO: handle deleted comments, show for user_id
ordered
end
def is_editable_by_user?(user)
if !user || user.id != self.user_id
return false
end
(Time.now.to_i - self.created_at.to_i < (60 * MAX_EDIT_MINS))
end
def url
self.story.comments_url + "/comments/#{self.short_id}"
end
end