Skip to content
Extraits de code Groupes Projets
user.rb 13,4 ko
Newer Older
  • Learn to ignore specific revisions
  • Raphael's avatar
    Raphael a validé
    #   Copyright (c) 2010, Diaspora Inc.  This file is
    
    Raphael's avatar
    Raphael a validé
    #   licensed under the Affero General Public License version 3 or later.  See
    
    Raphael's avatar
    Raphael a validé
    #   the COPYRIGHT file.
    
    require File.join(Rails.root, 'lib/diaspora/user')
    
    require File.join(Rails.root, 'lib/salmon/salmon')
    
    ilya's avatar
    ilya a validé
    class InvitedUserValidator < ActiveModel::Validator
      def validate(document)
        unless document.invitation_token
          unless document.person
            document.errors[:base] << "Unless you are being invited, you must have a person"
          end
        end
      end
    end
    
    
    class User
      include MongoMapper::Document
    
      include Diaspora::UserModules
    
    Raphael's avatar
    Raphael a validé
      include Encryptor::Private
    
    
      plugin MongoMapper::Devise
    
    
    Raphael's avatar
    Raphael a validé
    
    
    ilya's avatar
    ilya a validé
      devise :invitable, :database_authenticatable, :registerable,
    
             :recoverable, :rememberable, :trackable, :validatable
    
      key :serialized_private_key, String
    
    Sarah Mei's avatar
    Sarah Mei a validé
      key :invites, Integer, :default => 5
      key :invitation_token, String
      key :invitation_sent_at, DateTime
      key :inviter_ids, Array
      key :friend_ids, Array
    
    maxwell's avatar
    maxwell a validé
      key :pending_request_ids, Array
    
    Sarah Mei's avatar
    Sarah Mei a validé
      key :visible_post_ids, Array
      key :visible_person_ids, Array
    
      key :invite_messages, Hash
    
    
      before_validation :strip_username, :on => :create
    
      validates_uniqueness_of :username, :case_sensitive => false
    
      validates_format_of :username, :with => /\A[A-Za-z0-9_.]+\z/ 
    
      validates_with InvitedUserValidator
    
    
      one :person, :class_name => 'Person', :foreign_key => :owner_id
    
      validate :person_is_valid
      def person_is_valid
        if person.present? && !person.valid?
          person.errors.full_messages.each {|m| errors.add(:base, m)}
        end
      end
    
    Sarah Mei's avatar
    Sarah Mei a validé
      many :inviters, :in => :inviter_ids, :class_name => 'User'
      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 :aspects, :class_name => 'Aspect', :dependent => :destroy
    
    maxwell's avatar
    maxwell a validé
      many :services, :class_name => "OmniauthService"
    
    
      #after_create :seed_aspects
    
      before_destroy :unfriend_everyone, :remove_person
    
    ilya's avatar
    ilya a validé
    
    
      def strip_username
        if username.present?
          username.strip!
        end
      end
    
    
    Sarah Mei's avatar
    Sarah Mei a validé
      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 ########
    
      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}"
    
    Raphael's avatar
    Raphael a validé
      ######### Aspects ######################
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def aspect(opts = {})
    
        aspect = Aspect.new(opts)
        aspect.user = self
        aspect.save
        aspect
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def drop_aspect(aspect)
    
    danielvincent's avatar
    danielvincent a validé
        if aspect.people.size == 0
          aspect.destroy
    
    danielvincent's avatar
    danielvincent a validé
          raise "Aspect not empty"
        end
      end
    
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def move_friend(opts = {})
    
        return true if opts[:to] == opts[:from]
    
        if opts[:friend_id] && opts[:to] && opts[:from] 
          from_aspect = self.aspects.first(:_id => opts[:from])
          posts_to_move = from_aspect.posts.find_all_by_person_id(opts[:friend_id])
          if add_person_to_aspect(opts[:friend_id], opts[:to], :posts => posts_to_move)
            delete_person_from_aspect(opts[:friend_id], opts[:from], :posts => posts_to_move) 
    
            return true
          end
        end
        false
      end
    
      def add_person_to_aspect(person_id, aspect_id, opts = {})
    
        raise "Can not add person to an aspect you do not own" unless aspect = self.aspects.find_by_id(aspect_id) 
        raise "Can not add person you are not friends with" unless person = self.find_friend_by_id(person_id)
        raise 'Can not add person who is already in the aspect' if aspect.person_ids.include?(person_id)
        aspect.people << person 
    
        opts[:posts] ||= self.raw_visible_posts.all(:person_id => person_id)
        
        aspect.posts += opts[:posts]
    
      def delete_person_from_aspect(person_id, aspect_id, opts = {})
    
        raise "Can not delete a person from an aspect you do not own" unless aspect = self.aspects.find_by_id(aspect_id)
    
        aspect.person_ids.delete(person_id.to_id)
    
        opts[:posts] ||= aspect.posts.all(:person_id => person_id)
        aspect.posts -= opts[:posts]
    
      ######## Posting ########
    
      def post(class_name, options = {})
    
    Raphael's avatar
    Raphael a validé
          raise ArgumentError.new("No album_id given") unless options[:album_id]
    
    Sarah Mei's avatar
    Sarah Mei a validé
          aspect_ids = aspects_with_post(options[:album_id])
          aspect_ids.map! { |aspect| aspect.id }
    
    Raphael's avatar
    Raphael a validé
          aspect_ids = options.delete(:to)
    
        aspect_ids = validate_aspect_permissions(aspect_ids)
    
        intitial_post(class_name, aspect_ids, options)
      end
    
      def post_to_facebook(message)
        facebook = self.services.find_by_provider("facebook")
        if facebook
          Rails.logger.info("Sending a message: #{message} to Facebook")
          EventMachine::HttpRequest.new("https://graph.facebook.com/me/feed?message=#{message}&access_token=#{facebook.access_token}").post
        end
    
      def post_to_twitter(message)
        twitter = self.services.find_by_provider("twitter")
        if twitter
          oauth = Twitter::OAuth.new(SERVICES['twitter']['consumer_token'], SERVICES['twitter']['consumer_secret'])
          oauth.authorize_from_access(twitter.access_token, twitter.access_secret)
          client = Twitter::Base.new(oauth)
          client.update(message)
        end
      end
    
    
      def intitial_post(class_name, aspect_ids, options = {})
    
        post = build_post(class_name, options)
    
    Raphael's avatar
    Raphael a validé
        post.socket_to_uid(id, :aspect_ids => aspect_ids) if post.respond_to?(:socket_to_uid)
        push_to_aspects(post, aspect_ids)
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def update_post(post, post_hash = {})
    
    danielvincent's avatar
    danielvincent a validé
        if self.owns? post
    
    danielvincent's avatar
    danielvincent a validé
          post.update_attributes(post_hash)
    
      def validate_aspect_permissions(aspect_ids)
    
    Raphael's avatar
    Raphael a validé
        if aspect_ids == "all"
          return aspect_ids
        end
    
        aspect_ids = [aspect_ids.to_s] unless aspect_ids.is_a? Array
    
    
        if aspect_ids.nil? || aspect_ids.empty?
          raise ArgumentError.new("You must post to someone.")
        end
    
        aspect_ids.each do |aspect_id|
    
          unless self.aspects.find(aspect_id)
    
            raise ArgumentError.new("Cannot post to an aspect you do not own.")
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def build_post(class_name, options = {})
    
        options[:person] = self.person
        model_class = class_name.to_s.camelize.constantize
        post = model_class.instantiate(options)
        post.save
    
        self.raw_visible_posts << post
        self.save
        post
      end
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def push_to_aspects(post, aspect_ids)
    
    Raphael's avatar
    Raphael a validé
        if aspect_ids == :all || aspect_ids == "all"
          aspects = self.aspects
        elsif aspect_ids.is_a?(Array) && aspect_ids.first.class == Aspect
          aspects = aspect_ids
    
    Sarah Mei's avatar
    Sarah Mei a validé
          aspects = self.aspects.find_all_by_id(aspect_ids)
    
    Raphael's avatar
    Raphael a validé
        #send to the aspects
    
        target_people = []
    
    Raphael's avatar
    Raphael a validé
    
    
    Sarah Mei's avatar
    Sarah Mei a validé
        aspects.each { |aspect|
    
    Raphael's avatar
    Raphael a validé
          aspect.posts << post
          aspect.save
          target_people = target_people | aspect.people
    
    Raphael's avatar
    Raphael a validé
        }
    
    danielvincent's avatar
    danielvincent a validé
        push_to_hub(post) if post.respond_to?(:public) && post.public
    
    
        push_to_people(post, target_people)
      end
    
      def push_to_people(post, people)
    
        salmon = salmon(post)
    
    Sarah Mei's avatar
    Sarah Mei a validé
        people.each { |person|
    
          xml = salmon.xml_for person
    
    Sarah Mei's avatar
    Sarah Mei a validé
          push_to_person(person, xml)
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def push_to_person(person, xml)
    
    Raphael's avatar
    Raphael a validé
        Rails.logger.debug("#{self.real_name} is adding xml to message queue to #{person.receive_url}")
    
    Sarah Mei's avatar
    Sarah Mei a validé
        QUEUE.add_post_request(person.receive_url, xml)
    
    danielvincent's avatar
    danielvincent a validé
        QUEUE.process
      end
    
    danielvincent's avatar
    danielvincent a validé
      def push_to_hub(post)
    
    danielvincent's avatar
    danielvincent a validé
        Rails.logger.debug("Pushing update to pubsub server #{APP_CONFIG[:pubsub_server]} with url #{self.public_url}")
    
    danielvincent's avatar
    danielvincent a validé
        QUEUE.add_hub_notification(APP_CONFIG[:pubsub_server], self.public_url)
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def salmon(post)
    
        created_salmon = Salmon::SalmonSlap.create(self, post.to_diaspora_xml)
        created_salmon
    
    Raphael's avatar
    Raphael a validé
      end
    
    
      ######## Commenting  ########
      def comment(text, options = {})
    
        comment = build_comment(text, options)
        if comment
          dispatch_comment comment
          comment.socket_to_uid id
        end
        comment
      end
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def build_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
          comment
    
          Rails.logger.warn "this failed to save: #{comment.inspect}"
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def dispatch_comment(comment)
    
        if owns? comment.post
          comment.post_creator_signature = comment.sign_with_key(encryption_key)
          comment.save
    
    Raphael's avatar
    Raphael a validé
          push_to_people comment, people_in_aspects(aspects_with_post(comment.post.id))
    
        elsif owns? comment
          comment.save
    
          push_to_people comment, [comment.post.person]
    
      ######### Posts and Such ###############
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def retract(post)
        aspect_ids = aspects_with_post(post.id)
        aspect_ids.map! { |aspect| aspect.id.to_s }
    
    Raphael's avatar
    Raphael a validé
        post.unsocket_from_uid(self.id, :aspect_ids => aspect_ids) if post.respond_to? :unsocket_from_uid
    
    Raphael's avatar
    Raphael a validé
        push_to_people retraction, people_in_aspects(aspects_with_post(post.id))
    
      ########### Profile ######################
      def update_profile(params)
    
        if self.person.profile.update_attributes(params)
    
    Raphael's avatar
    Raphael a validé
          push_to_aspects profile, :all
    
      ###Invitations############
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def invite_user(opts = {})
    
    
          aspect_id = opts.delete(:aspect_id)
          if aspect_id == nil
            raise "Must invite into aspect"
    
          end
          aspect_object = self.aspects.find_by_id(aspect_id)
          if !(aspect_object)
    
          else
            u = User.find_by_email(opts[:email])
            if u.nil?  
            elsif friends.include?(u.person)
              raise "You are already friends with this person"          
            elsif not u.invited?
              self.send_friend_request_to(u.person, aspect_object)
              return
            elsif u.invited? && u.inviters.include?(self)
              raise "You already invited this person"          
            end
    
    Sarah Mei's avatar
    Sarah Mei a validé
            :to => "http://local_request.example.com",
            :from => self.person,
            :into => aspect_id
    
          invited_user = User.invite!(:email => opts[:email], :request => request, :inviter => self, :invite_message => opts[:invite_message])
    
          self.invites = self.invites - 1
    
          self.save!
          invited_user
        else
          raise "You have no invites"
        end
    
    ilya's avatar
    ilya a validé
      end
    
      def self.invite!(attributes={})
        inviter = attributes.delete(:inviter)
    
    ilya's avatar
    ilya a validé
        invitable = find_or_initialize_with_error_by(:email, attributes.delete(:email))
        invitable.attributes = attributes
    
        if invitable.inviters.include?(inviter)
          raise "You already invited this person"
        else
    
          invitable.inviters << inviter
    
          message = attributes.delete(:invite_message)
          if message
            invitable.invite_messages[inviter.id.to_s] = message
          end
    
    ilya's avatar
    ilya a validé
    
        if invitable.new_record?
          invitable.errors.clear if invitable.email.try(:match, Devise.email_regexp)
        else
          invitable.errors.add(:email, :taken) unless invitable.invited?
        end
    
        invitable.invite! if invitable.errors.empty?
        invitable
      end
    
    
    Sarah Mei's avatar
    Sarah Mei a validé
      def accept_invitation!(opts = {})
    
    ilya's avatar
    ilya a validé
        if self.invited?
    
    ilya's avatar
    ilya a validé
          self.username              = opts[:username]
    
    ilya's avatar
    ilya a validé
          self.password              = opts[:password]
          self.password_confirmation = opts[:password_confirmation]
          opts[:person][:diaspora_handle] = "#{opts[:username]}@#{APP_CONFIG[:terse_pod_url]}"
          opts[:person][:url] = APP_CONFIG[:pod_url]
    
          opts[:serialized_private_key] = User.generate_key
          self.serialized_private_key =  opts[:serialized_private_key]
          opts[:person][:serialized_public_key] = opts[:serialized_private_key].public_key
    
          person_hash = opts.delete(:person)
          self.person = Person.create(person_hash)
          self.person.save
    
          self.invitation_token = nil
    
    ilya's avatar
    ilya a validé
          self.save
          self
        end
      end
    
    
      ###Helpers############
    
        opts[:person][:diaspora_handle] = "#{opts[:username]}@#{APP_CONFIG[:terse_pod_url]}"
    
    ilya's avatar
    ilya a validé
        opts[:person][:url] = APP_CONFIG[:pod_url]
    
    Raphael's avatar
    Raphael a validé
    
    
        opts[:serialized_private_key] = generate_key
        opts[:person][:serialized_public_key] = opts[:serialized_private_key].public_key
    
    
      def seed_aspects
        aspect(:name => "Family")
        aspect(:name => "Work")
    
        "#{self.username}@#{APP_CONFIG[:terse_pod_url]}"
    
      def as_json(opts={})
        {
          :user => {
    
    Sarah Mei's avatar
    Sarah Mei a validé
            :posts            => self.raw_visible_posts.each { |post| post.as_json },
            :friends          => self.friends.each { |friend| friend.as_json },
            :aspects           => self.aspects.each { |aspect| aspect.as_json },
            :pending_requests => self.pending_requests.each { |request| request.as_json },
    
    
    
      def self.generate_key
        OpenSSL::PKey::RSA::generate 4096
      end
    
    Raphael's avatar
    Raphael a validé
    
    
    Sarah Mei's avatar
    Sarah Mei a validé
        OpenSSL::PKey::RSA.new(serialized_private_key)
    
    ilya's avatar
    ilya a validé
      protected
    
      def remove_person
        self.person.destroy
      end
    
    ilya's avatar
    ilya a validé
      def unfriend_everyone
    
    Sarah Mei's avatar
    Sarah Mei a validé
        friends.each { |friend|
    
    ilya's avatar
    ilya a validé
          if friend.owner?
    
    ilya's avatar
    ilya a validé
            friend.owner.unfriended_by self.person
    
    Sarah Mei's avatar
    Sarah Mei a validé
          else
            self.unfriend friend
    
    ilya's avatar
    ilya a validé
          end
        }
      end