require 'lib/diaspora/user/friending.rb' class User include MongoMapper::Document include Diaspora::UserModules::Friending devise :database_authenticatable, :registerable, :recoverable, :rememberable, :trackable, :validatable key :username, :unique => true key :friend_ids, Array key :pending_request_ids, Array key :visible_post_ids, Array key :visible_person_ids, Array one :person, :class_name => 'Person', :foreign_key => :owner_id many :friends, :in => :friend_ids, :class_name => 'Person' many :visible_people, :in => :visible_person_ids, :class_name => 'Person' # One of these needs to go many :pending_requests, :in => :pending_request_ids, :class_name => 'Request' many :raw_visible_posts, :in => :visible_post_ids, :class_name => 'Post' many :groups, :class_name => 'Group' before_validation_on_create :setup_person before_validation :do_bad_things def self.find_for_authentication(conditions={}) if conditions[:username] =~ /^([\w\.%\+\-]+)@([\w\-]+\.)+([\w]{2,})$/i # email regex conditions[:email] = conditions.delete(:username) end super end ######## Making things work ######## key :email, String ensure_index :email def method_missing(method, *args) self.person.send(method, *args) end def real_name "#{person.profile.first_name.to_s} #{person.profile.last_name.to_s}" end ######### Groups ###################### def group( opts = {} ) opts[:user] = self Group.create(opts) end def move_friend( opts = {}) return true if opts[:to] == opts[:from] friend = Person.first(:_id => opts[:friend_id]) if self.friend_ids.include?(friend.id) from_group = self.group_by_id(opts[:from]) to_group = self.group_by_id(opts[:to]) if from_group && to_group posts_to_move = from_group.posts.find_all_by_person_id(friend.id) to_group.people << friend to_group.posts << posts_to_move from_group.person_ids.delete(friend.id.to_id) posts_to_move.each{ |x| from_group.post_ids.delete(x.id)} from_group.save to_group.save return true end end false end ##querying with permissions def posts_visible_to_me(opts ={}) if opts[:from].class == Person Post.where(:person_id => opts[:from].id, :_id.in => self.visible_post_ids) elsif opts[:from].class == Group Post.where(:_id.in => opts[:from].post_ids) unless opts[:from].user != self else Post.where(:_id.in => self.visible_post_ids) end end ######## Posting ######## def post(class_name, options = {}) options[:person] = self.person if class_name == :photo raise ArgumentError.new("No album_id given") unless options[:album_id] group_ids = groups_with_post( options[:album_id] ) group_ids.map!{ |group| group.id } else group_ids = options.delete(:to) end group_ids = [group_ids] if group_ids.is_a? BSON::ObjectId raise ArgumentError.new("You must post to someone.") if group_ids.nil? || group_ids.empty? model_class = class_name.to_s.camelize.constantize post = model_class.instantiate(options) post.creator_signature = post.sign_with_key(encryption_key) post.save post.socket_to_uid(id, :group_ids => group_ids) if post.respond_to?(:socket_to_uid) push_to_groups(post, group_ids) self.raw_visible_posts << post self.save post end def push_to_groups( post, group_ids ) if group_ids == :all || group_ids == "all" groups = self.groups else groups = self.groups.find_all_by_id( group_ids ) end #send to the groups target_people = [] groups.each{ |group| group.posts << post group.save target_people = target_people | group.people } post.push_to( target_people ) end def visible_posts( opts = {} ) if opts[:by_members_of] return raw_visible_posts if opts[:by_members_of] == :all group = self.groups.find_by_id( opts[:by_members_of].id ) group.posts end end ######## Commenting ######## def comment(text, options = {}) raise "must comment on something!" unless options[:on] comment = Comment.new(:person_id => self.person.id, :text => text, :post => options[:on]) comment.creator_signature = comment.sign_with_key(encryption_key) if comment.save dispatch_comment comment comment.socket_to_uid id comment else Rails.logger.warn "this failed to save: #{comment.inspect}" false end end def dispatch_comment( comment ) if owns? comment.post comment.post_creator_signature = comment.sign_with_key(encryption_key) comment.save comment.push_downstream elsif owns? comment comment.save comment.push_upstream end end ######### Posts and Such ############### def retract( post ) post.unsocket_from_uid(self.id) if post.respond_to? :unsocket_from_uid retraction = Retraction.for(post) retraction.creator_signature = retraction.sign_with_key( encryption_key ) retraction.push_to( self.friends.all ) retraction end ########### Profile ###################### def update_profile(params) params[:profile].delete(:image_url) if params[:profile][:image_url].empty? if self.person.update_attributes(params) self.profile.push_to( self.friends.all ) true else false end end ###### Receiving ####### def receive xml object = Diaspora::Parser.from_xml(xml) Rails.logger.debug("Receiving object for #{self.real_name}:\n#{object.inspect}") Rails.logger.debug("From: #{object.person.inspect}") if object.person raise "In receive for #{self.real_name}, signature was not valid on: #{object.inspect}" unless object.signature_valid? if object.is_a? Retraction if object.type == 'Person' && object.signature_valid? Rails.logger.info( "the person id is #{object.post_id} the friend found is #{visible_person_by_id(object.post_id).inspect}") unfriended_by visible_person_by_id(object.post_id) else object.perform self.id groups = self.groups_with_person(object.person) groups.each{ |group| group.post_ids.delete(object.post_id.to_id) group.save } end elsif object.is_a? Request person = Diaspora::Parser.parse_or_find_person_from_xml( xml ) person.serialized_key ||= object.exported_key object.person = person object.person.save old_request = Request.first(:id => object.id) object.group_id = old_request.group_id if old_request object.save receive_friend_request(object) elsif object.is_a? Profile person = Diaspora::Parser.owner_id_from_xml xml person.profile = object person.save elsif object.is_a?(Comment) object.person = Diaspora::Parser.parse_or_find_person_from_xml( xml ).save if object.person.nil? self.visible_people << object.person self.save Rails.logger.debug("The person parsed from comment xml is #{object.person.inspect}") unless object.person.nil? object.person.save Rails.logger.debug("From: #{object.person.inspect}") if object.person raise "In receive for #{self.real_name}, signature was not valid on: #{object.inspect}" unless object.post.person == self.person || object.verify_post_creator_signature object.save dispatch_comment object unless owns?(object) object.socket_to_uid(id) if (object.respond_to?(:socket_to_uid) && !self.owns?(object)) else Rails.logger.debug("Saving object: #{object}") object.user_refs += 1 object.save self.raw_visible_posts << object self.save groups = self.groups_with_person(object.person) groups.each{ |group| group.posts << object group.save object.socket_to_uid(id, :group_id => group.id) if (object.respond_to?(:socket_to_uid) && !self.owns?(object)) } end end ###Helpers############ def self.instantiate!( opts = {} ) opts[:person][:email] = opts[:email] opts[:person][:serialized_key] = generate_key User.create!( opts) end def terse_url terse= self.url.gsub(/https?:\/\//, '') terse.gsub!(/www\./, '') terse = terse.chop! if terse[-1, 1] == '/' terse end def diaspora_handle "#{self.username}@#{self.terse_url}" end def do_bad_things self.password_confirmation = self.password end def visible_person_by_id( id ) id = id.to_id return self.person if id == self.person.id result = friends.detect{|x| x.id == id } result = visible_people.detect{|x| x.id == id } unless result result end def group_by_id( id ) id = id.to_id groups.detect{|x| x.id == id } end def album_by_id( id ) id = id.to_id albums.detect{|x| x.id == id } end def groups_with_post( id ) self.groups.find_all_by_post_ids( id.to_id ) end def groups_with_person person id = person.id.to_id groups.select { |g| g.person_ids.include? id} end def setup_person self.person.serialized_key ||= User.generate_key.export self.person.email ||= email self.person.save! end def all_group_ids self.groups.all.collect{|x| x.id} end def as_json(opts={}) { :user => { :posts => self.raw_visible_posts.each{|post| post.as_json}, :friends => self.friends.each {|friend| friend.as_json}, :groups => self.groups.each {|group| group.as_json}, :pending_requests => self.pending_requests.each{|request| request.as_json}, } } end protected def self.generate_key OpenSSL::PKey::RSA::generate 1024 end end