module MeducationSDK class Recommender include Helpers def self.recommend(items, options = {}) if items.is_a?(User) UserRecommender.new(items, options).recommend else new(items, options).recommend end end attr_reader :items def initialize(items, options = {}) @items = [items].flatten @options = options @limit = options[:limit] || 5 end def recommend recommendations = generate_recommendations if recommendations.size < @limit log "recommendations.size (#{recommendations.size}) is < limit of #{@limit}" recommendations += MeducationSDK::MediaFile.where('rating > 2').where(safe_for_email: true).per(@limit - recommendations.size).order(:random).to_a else log "recommendations.size (#{recommendations.size}) is >= limit of #{@limit}" end recommendations rescue => e log_error("!!Recommender Error!!") log_error(e.message) log_error(e.backtrace) MeducationSDK::MediaFile.where('rating > 2').where(safe_for_email: true).per(@limit).order(:random).to_a end def generate_recommendations groupings = {} correct_order = recommender_results.map do |result| groupings[result['type']] ||= [] groupings[result['type']] << result['id'] "#{result['type']}##{result['id']}" end output = groupings.map {|(type, ids)| sdk_class_for(type).where(id: ids) }.flatten output.sort_by!{ |item| correct_order.index("#{item.class.name}##{item.id}") } output end def recommender_results @recommender_results ||= begin results = JSON.parse(recommender_json) if @options[:limit] results[0..(@options[:limit] - 1)] else results end end end def recommender_json path = "/combined" separator = "?" @items.each do |item| item_type, item_id = if item.is_a?(BoardItem) [item.item_type, item.item_id] else [item.class.name, item.id] end path << separator << "#{spi_type_for(item_type)}/#{item_id}" separator = "&" end log "Calling #{config.recommender_host}:#{config.recommender_host}#{path}" response = Net::HTTP.get_response(config.recommender_host, path, config.recommender_port) body = response.body log "Received #{body}" body end def log(*args) config.logger.info(*args) end def log_error(*args) config.logger.error(*args) end def config MeducationSDK.config end class UserRecommender include Helpers def initialize(user, options = {}) @user = user @options = options @limit = options[:limit] || 5 end def recommend results = Loquor.get("/items/recommendations_for_user?user_id=#{@user.id}", user_id: @user.id) results.map do |result| sdk_class_for(result["type"]).new(result) end end end end end