284 lines
8 KiB
Ruby
284 lines
8 KiB
Ruby
# Copyright (c) 2010, Diaspora Inc. This file is
|
|
# licensed under the Affero General Public License version 3. See
|
|
# the COPYRIGHT file.
|
|
|
|
require File.expand_path('../../../lib/diaspora/user/friending', __FILE__)
|
|
require File.expand_path('../../../lib/diaspora/user/querying', __FILE__)
|
|
require File.expand_path('../../../lib/diaspora/user/receiving', __FILE__)
|
|
require File.expand_path('../../../lib/salmon/salmon', __FILE__)
|
|
|
|
class User
|
|
include MongoMapper::Document
|
|
include Diaspora::UserModules::Friending
|
|
include Diaspora::UserModules::Querying
|
|
include Diaspora::UserModules::Receiving
|
|
include Encryptor::Private
|
|
QUEUE = MessageHandler.new
|
|
|
|
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 :aspects, :class_name => 'Aspect'
|
|
|
|
after_create :seed_aspects
|
|
|
|
before_validation_on_create :downcase_username
|
|
|
|
def self.find_for_authentication(conditions={})
|
|
if conditions[:username] =~ /^([\w\.%\+\-]+)@([\w\-]+\.)+([\w]{2,})$/i # email regex
|
|
conditions[:email] = conditions.delete(:username)
|
|
else
|
|
conditions[:username].downcase!
|
|
end
|
|
super
|
|
end
|
|
|
|
######## Making things work ########
|
|
key :email, String
|
|
|
|
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
|
|
|
|
######### Aspects ######################
|
|
def aspect( opts = {} )
|
|
opts[:user] = self
|
|
Aspect.create(opts)
|
|
end
|
|
|
|
def drop_aspect( aspect )
|
|
if aspect.people.size == 0
|
|
aspect.destroy
|
|
else
|
|
raise "Aspect not empty"
|
|
end
|
|
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_aspect = self.aspect_by_id(opts[:from])
|
|
to_aspect = self.aspect_by_id(opts[:to])
|
|
if from_aspect && to_aspect
|
|
posts_to_move = from_aspect.posts.find_all_by_person_id(friend.id)
|
|
to_aspect.people << friend
|
|
to_aspect.posts << posts_to_move
|
|
from_aspect.person_ids.delete(friend.id.to_id)
|
|
posts_to_move.each{ |x| from_aspect.post_ids.delete(x.id)}
|
|
from_aspect.save
|
|
to_aspect.save
|
|
return true
|
|
end
|
|
end
|
|
false
|
|
end
|
|
|
|
######## Posting ########
|
|
def post(class_name, options = {})
|
|
if class_name == :photo
|
|
raise ArgumentError.new("No album_id given") unless options[:album_id]
|
|
aspect_ids = aspects_with_post( options[:album_id] )
|
|
aspect_ids.map!{ |aspect| aspect.id }
|
|
else
|
|
aspect_ids = options.delete(:to)
|
|
end
|
|
|
|
aspect_ids = validate_aspect_permissions(aspect_ids)
|
|
|
|
intitial_post(class_name, aspect_ids, options)
|
|
end
|
|
|
|
def intitial_post(class_name, aspect_ids, options = {})
|
|
post = build_post(class_name, options)
|
|
post.socket_to_uid(id, :aspect_ids => aspect_ids) if post.respond_to?(:socket_to_uid)
|
|
push_to_aspects(post, aspect_ids)
|
|
post
|
|
end
|
|
|
|
def repost( post, options = {} )
|
|
aspect_ids = validate_aspect_permissions(options[:to])
|
|
push_to_aspects(post, aspect_ids)
|
|
post
|
|
end
|
|
|
|
def update_post( post, post_hash = {} )
|
|
if self.owns? post
|
|
post.update_attributes(post_hash)
|
|
end
|
|
end
|
|
|
|
def validate_aspect_permissions(aspect_ids)
|
|
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.")
|
|
end
|
|
end
|
|
|
|
aspect_ids
|
|
end
|
|
|
|
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
|
|
|
|
def push_to_aspects( post, aspect_ids )
|
|
if aspect_ids == :all || aspect_ids == "all"
|
|
aspects = self.aspects
|
|
elsif aspect_ids.is_a?(Array) && aspect_ids.first.class == Aspect
|
|
aspects = aspect_ids
|
|
else
|
|
aspects = self.aspects.find_all_by_id( aspect_ids )
|
|
end
|
|
#send to the aspects
|
|
target_people = []
|
|
|
|
aspects.each{ |aspect|
|
|
aspect.posts << post
|
|
aspect.save
|
|
target_people = target_people | aspect.people
|
|
}
|
|
push_to_people(post, target_people)
|
|
end
|
|
|
|
def push_to_people(post, people)
|
|
people.each{|person|
|
|
salmon(post, :to => person)
|
|
}
|
|
end
|
|
|
|
def push_to_person( person, xml )
|
|
Rails.logger.debug("Adding xml for #{self} to message queue to #{url}")
|
|
QUEUE.add_post_request( person.receive_url, person.encrypt(xml) )
|
|
QUEUE.process
|
|
|
|
end
|
|
|
|
def salmon( post, opts = {} )
|
|
salmon = Salmon::SalmonSlap.create(self, post.to_diaspora_xml)
|
|
push_to_person( opts[:to], salmon.to_xml)
|
|
salmon
|
|
end
|
|
|
|
######## Commenting ########
|
|
def comment(text, options = {})
|
|
comment = build_comment(text, options)
|
|
if comment
|
|
dispatch_comment comment
|
|
comment.socket_to_uid id
|
|
end
|
|
comment
|
|
end
|
|
|
|
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
|
|
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
|
|
push_to_people comment, people_in_aspects(aspects_with_post(comment.post.id))
|
|
elsif owns? comment
|
|
comment.save
|
|
salmon comment, :to => comment.post.person
|
|
end
|
|
end
|
|
|
|
######### Posts and Such ###############
|
|
def retract( post )
|
|
aspect_ids = aspects_with_post( post.id )
|
|
aspect_ids.map!{|aspect| aspect.id.to_s}
|
|
|
|
post.unsocket_from_uid(self.id, :aspect_ids => aspect_ids) if post.respond_to? :unsocket_from_uid
|
|
retraction = Retraction.for(post)
|
|
push_to_people retraction, people_in_aspects(aspects_with_post(post.id))
|
|
retraction
|
|
end
|
|
|
|
########### Profile ######################
|
|
def update_profile(params)
|
|
if self.person.update_attributes(params)
|
|
push_to_aspects profile, :all
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
###Helpers############
|
|
def self.instantiate!( opts = {} )
|
|
opts[:person][:diaspora_handle] = "#{opts[:username]}@#{APP_CONFIG[:terse_pod_url]}"
|
|
opts[:person][:url] = APP_CONFIG[:pod_url]
|
|
opts[:person][:serialized_key] = generate_key
|
|
User.create(opts)
|
|
end
|
|
|
|
def seed_aspects
|
|
aspect(:name => "Family")
|
|
aspect(:name => "Work")
|
|
end
|
|
|
|
def diaspora_handle
|
|
"#{self.username}@#{APP_CONFIG[:terse_pod_url]}"
|
|
end
|
|
|
|
def downcase_username
|
|
username.downcase! if username
|
|
end
|
|
|
|
def as_json(opts={})
|
|
{
|
|
:user => {
|
|
: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},
|
|
}
|
|
}
|
|
end
|
|
def self.generate_key
|
|
OpenSSL::PKey::RSA::generate 4096
|
|
end
|
|
end
|