191 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
# frozen_string_literal: true
 | 
						|
 | 
						|
class ActivityPub::Parser::StatusParser
 | 
						|
  include JsonLdHelper
 | 
						|
 | 
						|
  NORMALIZED_LOCALE_NAMES = LanguagesHelper::SUPPORTED_LOCALES.keys.index_by(&:downcase).freeze
 | 
						|
 | 
						|
  # @param [Hash] json
 | 
						|
  # @param [Hash] options
 | 
						|
  # @option options [String] :followers_collection
 | 
						|
  # @option options [String] :actor_uri
 | 
						|
  # @option options [Hash]   :object
 | 
						|
  def initialize(json, **options)
 | 
						|
    @json    = json
 | 
						|
    @object  = options[:object] || json['object'] || json
 | 
						|
    @options = options
 | 
						|
  end
 | 
						|
 | 
						|
  def uri
 | 
						|
    id = @object['id']
 | 
						|
 | 
						|
    if id&.start_with?('bear:')
 | 
						|
      Addressable::URI.parse(id).query_values['u']
 | 
						|
    else
 | 
						|
      id
 | 
						|
    end
 | 
						|
  rescue Addressable::URI::InvalidURIError
 | 
						|
    id
 | 
						|
  end
 | 
						|
 | 
						|
  def url
 | 
						|
    return if @object['url'].blank?
 | 
						|
 | 
						|
    url = url_to_href(@object['url'], 'text/html')
 | 
						|
    url unless unsupported_uri_scheme?(url)
 | 
						|
  end
 | 
						|
 | 
						|
  def text
 | 
						|
    if @object['content'].present?
 | 
						|
      @object['content']
 | 
						|
    elsif content_language_map?
 | 
						|
      @object['contentMap'].values.first
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def spoiler_text
 | 
						|
    if @object['summary'].present?
 | 
						|
      @object['summary']
 | 
						|
    elsif summary_language_map?
 | 
						|
      @object['summaryMap'].values.first
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def title
 | 
						|
    if @object['name'].present?
 | 
						|
      @object['name']
 | 
						|
    elsif name_language_map?
 | 
						|
      @object['nameMap'].values.first
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def created_at
 | 
						|
    datetime = @object['published']&.to_datetime
 | 
						|
    datetime if datetime.present? && (0..9999).cover?(datetime.year)
 | 
						|
  rescue ArgumentError
 | 
						|
    nil
 | 
						|
  end
 | 
						|
 | 
						|
  def edited_at
 | 
						|
    @object['updated']&.to_datetime
 | 
						|
  rescue ArgumentError
 | 
						|
    nil
 | 
						|
  end
 | 
						|
 | 
						|
  def reply
 | 
						|
    @object['inReplyTo'].present?
 | 
						|
  end
 | 
						|
 | 
						|
  def sensitive
 | 
						|
    @object['sensitive']
 | 
						|
  end
 | 
						|
 | 
						|
  def visibility
 | 
						|
    if audience_to.any? { |to| ActivityPub::TagManager.instance.public_collection?(to) }
 | 
						|
      :public
 | 
						|
    elsif audience_cc.any? { |cc| ActivityPub::TagManager.instance.public_collection?(cc) }
 | 
						|
      :unlisted
 | 
						|
    elsif audience_to.include?(@options[:followers_collection])
 | 
						|
      :private
 | 
						|
    else
 | 
						|
      :direct
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def language
 | 
						|
    lang = raw_language_code
 | 
						|
    lang.presence && NORMALIZED_LOCALE_NAMES.fetch(lang.downcase.to_sym, lang)
 | 
						|
  end
 | 
						|
 | 
						|
  def favourites_count
 | 
						|
    @object['likes']['totalItems'] if @object.is_a?(Hash) && @object['likes'].is_a?(Hash)
 | 
						|
  end
 | 
						|
 | 
						|
  def reblogs_count
 | 
						|
    @object['shares']['totalItems'] if @object.is_a?(Hash) && @object['shares'].is_a?(Hash)
 | 
						|
  end
 | 
						|
 | 
						|
  def quote_policy
 | 
						|
    flags = 0
 | 
						|
    policy = @object.dig('interactionPolicy', 'canQuote')
 | 
						|
    return flags if policy.blank?
 | 
						|
 | 
						|
    flags |= quote_subpolicy(policy['automaticApproval'])
 | 
						|
    flags <<= 16
 | 
						|
    flags |= quote_subpolicy(policy['manualApproval'])
 | 
						|
 | 
						|
    flags
 | 
						|
  end
 | 
						|
 | 
						|
  def quote_uri
 | 
						|
    %w(quote _misskey_quote quoteUrl quoteUri).filter_map do |key|
 | 
						|
      value_or_id(as_array(@object[key]).first)
 | 
						|
    end.first
 | 
						|
  end
 | 
						|
 | 
						|
  def legacy_quote?
 | 
						|
    !@object.key?('quote')
 | 
						|
  end
 | 
						|
 | 
						|
  # The inlined quote; out of the attributes we support, only `https://w3id.org/fep/044f#quote` explicitly supports inlined objects
 | 
						|
  def quoted_object
 | 
						|
    as_array(@object['quote']).first
 | 
						|
  end
 | 
						|
 | 
						|
  def quote_approval_uri
 | 
						|
    as_array(@object['quoteAuthorization']).first
 | 
						|
  end
 | 
						|
 | 
						|
  private
 | 
						|
 | 
						|
  def quote_subpolicy(subpolicy)
 | 
						|
    flags = 0
 | 
						|
 | 
						|
    allowed_actors = as_array(subpolicy).dup
 | 
						|
    allowed_actors.uniq!
 | 
						|
 | 
						|
    flags |= Status::QUOTE_APPROVAL_POLICY_FLAGS[:public] if allowed_actors.delete('as:Public') || allowed_actors.delete('Public') || allowed_actors.delete('https://www.w3.org/ns/activitystreams#Public')
 | 
						|
    flags |= Status::QUOTE_APPROVAL_POLICY_FLAGS[:followers] if allowed_actors.delete(@options[:followers_collection])
 | 
						|
    # TODO: we don't actually store that collection URI
 | 
						|
    # flags |= Status::QUOTE_APPROVAL_POLICY_FLAGS[:followed]
 | 
						|
 | 
						|
    # Remove the special-meaning actor URI
 | 
						|
    allowed_actors.delete(@options[:actor_uri])
 | 
						|
 | 
						|
    # Any unrecognized actor is marked as unknown
 | 
						|
    flags |= Status::QUOTE_APPROVAL_POLICY_FLAGS[:unknown] unless allowed_actors.empty?
 | 
						|
 | 
						|
    flags
 | 
						|
  end
 | 
						|
 | 
						|
  def raw_language_code
 | 
						|
    if content_language_map?
 | 
						|
      @object['contentMap'].keys.first
 | 
						|
    elsif name_language_map?
 | 
						|
      @object['nameMap'].keys.first
 | 
						|
    elsif summary_language_map?
 | 
						|
      @object['summaryMap'].keys.first
 | 
						|
    end
 | 
						|
  end
 | 
						|
 | 
						|
  def audience_to
 | 
						|
    as_array(@object['to'] || @json['to']).map { |x| value_or_id(x) }
 | 
						|
  end
 | 
						|
 | 
						|
  def audience_cc
 | 
						|
    as_array(@object['cc'] || @json['cc']).map { |x| value_or_id(x) }
 | 
						|
  end
 | 
						|
 | 
						|
  def summary_language_map?
 | 
						|
    @object['summaryMap'].is_a?(Hash) && !@object['summaryMap'].empty?
 | 
						|
  end
 | 
						|
 | 
						|
  def content_language_map?
 | 
						|
    @object['contentMap'].is_a?(Hash) && !@object['contentMap'].empty?
 | 
						|
  end
 | 
						|
 | 
						|
  def name_language_map?
 | 
						|
    @object['nameMap'].is_a?(Hash) && !@object['nameMap'].empty?
 | 
						|
  end
 | 
						|
end
 |