class SearchObserver < ActiveRecord::Observer observe :topic, :post, :user, :category def self.scrub_html_for_search(html) HtmlScrubber.scrub(html) end def self.update_index(table, id, idx) Post.exec_sql("delete from #{table} where id = ?", id) sql = "insert into #{table} (id, search_data) values (?, to_tsvector('english', ?))" begin Post.exec_sql(sql, id, idx) rescue # don't allow concurrency to mess up saving a post end end def self.update_posts_index(post_id, cooked, title, category) idx = scrub_html_for_search(cooked) idx << " " << title idx << " " << category if category update_index('posts_search', post_id, idx) end def self.update_users_index(user_id, username, name) idx = username.dup idx << " " << (name || "") update_index('users_search', user_id, idx) end def self.update_categories_index(category_id, name) update_index('categories_search', category_id, name) end def after_save(obj) if obj.class == Post && obj.cooked_changed? category_name = obj.topic.category.name if obj.topic.category SearchObserver.update_posts_index(obj.id, obj.cooked, obj.topic.title, category_name) end if obj.class == User && (obj.username_changed? || obj.name_changed?) SearchObserver.update_users_index(obj.id, obj.username, obj.name) end if obj.class == Topic && obj.title_changed? if obj.posts post = obj.posts.where(post_number: 1).first if post category_name = obj.category.name if obj.category SearchObserver.update_posts_index(post.id, post.cooked, obj.title, category_name) end end end if obj.class == Category && obj.name_changed? SearchObserver.update_categories_index(obj.id, obj.name) end end class HtmlScrubber < Nokogiri::XML::SAX::Document attr_reader :scrubbed def initialize @scrubbed = "" end def self.scrub(html) me = self.new parser = Nokogiri::HTML::SAX::Parser.new(me) begin copy = "