223 lines
5.6 KiB
Ruby
223 lines
5.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 [Hash] :object
|
|
def initialize(json, **options)
|
|
@json = json
|
|
@object = options[:object] || json['object'] || json
|
|
@options = options
|
|
@account = options[:account]
|
|
@friend = options[:friend_domain]
|
|
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
|
|
url_to_href(@object['url'], 'text/html') if @object['url'].present?
|
|
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_to.include?('kmyblue:LocalPublic') && @friend
|
|
:public_unlisted
|
|
elsif audience_cc.any? { |cc| ActivityPub::TagManager.instance.public_collection?(cc) }
|
|
:unlisted
|
|
elsif audience_to.include?('kmyblue:LoginOnly') || audience_to.include?('as:LoginOnly') || audience_to.include?('LoginUser')
|
|
:login
|
|
elsif audience_to.include?(@options[:followers_collection])
|
|
:private
|
|
else
|
|
:direct
|
|
end
|
|
end
|
|
|
|
def distributable_visibility?
|
|
%i(public public_unlisted unlisted login).include?(visibility)
|
|
end
|
|
|
|
def searchability
|
|
from_audience = searchability_from_audience
|
|
return from_audience if from_audience
|
|
return nil if default_searchability_from_bio?
|
|
|
|
searchability_from_bio || (misskey_software? ? misskey_searchability : nil)
|
|
end
|
|
|
|
def limited_scope
|
|
case @object['limitedScope']
|
|
when 'Mutual'
|
|
:mutual
|
|
when 'Circle'
|
|
:circle
|
|
when 'Reply'
|
|
:reply
|
|
else
|
|
:none
|
|
end
|
|
end
|
|
|
|
def language
|
|
lang = raw_language_code || (misskey_software? ? 'ja' : nil)
|
|
lang.presence && NORMALIZED_LOCALE_NAMES.fetch(lang.downcase.to_sym, lang)
|
|
end
|
|
|
|
private
|
|
|
|
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 audience_searchable_by
|
|
return nil if @object['searchableBy'].nil?
|
|
|
|
@audience_searchable_by = as_array(@object['searchableBy']).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
|
|
|
|
def instance_info
|
|
@instance_info ||= InstanceInfo.find_by(domain: @account.domain)
|
|
end
|
|
|
|
def misskey_software?
|
|
info = instance_info
|
|
return false if info.nil?
|
|
|
|
%w(misskey calckey).include?(info.software)
|
|
end
|
|
|
|
def misskey_searchability
|
|
%i(public unlisted).include?(visibility) ? :public : :limited
|
|
end
|
|
|
|
SCAN_SEARCHABILITY_RE = /\[searchability:(public|followers|reactors|private)\]/
|
|
SCAN_SEARCHABILITY_FEDIBIRD_RE = /searchable_by_(all_users|followers_only|reacted_users_only|nobody)/
|
|
|
|
def default_searchability_from_bio?
|
|
note = @account.note
|
|
return false if note.blank?
|
|
|
|
note.include?('searchable_by_default_range')
|
|
end
|
|
|
|
def searchability_from_bio
|
|
note = @account.note
|
|
return nil if note.blank?
|
|
|
|
searchability_bio = note.scan(SCAN_SEARCHABILITY_FEDIBIRD_RE).first || note.scan(SCAN_SEARCHABILITY_RE).first
|
|
return nil unless searchability_bio
|
|
|
|
searchability = searchability_bio[0]
|
|
return nil if searchability.nil?
|
|
|
|
searchability = :public if %w(public all_users).include?(searchability)
|
|
searchability = :private if %w(followers followers_only).include?(searchability)
|
|
searchability = :direct if %w(reactors reacted_users_only).include?(searchability)
|
|
searchability = :limited if %w(private nobody).include?(searchability)
|
|
|
|
searchability
|
|
end
|
|
|
|
def searchability_from_audience
|
|
if audience_searchable_by.nil?
|
|
nil
|
|
elsif audience_searchable_by.any? { |uri| ActivityPub::TagManager.instance.public_collection?(uri) }
|
|
:public
|
|
elsif audience_searchable_by.include?('kmyblue:Limited') || audience_searchable_by.include?('as:Limited')
|
|
:limited
|
|
elsif audience_searchable_by.include?('kmyblue:LocalPublic') && @friend
|
|
:public_unlisted
|
|
elsif audience_searchable_by.include?(@account.followers_url)
|
|
:private
|
|
else
|
|
:direct
|
|
end
|
|
end
|
|
end
|