Improve ActivityPub representations (#3844)
* Improve webfinger templates and make tests more flexible * Clean up AS2 representation of actor * Refactor outbox * Create activities representation * Add representations of followers/following collections, do not redirect /users/:username route if format is empty * Remove unused translations * ActivityPub endpoint for single statuses, add ActivityPub::TagManager for better URL/URI generation * Add ActivityPub::TagManager#to * Represent all attachments as Document instead of Image/Video specifically (Because for remote ones we may not know for sure) Add mentions and hashtags representation to AP notes * Add AP-resolvable hashtag URIs * Use ActiveModelSerializers for ActivityPub * Clean up unused translations * Separate route for object and activity * Adjust cc/to matrices * Add to/cc to activities, ensure announce activity embeds target status and not the wrapper status, add "id" to all collections
This commit is contained in:
		| @@ -16,7 +16,9 @@ class AccountsController < ApplicationController | ||||
|         render xml: AtomSerializer.render(AtomSerializer.new.feed(@account, @entries.to_a)) | ||||
|       end | ||||
|  | ||||
|       format.activitystreams2 | ||||
|       format.json do | ||||
|         render json: @account, serializer: ActivityPub::ActorSerializer, adapter: ActivityPub::Adapter | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   | ||||
							
								
								
									
										28
									
								
								app/controllers/activitypub/outboxes_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								app/controllers/activitypub/outboxes_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::OutboxesController < Api::BaseController | ||||
|   before_action :set_account | ||||
|  | ||||
|   def show | ||||
|     @statuses = @account.statuses.permitted_for(@account, current_account).paginate_by_max_id(20, params[:max_id], params[:since_id]) | ||||
|     @statuses = cache_collection(@statuses, Status) | ||||
|  | ||||
|     render json: outbox_presenter, serializer: ActivityPub::CollectionSerializer, adapter: ActivityPub::Adapter | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def set_account | ||||
|     @account = Account.find_local!(params[:account_username]) | ||||
|   end | ||||
|  | ||||
|   def outbox_presenter | ||||
|     ActivityPub::CollectionPresenter.new( | ||||
|       id: account_outbox_url(@account), | ||||
|       type: :ordered, | ||||
|       current: account_outbox_url(@account), | ||||
|       size: @account.statuses_count, | ||||
|       items: @statuses | ||||
|     ) | ||||
|   end | ||||
| end | ||||
| @@ -1,27 +0,0 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::ActivityPub::ActivitiesController < Api::BaseController | ||||
|   include Authorization | ||||
|  | ||||
|   # before_action :set_follow, only: [:show_follow] | ||||
|   before_action :set_status, only: [:show_status] | ||||
|  | ||||
|   respond_to :activitystreams2 | ||||
|  | ||||
|   # Show a status in AS2 format, as either an Announce (reblog) or a Create (post) activity. | ||||
|   def show_status | ||||
|     authorize @status, :show? | ||||
|  | ||||
|     if @status.reblog? | ||||
|       render :show_status_announce | ||||
|     else | ||||
|       render :show_status_create | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def set_status | ||||
|     @status = Status.find(params[:id]) | ||||
|   end | ||||
| end | ||||
| @@ -1,19 +0,0 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::ActivityPub::NotesController < Api::BaseController | ||||
|   include Authorization | ||||
|  | ||||
|   before_action :set_status | ||||
|  | ||||
|   respond_to :activitystreams2 | ||||
|  | ||||
|   def show | ||||
|     authorize @status, :show? | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def set_status | ||||
|     @status = Status.find(params[:id]) | ||||
|   end | ||||
| end | ||||
| @@ -1,69 +0,0 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::ActivityPub::OutboxController < Api::BaseController | ||||
|   before_action :set_account | ||||
|  | ||||
|   respond_to :activitystreams2 | ||||
|  | ||||
|   def show | ||||
|     if params[:max_id] || params[:since_id] | ||||
|       show_outbox_page | ||||
|     else | ||||
|       show_base_outbox | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def show_base_outbox | ||||
|     @statuses = Status.as_outbox_timeline(@account) | ||||
|     @statuses = cache_collection(@statuses) | ||||
|  | ||||
|     set_maps(@statuses) | ||||
|  | ||||
|     set_first_last_page(@statuses) | ||||
|  | ||||
|     render :show | ||||
|   end | ||||
|  | ||||
|   def show_outbox_page | ||||
|     all_statuses = Status.as_outbox_timeline(@account) | ||||
|     @statuses = all_statuses.paginate_by_max_id(limit_param(DEFAULT_STATUSES_LIMIT), params[:max_id], params[:since_id]) | ||||
|  | ||||
|     all_statuses = cache_collection(all_statuses) | ||||
|     @statuses = cache_collection(@statuses) | ||||
|  | ||||
|     set_maps(@statuses) | ||||
|  | ||||
|     set_first_last_page(all_statuses) | ||||
|  | ||||
|     @next_page_url = api_activitypub_outbox_url(pagination_params(max_id: @statuses.last.id))    unless @statuses.empty? | ||||
|     @prev_page_url = api_activitypub_outbox_url(pagination_params(since_id: @statuses.first.id)) unless @statuses.empty? | ||||
|  | ||||
|     @paginated = @next_page_url || @prev_page_url | ||||
|     @part_of_url = api_activitypub_outbox_url | ||||
|  | ||||
|     set_pagination_headers(@next_page_url, @prev_page_url) | ||||
|  | ||||
|     render :show_page | ||||
|   end | ||||
|  | ||||
|   def cache_collection(raw) | ||||
|     super(raw, Status) | ||||
|   end | ||||
|  | ||||
|   def set_account | ||||
|     @account = Account.find(params[:id]) | ||||
|   end | ||||
|  | ||||
|   def set_first_last_page(statuses) # rubocop:disable Style/AccessorMethodName | ||||
|     return if statuses.empty? | ||||
|  | ||||
|     @first_page_url = api_activitypub_outbox_url(max_id: statuses.first.id + 1) | ||||
|     @last_page_url = api_activitypub_outbox_url(since_id: statuses.last.id - 1) | ||||
|   end | ||||
|  | ||||
|   def pagination_params(core_params) | ||||
|     params.permit(:local, :limit).merge(core_params) | ||||
|   end | ||||
| end | ||||
| @@ -5,5 +5,25 @@ class FollowerAccountsController < ApplicationController | ||||
|  | ||||
|   def index | ||||
|     @follows = Follow.where(target_account: @account).recent.page(params[:page]).per(FOLLOW_PER_PAGE).preload(:account) | ||||
|  | ||||
|     respond_to do |format| | ||||
|       format.html | ||||
|  | ||||
|       format.json do | ||||
|         render json: collection_presenter, serializer: ActivityPub::CollectionSerializer, adapter: ActivityPub::Adapter | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def collection_presenter | ||||
|     ActivityPub::CollectionPresenter.new( | ||||
|       id: account_followers_url(@account), | ||||
|       type: :ordered, | ||||
|       current: account_followers_url(@account), | ||||
|       size: @account.followers_count, | ||||
|       items: @follows.map { |f| ActivityPub::TagManager.instance.uri_for(f.account) } | ||||
|     ) | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -5,5 +5,25 @@ class FollowingAccountsController < ApplicationController | ||||
|  | ||||
|   def index | ||||
|     @follows = Follow.where(account: @account).recent.page(params[:page]).per(FOLLOW_PER_PAGE).preload(:target_account) | ||||
|  | ||||
|     respond_to do |format| | ||||
|       format.html | ||||
|  | ||||
|       format.json do | ||||
|         render json: collection_presenter, serializer: ActivityPub::CollectionSerializer, adapter: ActivityPub::Adapter | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def collection_presenter | ||||
|     ActivityPub::CollectionPresenter.new( | ||||
|       id: account_following_index_url(@account), | ||||
|       type: :ordered, | ||||
|       current: account_following_index_url(@account), | ||||
|       size: @account.following_count, | ||||
|       items: @follows.map { |f| ActivityPub::TagManager.instance.uri_for(f.target_account) } | ||||
|     ) | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -11,10 +11,22 @@ class StatusesController < ApplicationController | ||||
|   before_action :check_account_suspension | ||||
|  | ||||
|   def show | ||||
|     @ancestors   = @status.reply? ? cache_collection(@status.ancestors(current_account), Status) : [] | ||||
|     @descendants = cache_collection(@status.descendants(current_account), Status) | ||||
|     respond_to do |format| | ||||
|       format.html do | ||||
|         @ancestors   = @status.reply? ? cache_collection(@status.ancestors(current_account), Status) : [] | ||||
|         @descendants = cache_collection(@status.descendants(current_account), Status) | ||||
|  | ||||
|     render 'stream_entries/show' | ||||
|         render 'stream_entries/show' | ||||
|       end | ||||
|  | ||||
|       format.json do | ||||
|         render json: @status, serializer: ActivityPub::NoteSerializer, adapter: ActivityPub::Adapter | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def activity | ||||
|     render json: @status, serializer: ActivityPub::ActivitySerializer, adapter: ActivityPub::Adapter | ||||
|   end | ||||
|  | ||||
|   private | ||||
|   | ||||
| @@ -5,7 +5,27 @@ class TagsController < ApplicationController | ||||
|  | ||||
|   def show | ||||
|     @tag      = Tag.find_by!(name: params[:id].downcase) | ||||
|     @statuses = @tag.nil? ? [] : Status.as_tag_timeline(@tag, current_account, params[:local]).paginate_by_max_id(20, params[:max_id]) | ||||
|     @statuses = Status.as_tag_timeline(@tag, current_account, params[:local]).paginate_by_max_id(20, params[:max_id]) | ||||
|     @statuses = cache_collection(@statuses, Status) | ||||
|  | ||||
|     respond_to do |format| | ||||
|       format.html | ||||
|  | ||||
|       format.json do | ||||
|         render json: collection_presenter, serializer: ActivityPub::CollectionSerializer, adapter: ActivityPub::Adapter | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def collection_presenter | ||||
|     ActivityPub::CollectionPresenter.new( | ||||
|       id: tag_url(@tag), | ||||
|       type: :ordered, | ||||
|       current: tag_url(@tag), | ||||
|       size: @tag.statuses.count, | ||||
|       items: @statuses.map { |s| ActivityPub::TagManager.instance.uri_for(s) } | ||||
|     ) | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -1,8 +0,0 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| module Activitystreams2BuilderHelper | ||||
|   # Gets a usable name for an account, using display name or username. | ||||
|   def account_name(account) | ||||
|     account.display_name.presence || account.username | ||||
|   end | ||||
| end | ||||
							
								
								
									
										13
									
								
								app/lib/activitypub/adapter.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								app/lib/activitypub/adapter.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::Adapter < ActiveModelSerializers::Adapter::Base | ||||
|   def self.default_key_transform | ||||
|     :camel_lower | ||||
|   end | ||||
|  | ||||
|   def serializable_hash(options = nil) | ||||
|     options = serialization_options(options) | ||||
|     serialized_hash = { '@context': 'https://www.w3.org/ns/activitystreams' }.merge(ActiveModelSerializers::Adapter::Attributes.new(serializer, instance_options).serializable_hash(options)) | ||||
|     self.class.transform_key_casing!(serialized_hash, instance_options) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										69
									
								
								app/lib/activitypub/tag_manager.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								app/lib/activitypub/tag_manager.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| require 'singleton' | ||||
|  | ||||
| class ActivityPub::TagManager | ||||
|   include Singleton | ||||
|   include RoutingHelper | ||||
|  | ||||
|   COLLECTIONS = { | ||||
|     public: 'https://www.w3.org/ns/activitystreams#Public', | ||||
|   }.freeze | ||||
|  | ||||
|   def url_for(target) | ||||
|     return target.url if target.respond_to?(:local?) && !target.local? | ||||
|  | ||||
|     case target.object_type | ||||
|     when :person | ||||
|       short_account_url(target) | ||||
|     when :note, :comment, :activity | ||||
|       short_account_status_url(target.account, target) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def uri_for(target) | ||||
|     return target.uri if target.respond_to?(:local?) && !target.local? | ||||
|  | ||||
|     case target.object_type | ||||
|     when :person | ||||
|       account_url(target) | ||||
|     when :note, :comment, :activity | ||||
|       account_status_url(target.account, target) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   # Primary audience of a status | ||||
|   # Public statuses go out to primarily the public collection | ||||
|   # Unlisted and private statuses go out primarily to the followers collection | ||||
|   # Others go out only to the people they mention | ||||
|   def to(status) | ||||
|     case status.visibility | ||||
|     when 'public' | ||||
|       [COLLECTIONS[:public]] | ||||
|     when 'unlisted', 'private' | ||||
|       [account_followers_url(status.account)] | ||||
|     when 'direct' | ||||
|       status.mentions.map { |mention| uri_for(mention.account) } | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   # Secondary audience of a status | ||||
|   # Public statuses go out to followers as well | ||||
|   # Unlisted statuses go to the public as well | ||||
|   # Both of those and private statuses also go to the people mentioned in them | ||||
|   # Direct ones don't have a secondary audience | ||||
|   def cc(status) | ||||
|     cc = [] | ||||
|  | ||||
|     case status.visibility | ||||
|     when 'public' | ||||
|       cc << account_followers_url(status.account) | ||||
|     when 'unlisted' | ||||
|       cc << COLLECTIONS[:public] | ||||
|     end | ||||
|  | ||||
|     cc.concat(status.mentions.map { |mention| uri_for(mention.account) }) unless status.direct_visibility? | ||||
|  | ||||
|     cc | ||||
|   end | ||||
| end | ||||
							
								
								
									
										5
									
								
								app/presenters/activitypub/collection_presenter.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								app/presenters/activitypub/collection_presenter.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::CollectionPresenter < ActiveModelSerializers::Model | ||||
|   attributes :id, :type, :current, :size, :items | ||||
| end | ||||
							
								
								
									
										27
									
								
								app/serializers/activitypub/activity_serializer.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								app/serializers/activitypub/activity_serializer.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::ActivitySerializer < ActiveModel::Serializer | ||||
|   attributes :id, :type, :actor, :to, :cc | ||||
|  | ||||
|   has_one :proper, key: :object, serializer: ActivityPub::NoteSerializer | ||||
|  | ||||
|   def id | ||||
|     [ActivityPub::TagManager.instance.uri_for(object), '/activity'].join | ||||
|   end | ||||
|  | ||||
|   def type | ||||
|     object.reblog? ? 'Announce' : 'Create' | ||||
|   end | ||||
|  | ||||
|   def actor | ||||
|     ActivityPub::TagManager.instance.uri_for(object.account) | ||||
|   end | ||||
|  | ||||
|   def to | ||||
|     ActivityPub::TagManager.instance.to(object) | ||||
|   end | ||||
|  | ||||
|   def cc | ||||
|     ActivityPub::TagManager.instance.cc(object) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										53
									
								
								app/serializers/activitypub/actor_serializer.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								app/serializers/activitypub/actor_serializer.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::ActorSerializer < ActiveModel::Serializer | ||||
|   include RoutingHelper | ||||
|  | ||||
|   attributes :id, :type, :following, :followers, | ||||
|              :inbox, :outbox, :preferred_username, | ||||
|              :name, :summary, :icon, :image | ||||
|  | ||||
|   def id | ||||
|     account_url(object) | ||||
|   end | ||||
|  | ||||
|   def type | ||||
|     'Person' | ||||
|   end | ||||
|  | ||||
|   def following | ||||
|     account_following_index_url(object) | ||||
|   end | ||||
|  | ||||
|   def followers | ||||
|     account_followers_url(object) | ||||
|   end | ||||
|  | ||||
|   def inbox | ||||
|     nil | ||||
|   end | ||||
|  | ||||
|   def outbox | ||||
|     account_outbox_url(object) | ||||
|   end | ||||
|  | ||||
|   def preferred_username | ||||
|     object.username | ||||
|   end | ||||
|  | ||||
|   def name | ||||
|     object.display_name | ||||
|   end | ||||
|  | ||||
|   def summary | ||||
|     Formatter.instance.simplified_format(object) | ||||
|   end | ||||
|  | ||||
|   def icon | ||||
|     full_asset_url(object.avatar.url(:original)) | ||||
|   end | ||||
|  | ||||
|   def image | ||||
|     full_asset_url(object.header.url(:original)) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										26
									
								
								app/serializers/activitypub/collection_serializer.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								app/serializers/activitypub/collection_serializer.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::CollectionSerializer < ActiveModel::Serializer | ||||
|   def self.serializer_for(model, options) | ||||
|     return ActivityPub::ActivitySerializer if model.class.name == 'Status' | ||||
|     super | ||||
|   end | ||||
|  | ||||
|   attributes :id, :type, :total_items, | ||||
|              :current | ||||
|  | ||||
|   has_many :items, key: :ordered_items | ||||
|  | ||||
|   def type | ||||
|     case object.type | ||||
|     when :ordered | ||||
|       'OrderedCollection' | ||||
|     else | ||||
|       'Collection' | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def total_items | ||||
|     object.size | ||||
|   end | ||||
| end | ||||
							
								
								
									
										106
									
								
								app/serializers/activitypub/note_serializer.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								app/serializers/activitypub/note_serializer.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,106 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class ActivityPub::NoteSerializer < ActiveModel::Serializer | ||||
|   attributes :id, :type, :summary, :content, | ||||
|              :in_reply_to, :published, :url, | ||||
|              :actor, :to, :cc, :sensitive | ||||
|  | ||||
|   has_many :media_attachments, key: :attachment | ||||
|   has_many :virtual_tags, key: :tag | ||||
|  | ||||
|   def id | ||||
|     ActivityPub::TagManager.instance.uri_for(object) | ||||
|   end | ||||
|  | ||||
|   def type | ||||
|     'Note' | ||||
|   end | ||||
|  | ||||
|   def summary | ||||
|     object.spoiler_text.presence | ||||
|   end | ||||
|  | ||||
|   def content | ||||
|     Formatter.instance.format(object) | ||||
|   end | ||||
|  | ||||
|   def in_reply_to | ||||
|     ActivityPub::TagManager.instance.uri_for(object.thread) if object.reply? | ||||
|   end | ||||
|  | ||||
|   def published | ||||
|     object.created_at.iso8601 | ||||
|   end | ||||
|  | ||||
|   def url | ||||
|     ActivityPub::TagManager.instance.url_for(object) | ||||
|   end | ||||
|  | ||||
|   def actor | ||||
|     ActivityPub::TagManager.instance.uri_for(object.account) | ||||
|   end | ||||
|  | ||||
|   def to | ||||
|     ActivityPub::TagManager.instance.to(object) | ||||
|   end | ||||
|  | ||||
|   def cc | ||||
|     ActivityPub::TagManager.instance.cc(object) | ||||
|   end | ||||
|  | ||||
|   def virtual_tags | ||||
|     object.mentions + object.tags | ||||
|   end | ||||
|  | ||||
|   class MediaAttachmentSerializer < ActiveModel::Serializer | ||||
|     include RoutingHelper | ||||
|  | ||||
|     attributes :type, :media_type, :url | ||||
|  | ||||
|     def type | ||||
|       'Document' | ||||
|     end | ||||
|  | ||||
|     def media_type | ||||
|       object.file_content_type | ||||
|     end | ||||
|  | ||||
|     def url | ||||
|       object.local? ? full_asset_url(object.file.url(:original, false)) : object.remote_url | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   class MentionSerializer < ActiveModel::Serializer | ||||
|     attributes :type, :href, :name | ||||
|  | ||||
|     def type | ||||
|       'Mention' | ||||
|     end | ||||
|  | ||||
|     def href | ||||
|       ActivityPub::TagManager.instance.uri_for(object.account) | ||||
|     end | ||||
|  | ||||
|     def name | ||||
|       "@#{object.account.acct}" | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   class TagSerializer < ActiveModel::Serializer | ||||
|     include RoutingHelper | ||||
|  | ||||
|     attributes :type, :href, :name | ||||
|  | ||||
|     def type | ||||
|       'Hashtag' | ||||
|     end | ||||
|  | ||||
|     def href | ||||
|       tag_url(object) | ||||
|     end | ||||
|  | ||||
|     def name | ||||
|       "##{object.name}" | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -1,9 +0,0 @@ | ||||
| extends 'activitypub/types/person.activitystreams2.rabl' | ||||
|  | ||||
| object @account | ||||
|  | ||||
| attributes display_name: :name, username: :preferredUsername, note: :summary | ||||
|  | ||||
| node(:icon)   { |account| full_asset_url(account.avatar.url(:original)) } | ||||
| node(:image)  { |account| full_asset_url(account.header.url(:original)) } | ||||
| node(:outbox) { |account| api_activitypub_outbox_url(account.id) } | ||||
| @@ -1 +0,0 @@ | ||||
| node(:'@context') { 'https://www.w3.org/ns/activitystreams' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/base.activitystreams2.rabl' | ||||
|  | ||||
| node(:id) { request.original_url } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/intransient.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'Announce' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/intransient.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'Collection' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/intransient.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'Create' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/intransient.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'Note' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/types/collection.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'OrderedCollection' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/types/ordered_collection.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'OrderedCollectionPage' } | ||||
| @@ -1,3 +0,0 @@ | ||||
| extends 'activitypub/intransient.activitystreams2.rabl' | ||||
|  | ||||
| node(:type) { 'Person' } | ||||
| @@ -1,4 +0,0 @@ | ||||
| object @status | ||||
|  | ||||
| node(:actor)     { |status| TagManager.instance.url_for(status.account) } | ||||
| node(:published) { |status| status.created_at.to_time.xmlschema } | ||||
| @@ -1,8 +0,0 @@ | ||||
| extends 'activitypub/types/announce.activitystreams2.rabl' | ||||
| extends 'api/activitypub/activities/_show_status.activitystreams2.rabl' | ||||
|  | ||||
| object @status | ||||
|  | ||||
| node(:name)   { |status| t('activitypub.activity.announce.name', account_name: account_name(status.account)) } | ||||
| node(:url)    { |status| TagManager.instance.url_for(status) } | ||||
| node(:object) { |status| api_activitypub_status_url(status.reblog_of_id) } | ||||
| @@ -1,8 +0,0 @@ | ||||
| extends 'activitypub/types/create.activitystreams2.rabl' | ||||
| extends 'api/activitypub/activities/_show_status.activitystreams2.rabl' | ||||
|  | ||||
| object @status | ||||
|  | ||||
| node(:name)   { |status| t('activitypub.activity.create.name', account_name: account_name(status.account)) } | ||||
| node(:url)    { |status| TagManager.instance.url_for(status) } | ||||
| node(:object) { |status| api_activitypub_note_url(status) } | ||||
| @@ -1,11 +0,0 @@ | ||||
| extends 'activitypub/types/note.activitystreams2.rabl' | ||||
|  | ||||
| object @status | ||||
|  | ||||
| attributes :content | ||||
|  | ||||
| node(:name)         { |status| status.content } | ||||
| node(:url)          { |status| TagManager.instance.url_for(status) } | ||||
| node(:attributedTo) { |status| TagManager.instance.url_for(status.account) } | ||||
| node(:inReplyTo)    { |status| api_activitypub_note_url(status.thread) } if @status.thread | ||||
| node(:published)    { |status| status.created_at.to_time.xmlschema } | ||||
| @@ -1,12 +0,0 @@ | ||||
| extends 'activitypub/types/ordered_collection.activitystreams2.rabl' | ||||
|  | ||||
| object @account | ||||
|  | ||||
| node(:totalItems) { @statuses.count } | ||||
| node(:current)    { @first_page_url } if @first_page_url | ||||
| node(:first)      { @first_page_url } if @first_page_url | ||||
| node(:last)       { @last_page_url } if @last_page_url | ||||
|  | ||||
| node(:name)       { |account| t('activitypub.outbox.name', account_name: account_name(account)) } | ||||
| node(:summary)    { |account| t('activitypub.outbox.summary', account_name: account_name(account)) } | ||||
| node(:updated)    { |account| (@statuses.empty? ? account.created_at.to_time : @statuses.first.updated_at.to_time).xmlschema } | ||||
| @@ -1,16 +0,0 @@ | ||||
| extends 'activitypub/types/ordered_collection_page.activitystreams2.rabl' | ||||
|  | ||||
| object @account | ||||
|  | ||||
| node(:items) do | ||||
|   @statuses.map { |status| api_activitypub_status_url(status) } | ||||
| end | ||||
|  | ||||
| node(:next)       { @next_page_url } if @next_page_url | ||||
| node(:prev)       { @prev_page_url } if @prev_page_url | ||||
| node(:current)    { @first_page_url } if @first_page_url | ||||
| node(:first)      { @first_page_url } if @first_page_url | ||||
| node(:last)       { @last_page_url } if @last_page_url | ||||
| node(:partOf)     { @part_of_url } if @part_of_url | ||||
|  | ||||
| node(:updated)    { |account| (@statuses.empty? ? account.created_at.to_time : @statuses.first.updated_at.to_time).xmlschema } | ||||
| @@ -3,14 +3,14 @@ object @account | ||||
| node(:subject) { @canonical_account_uri } | ||||
|  | ||||
| node(:aliases) do | ||||
|   [TagManager.instance.url_for(@account), TagManager.instance.uri_for(@account)] | ||||
|   [short_account_url(@account), account_url(@account)] | ||||
| end | ||||
|  | ||||
| node(:links) do | ||||
|   [ | ||||
|     { rel: 'http://webfinger.net/rel/profile-page', type: 'text/html', href: TagManager.instance.url_for(@account) }, | ||||
|     { rel: 'http://webfinger.net/rel/profile-page', type: 'text/html', href: account_url(@account) }, | ||||
|     { rel: 'http://schemas.google.com/g/2010#updates-from', type: 'application/atom+xml', href: account_url(@account, format: 'atom') }, | ||||
|     { rel: 'self', type: 'application/activity+json', href: TagManager.instance.url_for(@account) }, | ||||
|     { rel: 'self', type: 'application/activity+json', href: account_url(@account) }, | ||||
|     { rel: 'salmon', href: api_salmon_url(@account.id) }, | ||||
|     { rel: 'magic-public-key', href: "data:application/magic-public-key,#{@magic_key}" }, | ||||
|     { rel: 'http://ostatus.org/schema/1.0/subscribe', template: "#{authorize_follow_url}?acct={uri}" }, | ||||
|   | ||||
| @@ -1,10 +1,11 @@ | ||||
| Nokogiri::XML::Builder.new do |xml| | ||||
|   xml.XRD(xmlns: 'http://docs.oasis-open.org/ns/xri/xrd-1.0') do | ||||
|     xml.Subject @canonical_account_uri | ||||
|     xml.Alias TagManager.instance.url_for(@account) | ||||
|     xml.Alias TagManager.instance.uri_for(@account) | ||||
|     xml.Alias short_account_url(@account) | ||||
|     xml.Alias account_url(@account) | ||||
|     xml.Link(rel: 'http://webfinger.net/rel/profile-page', type: 'text/html', href: TagManager.instance.url_for(@account)) | ||||
|     xml.Link(rel: 'http://schemas.google.com/g/2010#updates-from', type: 'application/atom+xml', href: account_url(@account, format: 'atom')) | ||||
|     xml.Link(rel: 'self', type: 'application/activity+json', href: account_url(@account)) | ||||
|     xml.Link(rel: 'salmon', href: api_salmon_url(@account.id)) | ||||
|     xml.Link(rel: 'magic-public-key', href: "data:application/magic-public-key,#{@magic_key}") | ||||
|     xml.Link(rel: 'http://ostatus.org/schema/1.0/subscribe', template: "#{authorize_follow_url}?acct={uri}") | ||||
|   | ||||
| @@ -14,4 +14,6 @@ ActiveSupport::Inflector.inflections(:en) do |inflect| | ||||
|   inflect.acronym 'StatsD' | ||||
|   inflect.acronym 'OEmbed' | ||||
|   inflect.acronym 'ActivityPub' | ||||
|   inflect.acronym 'PubSubHubbub' | ||||
|   inflect.acronym 'ActivityStreams' | ||||
| end | ||||
|   | ||||
| @@ -1,5 +1,4 @@ | ||||
| # Be sure to restart your server when you modify this file. | ||||
|  | ||||
| Mime::Type.register "application/json",           :json, %w( text/x-json application/jsonrequest application/jrd+json ) | ||||
| Mime::Type.register "text/xml",                   :xml,  %w( application/xml application/atom+xml application/xrd+xml ) | ||||
| Mime::Type.register "application/activity+json",  :activitystreams2 | ||||
| Mime::Type.register 'application/json', :json, %w(text/x-json application/jsonrequest application/jrd+json application/activity+json) | ||||
| Mime::Type.register 'text/xml',         :xml,  %w(application/xml application/atom+xml application/xrd+xml) | ||||
|   | ||||
| @@ -30,15 +30,6 @@ ca: | ||||
|     remote_follow: Seguir | ||||
|     reserved_username: El nom d'usuari està reservat | ||||
|     unfollow: Deixar de seguir | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} shared an activity." | ||||
|       create: | ||||
|         name: "%{account_name} created a note." | ||||
|     outbox: | ||||
|       name: "%{account_name}'s Outbox" | ||||
|       summary: A collection of activities from user %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Estàs segur? | ||||
|   | ||||
| @@ -44,15 +44,6 @@ en: | ||||
|     remote_follow: Remote follow | ||||
|     reserved_username: The username is reserved | ||||
|     unfollow: Unfollow | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} shared an activity." | ||||
|       create: | ||||
|         name: "%{account_name} created a note." | ||||
|     outbox: | ||||
|       name: "%{account_name}'s Outbox" | ||||
|       summary: A collection of activities from user %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Are you sure? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ fa: | ||||
|     posts: نوشته | ||||
|     remote_follow: پیگیری غیرمستقیم | ||||
|     unfollow: پایان پیگیری | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} فعالیتی آغاز کرد." | ||||
|       create: | ||||
|         name: "%{account_name} یادداشتی نوشت." | ||||
|     outbox: | ||||
|       name: صندوق خروجی %{account_name} | ||||
|       summary: مجموعهای از فعالیتهای کاربر %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: آیا مطمئن هستید؟ | ||||
|   | ||||
| @@ -30,15 +30,6 @@ fr: | ||||
|     remote_follow: Suivre à distance | ||||
|     reserved_username: Ce nom d’utilisateur⋅ice est réservé | ||||
|     unfollow: Ne plus suivre | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} a partagé une activité." | ||||
|       create: | ||||
|         name: "%{account_name} a créé une note." | ||||
|     outbox: | ||||
|       name: Boîte d’envoi de %{account_name} | ||||
|       summary: Liste d’activités de %{account_name} | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Êtes-vous certain⋅e ? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ he: | ||||
|     posts: הודעות | ||||
|     remote_follow: מעקב מרחוק | ||||
|     unfollow: הפסקת מעקב | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: הודעה שותפה על ידי %{account_name}. | ||||
|       create: | ||||
|         name: הודעה חדשה מאת %{account_name}. | ||||
|     outbox: | ||||
|       name: תיבת הדוא"ל היוצא של %{account_name} | ||||
|       summary: אוסף הפעילויות של %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: בטוח? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ id: | ||||
|     posts: Postingan | ||||
|     remote_follow: Mengikuti | ||||
|     unfollow: Berhenti mengikuti | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} membagikan aktivitas." | ||||
|       create: | ||||
|         name: "%{account_name} membuat catatan." | ||||
|     outbox: | ||||
|       name: "%{account_name} Outbox" | ||||
|       summary: Koleksi aktivitas dari pengguna %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Anda yakin? | ||||
|   | ||||
| @@ -30,15 +30,6 @@ ja: | ||||
|     remote_follow: リモートフォロー | ||||
|     reserved_username: このユーザー名は予約されています。 | ||||
|     unfollow: フォロー解除 | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} さんがアクティビティをシェアしました" | ||||
|       create: | ||||
|         name: "%{account_name} さんがノートを作成しました" | ||||
|     outbox: | ||||
|       name: "%{account_name} さんの送信トレイ" | ||||
|       summary: "%{account_name} さんからのアクティビティコレクション" | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: 本当に実行しますか? | ||||
|   | ||||
| @@ -30,15 +30,6 @@ ko: | ||||
|     remote_follow: 리모트 팔로우 | ||||
|     reserved_username: 이 아이디는 예약되어 있습니다. | ||||
|     unfollow: 팔로우 해제 | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} 님이 액티비티를 공유했습니다" | ||||
|       create: | ||||
|         name: "%{account_name} 님이 노트를 작성했습니다" | ||||
|     outbox: | ||||
|       name: "%{account_name} 님의 송신함" | ||||
|       summary: "%{account_name} 님의 액티비티 모음" | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: 정말로 실행하시겠습니까? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ | ||||
|     posts: Poster | ||||
|     remote_follow: Følg fra andre instanser | ||||
|     unfollow: Avfølg | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} delte en aktivitet." | ||||
|       create: | ||||
|         name: "%{account_name} laget en aktivitet." | ||||
|     outbox: | ||||
|       name: "%{account_name} sin utboks" | ||||
|       summary: En samling aktiviteter fra brukeren %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Er du sikker? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ oc: | ||||
|     posts: Estatuts | ||||
|     remote_follow: Sègre a distància | ||||
|     unfollow: Quitar de sègre | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} a partejat una activitat." | ||||
|       create: | ||||
|         name: "%{account_name} a creat una nòta." | ||||
|     outbox: | ||||
|       name: "%{account_name}'s Outbox" | ||||
|       summary: A collection of activities from user %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Sètz segur ? | ||||
|   | ||||
| @@ -44,15 +44,6 @@ pl: | ||||
|     remote_follow: Zdalne śledzenie | ||||
|     reserved_username: Ta nazwa użytkownika jest zarezerwowana. | ||||
|     unfollow: Przestań śledzić | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} udostępnił(a) aktywność." | ||||
|       create: | ||||
|         name: "%{account_name} utworzył(a) wpis." | ||||
|     outbox: | ||||
|       name: Skrzynka %{account_name} | ||||
|       summary: Zbiór aktywności użytkownika %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Jesteś tego pewien? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ pt-BR: | ||||
|     posts: Posts | ||||
|     remote_follow: Acesso remoto | ||||
|     unfollow: Unfollow | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} compartilhou uma atividade." | ||||
|       create: | ||||
|         name: "%{account_name} criou uma nota." | ||||
|     outbox: | ||||
|       name: "%{account_name}'s Outbox" | ||||
|       summary: Uma coleção de atividades do usuário %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Você tem certeza? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ pt: | ||||
|     posts: Posts | ||||
|     remote_follow: Seguir remotamente | ||||
|     unfollow: Deixar de seguir | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} anunciou uma atividade." | ||||
|       create: | ||||
|         name: "%{account_name} criou uma nota." | ||||
|     outbox: | ||||
|       name: "%{account_name}'s Outbox" | ||||
|       summary: Uma coleção de atividades do usuário %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Tens a certeza? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ th: | ||||
|     posts: โพสต์ | ||||
|     remote_follow: Remote follow | ||||
|     unfollow: เลิกติดตาม | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} แชร์กิจกรรม." | ||||
|       create: | ||||
|         name: "%{account_name} สร้างโน๊ต." | ||||
|     outbox: | ||||
|       name: "%{account_name}'s Outbox" | ||||
|       summary: รวมกิจกรรมของผู้ใช้ %{account_name}. | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: แน่ใจนะ? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ tr: | ||||
|     posts: Gönderiler | ||||
|     remote_follow: Uzaktan takip et | ||||
|     unfollow: Takibi bırak | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} bir aktivite paylaştı." | ||||
|       create: | ||||
|         name: "%{account_name} bir not oluşturdu." | ||||
|     outbox: | ||||
|       name: "%{account_name}'in Gönderdikleri" | ||||
|       summary: "%{account_name}'den gelen aktiviteler." | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: Emin misiniz? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ zh-CN: | ||||
|     posts: 嘟文 | ||||
|     remote_follow: 跨站关注 | ||||
|     unfollow: 取消关注 | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} 分享了一个活动。" | ||||
|       create: | ||||
|         name: "%{account_name} 创建了一个记事。" | ||||
|     outbox: | ||||
|       name: "%{account_name} 的集合" | ||||
|       summary: "%{account_name} 的活动集合" | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: 你确定吗? | ||||
|   | ||||
| @@ -29,15 +29,6 @@ zh-HK: | ||||
|     posts: 文章 | ||||
|     remote_follow: 跨站關注 | ||||
|     unfollow: 取消關注 | ||||
|   activitypub: | ||||
|     activity: | ||||
|       announce: | ||||
|         name: "%{account_name} 分享了一項活動。" | ||||
|       create: | ||||
|         name: "%{account_name} 新增了一篇筆記。" | ||||
|     outbox: | ||||
|       name: "%{account_name} 的活動" | ||||
|       summary: "%{account_name} 分享的活動列表。" | ||||
|   admin: | ||||
|     accounts: | ||||
|       are_you_sure: 你確定嗎? | ||||
|   | ||||
| @@ -26,7 +26,7 @@ Rails.application.routes.draw do | ||||
|     confirmations:      'auth/confirmations', | ||||
|   } | ||||
|  | ||||
|   get '/users/:username', to: redirect('/@%{username}'), constraints: { format: :html } | ||||
|   get '/users/:username', to: redirect('/@%{username}'), constraints: lambda { |req| req.format.nil? } | ||||
|  | ||||
|   resources :accounts, path: 'users', only: [:show], param: :username do | ||||
|     resources :stream_entries, path: 'updates', only: [:show] do | ||||
| @@ -38,10 +38,17 @@ Rails.application.routes.draw do | ||||
|     get :remote_follow,  to: 'remote_follow#new' | ||||
|     post :remote_follow, to: 'remote_follow#create' | ||||
|  | ||||
|     resources :statuses, only: [:show] do | ||||
|       member do | ||||
|         get :activity | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     resources :followers, only: [:index], controller: :follower_accounts | ||||
|     resources :following, only: [:index], controller: :following_accounts | ||||
|     resource :follow, only: [:create], controller: :account_follow | ||||
|     resource :unfollow, only: [:create], controller: :account_unfollow | ||||
|     resource :outbox, only: [:show], module: :activitypub | ||||
|   end | ||||
|  | ||||
|   get '/@:username', to: 'accounts#show', as: :short_account | ||||
| @@ -119,13 +126,6 @@ Rails.application.routes.draw do | ||||
|     # OEmbed | ||||
|     get '/oembed', to: 'oembed#show', as: :oembed | ||||
|  | ||||
|     # ActivityPub | ||||
|     namespace :activitypub do | ||||
|       get '/users/:id/outbox', to: 'outbox#show', as: :outbox | ||||
|       get '/statuses/:id', to: 'activities#show_status', as: :status | ||||
|       resources :notes, only: [:show] | ||||
|     end | ||||
|  | ||||
|     # JSON / REST API | ||||
|     namespace :v1 do | ||||
|       resources :statuses, only: [:create, :show, :destroy] do | ||||
|   | ||||
| @@ -38,7 +38,7 @@ RSpec.describe AccountsController, type: :controller do | ||||
|  | ||||
|     context 'activitystreams2' do | ||||
|       before do | ||||
|         get :show, params: { username: alice.username }, format: 'activitystreams2' | ||||
|         get :show, params: { username: alice.username }, format: 'json' | ||||
|       end | ||||
|  | ||||
|       it 'assigns @account' do | ||||
|   | ||||
| @@ -1,69 +0,0 @@ | ||||
| require 'rails_helper' | ||||
|  | ||||
| RSpec.describe Api::ActivityPub::ActivitiesController, type: :controller do | ||||
|   render_views | ||||
|  | ||||
|   let(:user)  { Fabricate(:user, account: Fabricate(:account, username: 'alice')) } | ||||
|  | ||||
|   describe 'GET #show' do | ||||
|     describe 'normal status' do | ||||
|       public_status = nil | ||||
|  | ||||
|       before do | ||||
|         public_status = Fabricate(:status, account: user.account, text: 'Hello world', visibility: :public) | ||||
|  | ||||
|         @request.env['HTTP_ACCEPT'] = 'application/activity+json' | ||||
|         get :show_status, params: { id: public_status.id } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('type' => 'Create') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('type' => 'Create') | ||||
|         expect(json_data).to include('object' => api_activitypub_note_url(public_status)) | ||||
|         expect(json_data).to include('url' => TagManager.instance.url_for(public_status)) | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     describe 'reblog' do | ||||
|       original = nil | ||||
|       reblog = nil | ||||
|  | ||||
|       before do | ||||
|         original = Fabricate(:status, account: user.account, text: 'Hello world', visibility: :public) | ||||
|         reblog = Fabricate(:status, account: user.account, reblog_of_id: original.id, visibility: :public) | ||||
|  | ||||
|         @request.env['HTTP_ACCEPT'] = 'application/activity+json' | ||||
|         get :show_status, params: { id: reblog.id } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('type' => 'Announce') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('type' => 'Announce') | ||||
|         expect(json_data).to include('object' => api_activitypub_status_url(original)) | ||||
|         expect(json_data).to include('url' => TagManager.instance.url_for(reblog)) | ||||
|       end | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -1,73 +0,0 @@ | ||||
| require 'rails_helper' | ||||
|  | ||||
| RSpec.describe Api::ActivityPub::NotesController, type: :controller do | ||||
|   render_views | ||||
|  | ||||
|   let(:user_alice)  { Fabricate(:user, account: Fabricate(:account, username: 'alice')) } | ||||
|   let(:user_bob)  { Fabricate(:user, account: Fabricate(:account, username: 'bob')) } | ||||
|  | ||||
|   describe 'GET #show' do | ||||
|     describe 'normal status' do | ||||
|       public_status = nil | ||||
|  | ||||
|       before do | ||||
|         public_status = Fabricate(:status, account: user_alice.account, text: 'Hello world', visibility: :public) | ||||
|  | ||||
|         @request.env['HTTP_ACCEPT'] = 'application/activity+json' | ||||
|         get :show, params: { id: public_status.id } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('type' => 'Note') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('name' => 'Hello world') | ||||
|         expect(json_data).to include('content' => 'Hello world') | ||||
|         expect(json_data).to include('published') | ||||
|         expect(json_data).to include('url' => TagManager.instance.url_for(public_status)) | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     describe 'reply' do | ||||
|       original = nil | ||||
|       reply = nil | ||||
|  | ||||
|       before do | ||||
|         original = Fabricate(:status, account: user_alice.account, text: 'Hello world', visibility: :public) | ||||
|         reply = Fabricate(:status, account: user_bob.account, text: 'Hello world', in_reply_to_id: original.id, visibility: :public) | ||||
|  | ||||
|         @request.env['HTTP_ACCEPT'] = 'application/activity+json' | ||||
|         get :show, params: { id: reply.id } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('type' => 'Note') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('name' => 'Hello world') | ||||
|         expect(json_data).to include('content' => 'Hello world') | ||||
|         expect(json_data).to include('published') | ||||
|         expect(json_data).to include('url' => TagManager.instance.url_for(reply)) | ||||
|         expect(json_data).to include('inReplyTo' => api_activitypub_note_url(original)) | ||||
|       end | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -1,156 +0,0 @@ | ||||
| require 'rails_helper' | ||||
|  | ||||
| RSpec.describe Api::ActivityPub::OutboxController, type: :controller do | ||||
|   render_views | ||||
|  | ||||
|   let(:user)  { Fabricate(:user, account: Fabricate(:account, username: 'alice')) } | ||||
|  | ||||
|   describe 'GET #show' do | ||||
|     before do | ||||
|       @request.headers['ACCEPT'] = 'application/activity+json' | ||||
|     end | ||||
|  | ||||
|     describe 'collection with small number of statuses' do | ||||
|       public_status = nil | ||||
|  | ||||
|       before do | ||||
|         public_status = Fabricate(:status, account: user.account, text: 'Hello world', visibility: :public) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :private) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :unlisted) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :direct) | ||||
|  | ||||
|         get :show, params: { id: user.account.id } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns AS2 JSON body' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('type' => 'OrderedCollection') | ||||
|         expect(json_data).to include('totalItems' => 1) | ||||
|         expect(json_data).to include('current') | ||||
|         expect(json_data).to include('first') | ||||
|         expect(json_data).to include('last') | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     describe 'collection with large number of statuses' do | ||||
|       before do | ||||
|         30.times do | ||||
|           Fabricate(:status, account: user.account, text: 'Hello world', visibility: :public) | ||||
|         end | ||||
|  | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :private) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :unlisted) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :direct) | ||||
|  | ||||
|         get :show, params: { id: user.account.id } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns AS2 JSON body' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('type' => 'OrderedCollection') | ||||
|         expect(json_data).to include('totalItems' => 30) | ||||
|         expect(json_data).to include('current') | ||||
|         expect(json_data).to include('first') | ||||
|         expect(json_data).to include('last') | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     describe 'page with small number of statuses' do | ||||
|       statuses = [] | ||||
|  | ||||
|       before do | ||||
|         5.times do | ||||
|           statuses << Fabricate(:status, account: user.account, text: 'Hello world', visibility: :public) | ||||
|         end | ||||
|  | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :private) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :unlisted) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :direct) | ||||
|  | ||||
|         get :show, params: { id: user.account.id, max_id: statuses.last.id + 1 } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns AS2 JSON body' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('type' => 'OrderedCollectionPage') | ||||
|         expect(json_data).to include('partOf') | ||||
|         expect(json_data).to include('items') | ||||
|         expect(json_data['items'].length).to eq(5) | ||||
|         expect(json_data).to include('prev') | ||||
|         expect(json_data).to include('next') | ||||
|         expect(json_data).to include('current') | ||||
|         expect(json_data).to include('first') | ||||
|         expect(json_data).to include('last') | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     describe 'page with large number of statuses' do | ||||
|       statuses = [] | ||||
|  | ||||
|       before do | ||||
|         30.times do | ||||
|           statuses << Fabricate(:status, account: user.account, text: 'Hello world', visibility: :public) | ||||
|         end | ||||
|  | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :private) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :unlisted) | ||||
|         Fabricate(:status, account: user.account, text: 'Hello world', visibility: :direct) | ||||
|  | ||||
|         get :show, params: { id: user.account.id, max_id: statuses.last.id + 1 } | ||||
|       end | ||||
|  | ||||
|       it 'returns http success' do | ||||
|         expect(response).to have_http_status(:success) | ||||
|       end | ||||
|  | ||||
|       it 'sets Content-Type header to AS2' do | ||||
|         expect(response.header['Content-Type']).to include 'application/activity+json' | ||||
|       end | ||||
|  | ||||
|       it 'returns AS2 JSON body' do | ||||
|         json_data = JSON.parse(response.body) | ||||
|         expect(json_data).to include('@context' => 'https://www.w3.org/ns/activitystreams') | ||||
|         expect(json_data).to include('id' => @request.url) | ||||
|         expect(json_data).to include('type' => 'OrderedCollectionPage') | ||||
|         expect(json_data).to include('partOf') | ||||
|         expect(json_data).to include('items') | ||||
|         expect(json_data['items'].length).to eq(20) | ||||
|         expect(json_data).to include('prev') | ||||
|         expect(json_data).to include('next') | ||||
|         expect(json_data).to include('current') | ||||
|         expect(json_data).to include('first') | ||||
|         expect(json_data).to include('last') | ||||
|       end | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -9,7 +9,7 @@ describe WellKnown::WebfingerController, type: :controller do | ||||
|     end | ||||
|  | ||||
|     before do | ||||
|       alice.private_key = <<PEM | ||||
|       alice.private_key = <<-PEM | ||||
| -----BEGIN RSA PRIVATE KEY----- | ||||
| MIICXQIBAAKBgQDHgPoPJlrfMZrVcuF39UbVssa8r4ObLP3dYl9Y17Mgp5K4mSYD | ||||
| R/Y2ag58tSi6ar2zM3Ze3QYsNfTq0NqN1g89eAu0MbSjWqpOsgntRPJiFuj3hai2 | ||||
| @@ -27,7 +27,7 @@ FTX8IvYBNTbpEttc1VCf/0ccnNpfb0CrFNSPWxRj7t7D | ||||
| -----END RSA PRIVATE KEY----- | ||||
| PEM | ||||
|  | ||||
|       alice.public_key = <<PEM | ||||
|       alice.public_key = <<-PEM | ||||
| -----BEGIN PUBLIC KEY----- | ||||
| MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDHgPoPJlrfMZrVcuF39UbVssa8 | ||||
| r4ObLP3dYl9Y17Mgp5K4mSYDR/Y2ag58tSi6ar2zM3Ze3QYsNfTq0NqN1g89eAu0 | ||||
| @@ -48,29 +48,23 @@ PEM | ||||
|     it 'returns JSON when account can be found' do | ||||
|       get :show, params: { resource: alice.to_webfinger_s }, format: :json | ||||
|  | ||||
|       json = body_as_json | ||||
|  | ||||
|       expect(response).to have_http_status(:success) | ||||
|       expect(response.content_type).to eq 'application/jrd+json' | ||||
|       expect(response.body).to eq "{\"subject\":\"acct:alice@cb6e6126.ngrok.io\",\"aliases\":[\"https://cb6e6126.ngrok.io/@alice\",\"https://cb6e6126.ngrok.io/users/alice\"],\"links\":[{\"rel\":\"http://webfinger.net/rel/profile-page\",\"type\":\"text/html\",\"href\":\"https://cb6e6126.ngrok.io/@alice\"},{\"rel\":\"http://schemas.google.com/g/2010#updates-from\",\"type\":\"application/atom+xml\",\"href\":\"https://cb6e6126.ngrok.io/users/alice.atom\"},{\"rel\":\"self\",\"type\":\"application/activity+json\",\"href\":\"https://cb6e6126.ngrok.io/@alice\"},{\"rel\":\"salmon\",\"href\":\"#{api_salmon_url(alice.id)}\"},{\"rel\":\"magic-public-key\",\"href\":\"data:application/magic-public-key,RSA.x4D6DyZa3zGa1XLhd_VG1bLGvK-Dmyz93WJfWNezIKeSuJkmA0f2NmoOfLUoumq9szN2Xt0GLDX06tDajdYPPXgLtDG0o1qqTrIJ7UTyYhbo94Wotl9iJvEwa5IjP1Mn00YJ_KvFrzKCm15PC7up6r-NtHsqoYS8X1KAqcbnptU=.AQAB\"},{\"rel\":\"http://ostatus.org/schema/1.0/subscribe\",\"template\":\"https://cb6e6126.ngrok.io/authorize_follow?acct={uri}\"}]}" | ||||
|       expect(json[:subject]).to eq 'acct:alice@cb6e6126.ngrok.io' | ||||
|       expect(json[:aliases]).to include('https://cb6e6126.ngrok.io/@alice', 'https://cb6e6126.ngrok.io/users/alice') | ||||
|     end | ||||
|  | ||||
|     it 'returns JSON when account can be found' do | ||||
|       get :show, params: { resource: alice.to_webfinger_s }, format: :xml | ||||
|  | ||||
|       xml = Nokogiri::XML(response.body) | ||||
|  | ||||
|       expect(response).to have_http_status(:success) | ||||
|       expect(response.content_type).to eq 'application/xrd+xml' | ||||
|       expect(response.body).to eq <<"XML" | ||||
| <?xml version="1.0"?> | ||||
| <XRD xmlns="http://docs.oasis-open.org/ns/xri/xrd-1.0"> | ||||
|   <Subject>acct:alice@cb6e6126.ngrok.io</Subject> | ||||
|   <Alias>https://cb6e6126.ngrok.io/@alice</Alias> | ||||
|   <Alias>https://cb6e6126.ngrok.io/users/alice</Alias> | ||||
|   <Link rel="http://webfinger.net/rel/profile-page" type="text/html" href="https://cb6e6126.ngrok.io/@alice"/> | ||||
|   <Link rel="http://schemas.google.com/g/2010#updates-from" type="application/atom+xml" href="https://cb6e6126.ngrok.io/users/alice.atom"/> | ||||
|   <Link rel="salmon" href="#{api_salmon_url(alice.id)}"/> | ||||
|   <Link rel="magic-public-key" href="data:application/magic-public-key,RSA.x4D6DyZa3zGa1XLhd_VG1bLGvK-Dmyz93WJfWNezIKeSuJkmA0f2NmoOfLUoumq9szN2Xt0GLDX06tDajdYPPXgLtDG0o1qqTrIJ7UTyYhbo94Wotl9iJvEwa5IjP1Mn00YJ_KvFrzKCm15PC7up6r-NtHsqoYS8X1KAqcbnptU=.AQAB"/> | ||||
|   <Link rel="http://ostatus.org/schema/1.0/subscribe" template="https://cb6e6126.ngrok.io/authorize_follow?acct={uri}"/> | ||||
| </XRD> | ||||
| XML | ||||
|       expect(xml.at_xpath('//xmlns:Subject').content).to eq 'acct:alice@cb6e6126.ngrok.io' | ||||
|       expect(xml.xpath('//xmlns:Alias').map(&:content)).to include('https://cb6e6126.ngrok.io/@alice', 'https://cb6e6126.ngrok.io/users/alice') | ||||
|     end | ||||
|  | ||||
|     it 'returns http not found when account cannot be found' do | ||||
| @@ -80,19 +74,22 @@ XML | ||||
|     end | ||||
|  | ||||
|     it 'returns JSON when account can be found with alternate domains' do | ||||
|       Rails.configuration.x.alternate_domains = ["foo.org"] | ||||
|       username, domain = alice.to_webfinger_s.split("@") | ||||
|       Rails.configuration.x.alternate_domains = ['foo.org'] | ||||
|       username, = alice.to_webfinger_s.split('@') | ||||
|  | ||||
|       get :show, params: { resource: "#{username}@foo.org" }, format: :json | ||||
|  | ||||
|       json = body_as_json | ||||
|  | ||||
|       expect(response).to have_http_status(:success) | ||||
|       expect(response.content_type).to eq 'application/jrd+json' | ||||
|       expect(response.body).to eq "{\"subject\":\"acct:alice@cb6e6126.ngrok.io\",\"aliases\":[\"https://cb6e6126.ngrok.io/@alice\",\"https://cb6e6126.ngrok.io/users/alice\"],\"links\":[{\"rel\":\"http://webfinger.net/rel/profile-page\",\"type\":\"text/html\",\"href\":\"https://cb6e6126.ngrok.io/@alice\"},{\"rel\":\"http://schemas.google.com/g/2010#updates-from\",\"type\":\"application/atom+xml\",\"href\":\"https://cb6e6126.ngrok.io/users/alice.atom\"},{\"rel\":\"self\",\"type\":\"application/activity+json\",\"href\":\"https://cb6e6126.ngrok.io/@alice\"},{\"rel\":\"salmon\",\"href\":\"#{api_salmon_url(alice.id)}\"},{\"rel\":\"magic-public-key\",\"href\":\"data:application/magic-public-key,RSA.x4D6DyZa3zGa1XLhd_VG1bLGvK-Dmyz93WJfWNezIKeSuJkmA0f2NmoOfLUoumq9szN2Xt0GLDX06tDajdYPPXgLtDG0o1qqTrIJ7UTyYhbo94Wotl9iJvEwa5IjP1Mn00YJ_KvFrzKCm15PC7up6r-NtHsqoYS8X1KAqcbnptU=.AQAB\"},{\"rel\":\"http://ostatus.org/schema/1.0/subscribe\",\"template\":\"https://cb6e6126.ngrok.io/authorize_follow?acct={uri}\"}]}" | ||||
|       expect(json[:subject]).to eq 'acct:alice@cb6e6126.ngrok.io' | ||||
|       expect(json[:aliases]).to include('https://cb6e6126.ngrok.io/@alice', 'https://cb6e6126.ngrok.io/users/alice') | ||||
|     end | ||||
|  | ||||
|     it 'returns http not found when account can not be found with alternate domains' do | ||||
|       Rails.configuration.x.alternate_domains = ["foo.org"] | ||||
|       username, domain = alice.to_webfinger_s.split("@") | ||||
|       Rails.configuration.x.alternate_domains = ['foo.org'] | ||||
|       username, = alice.to_webfinger_s.split('@') | ||||
|  | ||||
|       get :show, params: { resource: "#{username}@bar.org" }, format: :json | ||||
|  | ||||
|   | ||||
| @@ -1,15 +0,0 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| require 'rails_helper' | ||||
|  | ||||
| describe Activitystreams2BuilderHelper, type: :helper do | ||||
|   it 'returns display name if present' do | ||||
|     account = Fabricate(:account, display_name: 'display name', username: 'username') | ||||
|     expect(account_name(account)).to eq 'display name' | ||||
|   end | ||||
|  | ||||
|   it 'returns username if display name is not present' do | ||||
|     account = Fabricate(:account, display_name: '', username: 'username') | ||||
|     expect(account_name(account)).to eq 'username' | ||||
|   end | ||||
| end | ||||
		Reference in New Issue
	
	Block a user