2016-09-24 03:00:46 +00:00
|
|
|
# frozen_string_literal: true
|
2017-11-25 19:23:47 +00:00
|
|
|
|
2016-09-21 17:22:40 +00:00
|
|
|
class Topic < ApplicationRecord
|
2017-02-09 21:53:19 +00:00
|
|
|
ATTRS_TO_WATCH = %w(name desc link metacode_id permission defer_to_map_id).freeze
|
2016-02-03 13:38:41 +00:00
|
|
|
include TopicsHelper
|
2017-01-24 20:10:40 +00:00
|
|
|
include Attachable
|
2016-02-03 13:38:41 +00:00
|
|
|
|
|
|
|
belongs_to :user
|
2016-07-26 00:14:23 +00:00
|
|
|
belongs_to :defer_to_map, class_name: 'Map', foreign_key: 'defer_to_map_id'
|
2017-02-09 21:53:19 +00:00
|
|
|
belongs_to :updated_by, class_name: 'User'
|
2016-02-03 13:38:41 +00:00
|
|
|
|
2016-09-28 02:32:28 +00:00
|
|
|
has_many :synapses1, class_name: 'Synapse', foreign_key: 'topic1_id', dependent: :destroy
|
|
|
|
has_many :synapses2, class_name: 'Synapse', foreign_key: 'topic2_id', dependent: :destroy
|
2016-07-26 00:14:23 +00:00
|
|
|
has_many :topics1, through: :synapses2, source: :topic1
|
|
|
|
has_many :topics2, through: :synapses1, source: :topic2
|
2016-02-03 13:38:41 +00:00
|
|
|
|
|
|
|
has_many :mappings, as: :mappable, dependent: :destroy
|
2016-07-26 00:14:23 +00:00
|
|
|
has_many :maps, through: :mappings
|
2017-02-11 05:20:42 +00:00
|
|
|
has_many :follows, as: :followed, dependent: :destroy
|
2017-11-25 19:23:47 +00:00
|
|
|
has_many :followers, through: :follows, source: :user
|
2016-02-03 13:38:41 +00:00
|
|
|
|
2016-04-24 15:50:35 +00:00
|
|
|
belongs_to :metacode
|
|
|
|
|
2017-01-28 21:40:41 +00:00
|
|
|
before_create :set_perm_by_defer
|
2016-10-06 13:12:01 +00:00
|
|
|
before_create :create_metamap?
|
2017-02-11 05:20:42 +00:00
|
|
|
after_create :after_created_async
|
2016-12-16 21:51:52 +00:00
|
|
|
after_update :after_updated
|
2017-02-11 05:20:42 +00:00
|
|
|
after_update :after_updated_async
|
2017-11-25 19:23:47 +00:00
|
|
|
# before_destroy :before_destroyed
|
2016-10-06 13:12:01 +00:00
|
|
|
|
2016-02-03 13:38:41 +00:00
|
|
|
validates :permission, presence: true
|
|
|
|
validates :permission, inclusion: { in: Perm::ISSIONS.map(&:to_s) }
|
2016-07-26 00:14:23 +00:00
|
|
|
|
2016-02-03 13:38:41 +00:00
|
|
|
def synapses
|
2016-09-30 06:42:07 +00:00
|
|
|
synapses1.or(synapses2)
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
2016-07-26 00:14:23 +00:00
|
|
|
|
2016-02-03 13:38:41 +00:00
|
|
|
def relatives
|
2016-09-30 06:42:07 +00:00
|
|
|
topics1.or(topics2)
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
|
|
|
|
2016-09-05 03:52:35 +00:00
|
|
|
scope :relatives, ->(topic_id = nil, user = nil) {
|
2016-09-05 03:55:19 +00:00
|
|
|
# should only see topics through *visible* synapses
|
|
|
|
# e.g. Topic A (commons) -> synapse (private) -> Topic B (commons) must be filtered out
|
2016-09-30 06:42:07 +00:00
|
|
|
topic_ids = Pundit.policy_scope(user, Synapse.where(topic1_id: topic_id)).pluck(:topic2_id)
|
|
|
|
topic_ids += Pundit.policy_scope(user, Synapse.where(topic2_id: topic_id)).pluck(:topic1_id)
|
|
|
|
where(id: topic_ids.uniq)
|
2016-09-05 03:52:35 +00:00
|
|
|
}
|
|
|
|
|
2016-07-26 00:14:23 +00:00
|
|
|
delegate :name, to: :user, prefix: true
|
2016-02-03 13:38:41 +00:00
|
|
|
|
|
|
|
def user_image
|
|
|
|
user.image.url
|
|
|
|
end
|
|
|
|
|
2016-10-23 14:11:38 +00:00
|
|
|
def map_count(user)
|
|
|
|
Pundit.policy_scope(user, maps).count
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
|
|
|
|
2016-10-23 14:11:38 +00:00
|
|
|
def synapse_count(user)
|
|
|
|
Pundit.policy_scope(user, synapses).count
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
|
|
|
|
2016-10-23 14:11:38 +00:00
|
|
|
def inmaps(user)
|
|
|
|
Pundit.policy_scope(user, maps).map(&:name)
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
|
|
|
|
2016-10-23 14:11:38 +00:00
|
|
|
def inmapsLinks(user)
|
|
|
|
Pundit.policy_scope(user, maps).map(&:id)
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
|
|
|
|
2016-10-23 14:11:38 +00:00
|
|
|
def as_json(options = {})
|
2017-11-25 19:23:47 +00:00
|
|
|
super(methods: %i(user_name user_image collaborator_ids))
|
2016-10-23 14:11:38 +00:00
|
|
|
.merge(inmaps: inmaps(options[:user]), inmapsLinks: inmapsLinks(options[:user]),
|
|
|
|
map_count: map_count(options[:user]), synapse_count: synapse_count(options[:user]))
|
2016-04-24 15:50:35 +00:00
|
|
|
end
|
|
|
|
|
2017-01-22 21:42:04 +00:00
|
|
|
def as_rdf
|
|
|
|
output = ''
|
|
|
|
output += %(d:topic_#{id} a mm:Topic ;\n)
|
|
|
|
output += %( rdfs:label "#{name}" ;\n)
|
|
|
|
output += %( rdfs:comment "#{desc}" ;\n) if desc.present?
|
|
|
|
output += %( foaf:homepage <#{link}> ;\n) if link.present?
|
|
|
|
output += %( mm:mapper d:mapper_#{user_id} ;\n)
|
|
|
|
output += %( mm:metacode "#{metacode.name}" ;\n)
|
|
|
|
output[-2] = '.' # change last ; to a .
|
|
|
|
output += %(\n)
|
|
|
|
output
|
|
|
|
end
|
|
|
|
|
2016-04-24 15:50:35 +00:00
|
|
|
def collaborator_ids
|
|
|
|
if defer_to_map
|
2017-11-25 19:23:47 +00:00
|
|
|
defer_to_map.editors.reject { |mapper| mapper == user }.map(&:id)
|
2016-04-24 15:50:35 +00:00
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
|
|
|
|
2017-01-03 21:12:58 +00:00
|
|
|
def filtered
|
|
|
|
{
|
|
|
|
id: id,
|
|
|
|
permission: permission,
|
|
|
|
user_id: user_id,
|
|
|
|
collaborator_ids: collaborator_ids
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2016-07-26 00:14:23 +00:00
|
|
|
# TODO: move to a decorator?
|
2016-03-25 08:35:43 +00:00
|
|
|
def synapses_csv(output_format = 'array')
|
|
|
|
output = []
|
|
|
|
synapses.each do |synapse|
|
|
|
|
if synapse.category == 'from-to'
|
2016-09-28 02:32:28 +00:00
|
|
|
if synapse.topic1_id == id
|
|
|
|
output << synapse.topic1_id.to_s + '->' + synapse.topic2_id.to_s
|
|
|
|
elsif synapse.topic2_id == id
|
|
|
|
output << synapse.topic2_id.to_s + '<-' + synapse.topic1_id.to_s
|
2016-03-25 08:35:43 +00:00
|
|
|
else
|
2016-07-26 00:14:23 +00:00
|
|
|
raise 'invalid synapse on topic in synapse_csv'
|
2016-03-25 08:35:43 +00:00
|
|
|
end
|
|
|
|
elsif synapse.category == 'both'
|
2016-09-28 02:32:28 +00:00
|
|
|
if synapse.topic1_id == id
|
|
|
|
output << synapse.topic1_id.to_s + '<->' + synapse.topic2_id.to_s
|
|
|
|
elsif synapse.topic2_id == id
|
|
|
|
output << synapse.topic2_id.to_s + '<->' + synapse.topic1_id.to_s
|
2016-03-25 08:35:43 +00:00
|
|
|
else
|
2016-07-26 00:14:23 +00:00
|
|
|
raise 'invalid synapse on topic in synapse_csv'
|
2016-03-25 08:35:43 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
if output_format == 'array'
|
|
|
|
return output
|
|
|
|
elsif output_format == 'text'
|
|
|
|
return output.join('; ')
|
|
|
|
else
|
2016-07-26 00:14:23 +00:00
|
|
|
raise 'invalid argument to synapses_csv'
|
2016-03-25 08:35:43 +00:00
|
|
|
end
|
|
|
|
output
|
|
|
|
end
|
|
|
|
|
2016-02-03 13:38:41 +00:00
|
|
|
def topic_autocomplete_method
|
2016-07-26 00:14:23 +00:00
|
|
|
"Get: #{name}"
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|
2016-07-26 00:14:23 +00:00
|
|
|
|
2016-10-06 13:12:01 +00:00
|
|
|
protected
|
2016-12-06 21:46:46 +00:00
|
|
|
|
2017-01-28 21:40:41 +00:00
|
|
|
def set_perm_by_defer
|
|
|
|
permission = defer_to_map.permission if defer_to_map
|
|
|
|
end
|
|
|
|
|
2016-12-06 21:46:46 +00:00
|
|
|
def create_metamap?
|
|
|
|
return unless (link == '') && (metacode.name == 'Metamap')
|
|
|
|
|
|
|
|
@map = Map.create(name: name, permission: permission, desc: '',
|
|
|
|
arranged: true, user_id: user_id)
|
|
|
|
self.link = Rails.application.routes.url_helpers
|
|
|
|
.map_url(host: ENV['MAILER_DEFAULT_URL'], id: @map.id)
|
|
|
|
end
|
2017-11-25 19:23:47 +00:00
|
|
|
|
2017-02-11 05:20:42 +00:00
|
|
|
def after_created_async
|
2017-11-25 19:23:47 +00:00
|
|
|
FollowService.follow(self, user, 'created')
|
2017-02-11 05:20:42 +00:00
|
|
|
# notify users following the topic creator
|
|
|
|
end
|
|
|
|
handle_asynchronously :after_created_async
|
2016-12-16 21:51:52 +00:00
|
|
|
|
|
|
|
def after_updated
|
2017-02-09 21:53:19 +00:00
|
|
|
if ATTRS_TO_WATCH.any? { |k| changed_attributes.key?(k) }
|
|
|
|
new = attributes.select { |k| ATTRS_TO_WATCH.include?(k) }
|
|
|
|
old = changed_attributes.select { |k| ATTRS_TO_WATCH.include?(k) }
|
2017-01-24 00:30:13 +00:00
|
|
|
meta = new.merge(old) # we are prioritizing the old values, keeping them
|
2017-02-09 21:53:19 +00:00
|
|
|
meta['changed'] = changed_attributes.keys.select { |k| ATTRS_TO_WATCH.include?(k) }
|
|
|
|
Events::TopicUpdated.publish!(self, updated_by, meta)
|
2017-01-24 00:30:13 +00:00
|
|
|
maps.each do |map|
|
2017-01-03 21:12:58 +00:00
|
|
|
ActionCable.server.broadcast 'map_' + map.id.to_s, type: 'topicUpdated', id: id
|
2017-01-24 00:30:13 +00:00
|
|
|
end
|
2016-12-16 21:51:52 +00:00
|
|
|
end
|
|
|
|
end
|
2017-11-25 19:23:47 +00:00
|
|
|
|
2017-02-11 05:20:42 +00:00
|
|
|
def after_updated_async
|
|
|
|
if ATTRS_TO_WATCH.any? { |k| changed_attributes.key?(k) }
|
|
|
|
FollowService.follow(self, updated_by, 'contributed')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
handle_asynchronously :after_updated_async
|
|
|
|
|
|
|
|
def before_destroyed
|
|
|
|
# hard to know how to do this yet, because the topic actually gets destroyed
|
2017-11-25 19:23:47 +00:00
|
|
|
# NotificationService.notify_followers(self, 'topic_deleted', ?)
|
2017-02-11 05:20:42 +00:00
|
|
|
end
|
2016-02-03 13:38:41 +00:00
|
|
|
end
|