Skip to content
Extraits de code Groupes Projets
user.rb 8,48 ko
Newer Older
require 'lib/diaspora/user/friending.rb'

class User
  include MongoMapper::Document
  include Diaspora::UserModules::Friending
  devise :database_authenticatable, :registerable,
         :recoverable, :rememberable, :trackable, :validatable
maxwell's avatar
maxwell a validé
         
maxwell's avatar
maxwell a validé
  key :pending_request_ids, Array
Raphael's avatar
Raphael a validé
  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
maxwell's avatar
maxwell a validé
  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 
  ######## Making things work ########
  key :email, String
Raphael's avatar
Raphael a validé
  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}"
  ######### Groups ######################
  def group( opts = {} )
    opts[:user] = self
    Group.create(opts)
  end
maxwell's avatar
maxwell a validé
  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
ilya's avatar
ilya a validé
        from_group.person_ids.delete(friend.id.to_id)
maxwell's avatar
maxwell a validé
        posts_to_move.each{ |x| from_group.post_ids.delete(x.id)}
        from_group.save
        to_group.save
        return true
      end
    end
    false
  end
  ######## Posting ########
  def post(class_name, options = {})
    options[:person] = self.person
      raise ArgumentError("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
Raphael's avatar
Raphael a validé

    group_ids = [group_ids] if group_ids.is_a? BSON::ObjectID
    raise ArgumentError("You must post to someone.") if group_ids.nil? || group_ids.empty?
Raphael's avatar
Raphael a validé

    model_class = class_name.to_s.camelize.constantize
    post = model_class.instantiate(options)
    post.creator_signature = post.sign_with_key(encryption_key)
ilya's avatar
ilya a validé
    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
Raphael's avatar
Raphael a validé
    target_people = [] 

    groups.each{ |group|
Raphael's avatar
Raphael a validé
      target_people = target_people | group.people
    }
    post.push_to( target_people )
  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 )

  ######## 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
      Rails.logger.warn "this failed to save: #{comment.inspect}"
  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
  
  ######### Posts and Such ###############
Raphael's avatar
Raphael a validé
    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 )
  ########### 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:\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?
Raphael's avatar
Raphael a 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)
    elsif object.is_a? Request
      person = Diaspora::Parser.parse_or_find_person_from_xml( xml )
Raphael's avatar
Raphael a validé
      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
Raphael's avatar
Raphael a validé
      object.save
      receive_friend_request(object)
    elsif object.is_a? Profile
      person = Diaspora::Parser.owner_id_from_xml xml
Raphael's avatar
Raphael a validé
      person.profile = object
      person.save  
    elsif object.is_a?(Comment) 
Raphael's avatar
Raphael a validé
      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?
Raphael's avatar
Raphael a validé
      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
Raphael's avatar
Raphael a validé
      object.save
      dispatch_comment object unless owns?(object)
Raphael's avatar
Raphael a validé
      object.socket_to_uid(id)  if (object.respond_to?(:socket_to_uid) && !self.owns?(object))
      Rails.logger.debug("Saving object: #{object}")
      
      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))
  def self.instantiate!( opts = {} )
    opts[:person][:email] = opts[:email]
    opts[:person][:serialized_key] = generate_key
  def terse_url
    terse= self.url.gsub(/https?:\/\//, '')
    terse.gsub!(/www\./, '')
    terse = terse.chop! if terse[-1, 1] == '/'
    terse
  end

  def do_bad_things
    self.password_confirmation = self.password
  end 

  def visible_person_by_id( 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
    groups.detect{|x| x.id == id }
Raphael's avatar
Raphael a validé

  def album_by_id( id )
    albums.detect{|x| x.id == id }
  end

Raphael's avatar
Raphael a validé
    self.groups.find_all_by_post_ids( id.to_id )
  def groups_with_person person
    id = person.id.to_id
    groups.select { |g| g.person_ids.include? id}
    self.person.serialized_key ||= User.generate_key.export
Raphael's avatar
Raphael a validé
    self.person.email ||= email
ilya's avatar
ilya a validé
  def all_group_ids
ilya's avatar
ilya a validé
    self.groups.all.collect{|x| x.id}
ilya's avatar
ilya a validé
  end
Raphael's avatar
Raphael a validé
  def self.generate_key
    OpenSSL::PKey::RSA::generate 1024 
ilya's avatar
ilya a validé
  end