pleroma/packages/pl-api/lib/features.ts
2024-09-13 13:05:06 +02:00

1238 lines
35 KiB
TypeScript

import semverCoerce from 'semver/functions/coerce';
import gte from 'semver/functions/gte';
import semverParse from 'semver/functions/parse';
import type { Instance } from './entities';
/** Truthy array convenience function */
const any = (arr: Array<any>): boolean => arr.some(Boolean);
/**
* Ditto, a Nostr server with Mastodon API.
* @see {@link https://gitlab.com/soapbox-pub/ditto}
*/
const DITTO = 'Ditto';
/**
* Firefish, a fork of Misskey. Formerly known as Calckey.
* @see {@link https://joinfirefish.org/}
*/
const FIREFISH = 'Firefish';
/**
* Friendica, decentralized social platform implementing multiple federation protocols.
* @see {@link https://friendi.ca/}
*/
const FRIENDICA = 'Friendica';
/**
* GoToSocial, an ActivityPub server writter in Golang.
* @see {@link https://gotosocial.org/}
*/
const GOTOSOCIAL = 'GoToSocial';
/**
* Iceshrimp, yet another Misskey fork.
* @see {@link https://iceshrimp.dev/}
*/
const ICESHRIMP = 'Iceshrimp';
/**
* Mastodon, the software upon which this is all based.
* @see {@link https://joinmastodon.org/}
*/
const MASTODON = 'Mastodon';
/**
* Mitra, a Rust backend with cryptocurrency integrations.
* @see {@link https://codeberg.org/silverpill/mitra}
*/
const MITRA = 'Mitra';
/**
* Pixelfed, a federated image sharing platform.
* @see {@link https://pixelfed.org/}
*/
const PIXELFED = 'Pixelfed';
/**
* Pleroma, a feature-rich alternative written in Elixir.
* @see {@link https://pleroma.social/}
*/
const PLEROMA = 'Pleroma';
/**
* Takahē, backend with support for serving multiple domains.
* @see {@link https://jointakahe.org/}
*/
const TAKAHE = 'Takahe';
/**
* Toki, a C# Fediverse server.
* @see {@link https://github.com/purifetchi/Toki}
*/
const TOKI = 'Toki';
/**
* Akkoma, a Pleroma fork.
* @see {@link https://akkoma.dev/AkkomaGang/akkoma}
*/
const AKKOMA = 'akkoma';
/**
* glitch-soc, fork of Mastodon with a number of experimental features.
* @see {@link https://glitch-soc.github.io/docs/}
*/
const GLITCH = 'glitch';
/**
* Pl, fork of Pleroma developed by pl-api author.
* @see {@link https://github.com/mkljczk/pl}
*/
const PL = 'pl';
/**
* Rebased, fork of Pleroma developed by Soapbox author.
* @see {@link https://gitlab.com/soapbox-pub/rebased}
*/
const REBASED = 'soapbox';
/** Backend name reserved only for tests. */
const UNRELEASED = 'unreleased';
/** Parse features for the given instance */
const getFeatures = (instance: Instance) => {
const v = parseVersion(instance.version || '');
const federation = !!instance.pleroma.metadata.federation.enabled;
return {
version: v,
/**
* Ability to set description of profile avatar and header.
* @see PATCH /api/v1/accounts/update_credentials
*/
accountAvatarDescription: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
v.software === PLEROMA && v.build === PL,
]),
/**
* Pleroma backups.
* @see GET /api/v1/pleroma/backups
* @see POST /api/v1/pleroma/backups
*/
accountBackups: v.software === PLEROMA,
/**
* The accounts API allows an acct instead of an ID.
* @see GET /api/v1/accounts/:acct_or_id
*/
accountByUsername: v.software === PLEROMA,
/**
* Ability to create accounts.
* @see POST /api/v1/accounts
*/
accountCreation: true,
/**
* @see PATCH /api/v1/accounts/update_credentials
*/
accountEnableRss: v.software === GOTOSOCIAL,
/**
* Ability to pin other accounts on one's profile.
* @see POST /api/v1/accounts/:id/pin
* @see POST /api/v1/accounts/:id/unpin
* @see GET /api/v1/pleroma/accounts/:id/endorsements
*/
accountEndorsements: v.software === PLEROMA && gte(v.version, '2.5.0'),
/**
* Ability to set one's location on their profile.
* @see PATCH /api/v1/accounts/update_credentials
*/
accountLocation: any([
v.software === PLEROMA && v.build === REBASED && gte(v.version, '2.5.0'),
v.software === PLEROMA && v.build === PL,
]),
/**
* Look up an account by the acct.
* @see GET /api/v1/accounts/lookup
*/
accountLookup: any([
v.software === DITTO,
v.software === FIREFISH,
v.software === GOTOSOCIAL,
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA && gte(v.version, '2.5.0'),
v.software === TAKAHE && gte(v.version, '0.6.1'),
v.software === TOKI,
]),
/**
* Move followers to a different ActivityPub account.
* @see POST /api/pleroma/move_account
*/
accountMoving: v.software === PLEROMA && gte(v.version, '2.5.0'),
/**
* Ability to subscribe to notifications every time an account posts.
* @see POST /api/v1/accounts/:id/follow
*/
accountNotifies: any([
v.software === MASTODON,
v.software === PLEROMA && gte(v.version, '2.5.0'),
v.software === GOTOSOCIAL,
]),
/**
* Ability to address a status to a list of users.
* @see POST /api/v1/statuses
*/
addressableLists: v.software === PLEROMA && gte(v.version, '1.0.2'),
/**
* Can display announcements set by admins.
* @see GET /api/v1/announcements
* @see POST /api/v1/announcements/:id/dismiss
* @see {@link https://docs.joinmastodon.org/methods/announcements/}
*/
announcements: any([
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA,
v.software === TAKAHE && gte(v.version, '0.7.0'),
]),
/**
* Can emoji react to announcements set by admins.
* @see PUT /api/v1/announcements/:id/reactions/:name
* @see DELETE /api/v1/announcements/:id/reactions/:name
* @see {@link https://docs.joinmastodon.org/methods/announcements/}
*/
announcementsReactions: v.software === MASTODON,
/**
* Set your birthday and view upcoming birthdays.
* @see GET /api/v1/pleroma/birthdays
* @see POST /api/v1/accounts
* @see PATCH /api/v1/accounts/update_credentials
*/
birthdays: v.software === PLEROMA,
/**
* Allow to bite users.
* see POST /api/v1/bite
*/
bites: any([
v.software === TOKI,
instance.api_versions['bites.pleroma.pl-api'] >= 1,
]),
/** Whether people who blocked you are visible through the API. */
blockersVisible: instance.api_versions['blockers_visible.pleroma.pl-api'] >= 1,
/**
* Can group bookmarks in folders.
* @see GET /api/v1/pleroma/bookmark_folders
* @see POST /api/v1/pleroma/bookmark_folders
* @see PATCH /api/v1/pleroma/bookmark_folders/:id
* @see DELETE /api/v1/pleroma/bookmark_folders/:id
*/
bookmarkFolders: instance.api_versions['bookmark_folders.pleroma.pl-api'] >= 1,
/**
* Can bookmark statuses.
* @see POST /api/v1/statuses/:id/bookmark
* @see GET /api/v1/bookmarks
*/
bookmarks: any([
v.software === DITTO,
v.software === FIREFISH,
v.software === GOTOSOCIAL,
v.software === ICESHRIMP,
v.software === FRIENDICA,
v.software === MASTODON,
v.software === PIXELFED,
v.software === PLEROMA,
v.software === TAKAHE && gte(v.version, '0.9.0'),
v.software === TOKI,
]),
/**
* Accounts can be marked as bots.
* @see PATCH /api/v1/accounts/update_credentials
*/
bots: any([
v.software === GOTOSOCIAL,
v.software === MASTODON,
v.software === PLEROMA,
]),
/**
* Can display a timeline of statuses from instances selected by instance admin.
* @see GET /api/v1/timelines/bubble
*/
bubbleTimeline: instance.api_versions['bubble_timeline.pleroma.pl-api'] >= 1,
/**
* Pleroma chats API.
* @see {@link https://docs.pleroma.social/backend/development/API/chats/}
*/
chats: instance.api_versions['chat_messages.pleroma.pl-api'] >= 1,
/**
* Ability to delete a chat.
* @see DELETE /api/v1/pleroma/chats/:id
*/
chatsDelete: any([
v.build === REBASED,
v.software === PLEROMA && v.build === PL,
]),
/**
* Mastodon's newer solution for direct messaging.
* @see {@link https://docs.joinmastodon.org/methods/conversations/}
*/
conversations: any([
v.software === FIREFISH,
v.software === FRIENDICA,
v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PIXELFED,
v.software === PLEROMA,
v.software === TAKAHE,
]),
/**
* @see GET /api/v1/conversations
*/
conversationsByRecipients: v.software === PLEROMA,
/**
* @see POST /api/v1/statuses
*/
createStatusExpiration: v.software === PLEROMA,
/**
* Ability to address recipients of a status explicitly (with `to`).
* @see POST /api/v1/statuses
*/
createStatusExplicitAddressing: any([
v.software === DITTO,
v.software === PLEROMA,
]),
/**
* @see POST /api/v1/statuses
*/
createStatusReplyToConversation: v.software === PLEROMA,
/**
* @see POST /api/v1/statuses
*/
createStatusListScope: v.software === PLEROMA,
/**
* @see POST /api/v1/statuses
*/
createStatusLocalScope: v.software === PLEROMA,
/**
* @see POST /api/v1/statuses
*/
createStatusPreview: v.software === PLEROMA,
/**
* Ability to add non-standard reactions to a status.
*/
customEmojiReacts: any([
instance.api_versions['custom_emoji_reactions.pleroma.pl-api'] >= 1,
instance.api_versions['custom_emoji_reactions.pleroma.pl-api'] >= 1,
v.software === PLEROMA && gte(v.version, '2.6.0'),
]),
/**
* @see POST /api/v1/accounts/delete
* @see POST /api/pleroma/delete_account
*/
deleteAccount: any([
v.software === GOTOSOCIAL,
v.software === PLEROMA,
]),
/**
* Allow to register on a given domain
* @see GET /api/v1/pleroma/admin/domains
* @see POST /api/v1/pleroma/admin/domains
* @see PATCH /api/v1/pleroma/admin/domains/:id
* @see DELETE /api/v1/pleroma/admin/domains/:id
*/
domains: any([instance?.pleroma.metadata.multitenancy.enabled]),
/**
* Ability to edit profile information.
* @see PATCH /api/v1/accounts/update_credentials
*/
editProfile: any([
v.software === FIREFISH,
v.software === FRIENDICA,
v.software === GOTOSOCIAL,
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === MITRA,
v.software === PIXELFED,
v.software === PLEROMA,
v.software === TAKAHE && gte(v.version, '0.7.0'),
v.software === TOKI,
]),
/**
* Ability to edit published posts.
* @see PUT /api/v1/statuses/:id
*/
editStatuses: any([
v.software === FRIENDICA && gte(v.version, '2022.12.0'),
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === MITRA,
v.software === TAKAHE && gte(v.version, '0.8.0'),
instance.api_versions['editing.pleroma.pl-api'] >= 1,
]),
/**
* Soapbox email list.
* @see POST /api/v1/accounts
* @see PATCH /api/v1/accounts/update_credentials
* @see GET /api/v1/pleroma/admin/email_list/subscribers.csv
* @see GET /api/v1/pleroma/admin/email_list/unsubscribers.csv
* @see GET /api/v1/pleroma/admin/email_list/combined.csv
*/
emailList: instance.api_versions['email_list.pleroma.pl-api'] >= 1,
/**
* Ability to embed posts on external sites.
* @see GET /api/oembed
*/
embeds: v.software === MASTODON,
/**
* Ability to add emoji reactions to a status.
* @see PUT /api/v1/pleroma/statuses/:id/reactions/:emoji
* @see DELETE /api/v1/pleroma/statuses/:id/reactions/:emoji
*
* @see POST /v1/statuses/:id/react/:emoji
* @see POST /v1/statuses/:id/unreact/:emoji
*/
emojiReacts: any([
v.software === PLEROMA,
instance ? instance.configuration.reactions.max_reactions > 0 : false,
]),
/**
* @see GET /api/v1/pleroma/statuses/:id/reactions/:emoji?
*
* @see GET /api/v1/statuses/:id/emoji_reactioned_by
*/
emojiReactsList: any([
v.software === PLEROMA,
instance.api_versions['emoji_reaction.fedibird.pl-api'] >= 1,
]),
/**
* Ability to create and perform actions on events.
* @see POST /api/v1/pleroma/events
* @see GET /api/v1/pleroma/events/joined_events
* @see PUT /api/v1/pleroma/events/:id
* @see GET /api/v1/pleroma/events/:id/participations
* @see GET /api/v1/pleroma/events/:id/participation_requests
* @see POST /api/v1/pleroma/events/:id/participation_requests/:participant_id/authorize
* @see POST /api/v1/pleroma/events/:id/participation_requests/:participant_id/reject
* @see POST /api/v1/pleroma/events/:id/join
* @see POST /api/v1/pleroma/events/:id/leave
* @see GET /api/v1/pleroma/events/:id/ics
* @see GET /api/v1/pleroma/search/location
*/
events: instance.api_versions['events.pleroma.pl-api'] >= 1,
/** Whether to allow exporting follows/blocks/mutes to CSV by paginating the API. */
exportData: true,
/** Whether the accounts who favourited or emoji-reacted to a status can be viewed through the API. */
exposableReactions: any([
v.software === FIREFISH,
v.software === FRIENDICA,
v.software === GOTOSOCIAL,
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === TAKAHE && gte(v.version, '0.6.1'),
v.software === TOKI,
instance.api_versions['exposable_reactions.pleroma.pl-api'] >= 1,
]),
/**
* Can see accounts' followers you know
* @see GET /api/v1/accounts/familiar_followers
*/
familiarFollowers: any([
v.software === DITTO,
v.software === MASTODON,
v.software === PLEROMA && gte(v.version, '2.6.0') && v.build === REBASED,
v.software === PLEROMA && gte(v.version, '2.7.0'),
v.software === TAKAHE,
]),
/** Whether the instance federates. */
federating: federation,
/**
* Can edit and manage timeline filters (aka "muted words").
* @see {@link https://docs.joinmastodon.org/methods/filters/#v1}
*/
filters: any([
v.software === GOTOSOCIAL,
v.software === PLEROMA,
]),
/**
* Can edit and manage timeline filters (aka "muted words").
* @see {@link https://docs.joinmastodon.org/methods/filters/}
*/
filtersV2: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.0'),
v.software === MASTODON,
]),
/**
* Allows setting the focal point of a media attachment.
* @see {@link https://docs.joinmastodon.org/methods/media/}
*/
focalPoint: any([
v.software === GOTOSOCIAL,
v.software === MASTODON,
]),
/**
* Ability to follow hashtags.
* @see POST /api/v1/tags/:name/follow
* @see POST /api/v1/tags/:name/unfollow
*/
followHashtags: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
v.software === MASTODON && gte(v.compatVersion, '4.0.0'),
v.software === PLEROMA && v.build === AKKOMA,
v.software === PLEROMA && v.build === PL,
v.software === TAKAHE && gte(v.version, '0.9.0'),
]),
/**
* Ability to lock accounts and manually approve followers.
* @see PATCH /api/v1/accounts/update_credentials
*/
followRequests: any([
v.software === GOTOSOCIAL,
v.software === MASTODON,
v.software === MITRA,
v.software === PLEROMA,
v.software === TOKI,
]),
/**
* Ability to list followed hashtags.
* @see GET /api/v1/followed_tags
*/
followedHashtagsList: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
v.software === MASTODON && gte(v.compatVersion, '4.1.0'),
v.software === PLEROMA && v.build === AKKOMA,
v.software === PLEROMA && v.build === PL,
v.software === TAKAHE && gte(v.version, '0.9.0'),
]),
/**
* Whether client settings can be retrieved from the API.
* @see GET /api/pleroma/frontend_configurations
*/
frontendConfigurations: any([
v.software === DITTO,
v.software === PLEROMA,
]),
/**
* Groups.
* @see POST /api/v1/groups
* @see GET /api/v1/groups
* @see GET /api/v1/groups/:id
* @see POST /api/v1/groups/:id/join
* @see POST /api/v1/groups/:id/leave
* @see GET /api/v1/groups/:id/memberships
* @see PUT /api/v1/groups/:group_id
* @see DELETE /api/v1/groups/:group_id
* @see GET /api/v1/groups/:group_id/membership_requests
* @see POST /api/v1/groups/:group_id/membership_requests/:account_id/authorize
* @see POST /api/v1/groups/:group_id/membership_requests/:account_id/reject
* @see DELETE /api/v1/groups/:group_id/statuses/:id
* @see POST /api/v1/groups/:group_id/kick?account_ids[]=…
* @see GET /api/v1/groups/:group_id/blocks
* @see POST /api/v1/groups/:group_id/blocks?account_ids[]=…
* @see DELETE /api/v1/groups/:group_id/blocks?account_ids[]=…
* @see POST /api/v1/groups/:group_id/promote?role=new_role&account_ids[]=…
* @see POST /api/v1/groups/:group_id/demote?role=new_role&account_ids[]=…
* @see GET /api/v1/admin/groups
* @see GET /api/v1/admin/groups/:group_id
* @see POST /api/v1/admin/groups/:group_id/suspend
* @see POST /api/v1/admin/groups/:group_id/unsuspend
* @see DELETE /api/v1/admin/groups/:group_id
*/
groups: instance.api_versions['pleroma:groups.pleroma.pl-api'] >= 1,
/**
* Can hide follows/followers lists and counts.
* @see PATCH /api/v1/accounts/update_credentials
*/
hideNetwork: any([
v.software === GOTOSOCIAL && gte(v.version, '0.15.0'),
v.software === PLEROMA,
]),
/**
* Import a .csv file with a list of blocked users.
* @see POST /api/pleroma/blocks_import
* @see POST /api/v1/import
*/
importBlocks: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
v.software === PLEROMA,
]),
/**
* Import a .csv file with a list of followed users.
* @see POST /api/pleroma/follow_import
* @see POST /api/v1/import
*/
importFollows: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
v.software === PLEROMA,
]),
/**
* Import a .csv file with a list of muted users.
* @see POST /api/pleroma/mutes_import
*/
importMutes: v.software === PLEROMA,
/**
* Allow to specify mode of data import to either `merge` or `overwrite`.
* @see POST /api/v1/import
*/
importOverwrite: v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
/**
* View posts from specific instance.
* @see GET /api/v1/timelines/public
*/
instanceTimeline: v.software === PLEROMA,
/**
* Mastodon server information API v2.
* @see GET /api/v2/instance
* @see {@link https://docs.joinmastodon.org/methods/instance/#v2}
*/
instanceV2: any([
v.software === GOTOSOCIAL,
v.software === MASTODON && gte(v.compatVersion, '4.0.0'),
v.software === PLEROMA && v.build === REBASED && gte(v.version, '2.6.0'),
v.software === PLEROMA && gte(v.version, '2.7.0'),
]),
interactionRequests: v.software === GOTOSOCIAL && gte(v.version, '0.16.1'),
/**
* Server-side status language detection.
*/
languageDetection: instance.api_versions['language_detection.pleroma.pl-api'] >= 1,
/**
* Can translate multiple statuses in a single request.
* @see POST /api/v1/pl/statuses/translate
*/
lazyTranslations: instance.api_versions['translations.pl.pl-api'] >= 1,
/**
* Can create, view, and manage lists.
* @see {@link https://docs.joinmastodon.org/methods/lists/}
* @see GET /api/v1/timelines/list/:list_id
*/
lists: any([
v.software === FIREFISH,
v.software === FRIENDICA,
v.software === GOTOSOCIAL,
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA,
]),
/**
* Ability to post statuses that don't federate.
* @see POST /api/v1/statuses
*/
localOnlyStatuses: federation && v.software === GOTOSOCIAL,
/**
* Can sign in using username instead of e-mail address.
*/
logInWithUsername: any([
v.software === PLEROMA,
v.software === TOKI,
]),
/**
* Can view and manage ActivityPub aliases through the API.
* @see GET /api/pleroma/aliases
* @see PATCH /api/v1/accounts/update_credentials
*/
manageAccountAliases: v.software === PLEROMA,
/**
* @see GET /api/pleroma/accounts/mfa
* @see GET /api/pleroma/accounts/mfa/backup_codes
* @see GET /api/pleroma/accounts/mfa/setup/:method
* @see POST /api/pleroma/accounts/mfa/confirm/:method
* @see DELETE /api/pleroma/accounts/mfa/:method
*/
manageMfa: v.software === PLEROMA,
/**
* Can perform moderation actions with account and reports.
* @see {@link https://docs.joinmastodon.org/methods/admin/}
* @see GET /api/v1/admin/reports
* @see POST /api/v1/admin/reports/:report_id/resolve
* @see POST /api/v1/admin/reports/:report_id/reopen
* @see POST /api/v1/admin/accounts/:account_id/action
* @see POST /api/v1/admin/accounts/:account_id/approve
*/
mastodonAdmin: any([
v.software === DITTO,
v.software === GOTOSOCIAL,
v.software === MASTODON,
v.software === PLEROMA && v.build === REBASED && gte(v.version, '2.5.0'),
v.software === PLEROMA && v.build === PL,
]),
/**
* Can perform moderation actions with account and reports.
* @see {@link https://docs.joinmastodon.org/methods/admin/}
* @see GET /api/v2/admin/accounts
*/
mastodonAdminV2: any([
v.software === MASTODON && gte(v.version, '3.5.0'),
]),
/**
* Supports V2 media uploads.
* @see POST /api/v2/media
*/
mediaV2: any([
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === MITRA,
v.software === PLEROMA,
v.software === TAKAHE,
v.software === TOKI,
]),
/**
* Ability to include multiple language variants for a post.
* @see POST /api/v1/statuses
*/
multiLanguage: instance.api_versions['multi_language.pleroma.pl-api'] >= 1,
/**
* Ability to hide notifications from people you don't follow.
* @see PUT /api/pleroma/notification_settings
*/
muteStrangers: v.software === PLEROMA,
/**
* Ability to mute users.
* @see GET /api/v1/mutes
* @see POST /api/v1/accounts/:id/mute
* @see POST /api/v1/accounts/:id/unmute
*/
mutes: any([
v.software === FIREFISH,
v.software === FRIENDICA,
v.software === GOTOSOCIAL && gte(v.version, '0.16.0'),
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === MITRA,
v.software === PIXELFED,
v.software === PLEROMA,
v.software === TAKAHE,
]),
/**
* Ability to specify how long the account mute should last.
* @see PUT /api/v1/accounts/:id/mute
*/
mutesDuration: any([
v.software === FIREFISH,
v.software === GOTOSOCIAL && gte(v.version, '0.16.0'),
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA,
v.software === TAKAHE,
]),
/**
* Add private notes to accounts.
* @see POST /api/v1/accounts/:id/note
* @see GET /api/v1/accounts/relationships
*/
notes: any([
v.software === MASTODON,
v.software === PLEROMA && gte(v.version, '2.5.0'),
v.software === GOTOSOCIAL,
]),
/**
* @see DELETE /api/v1/notifications/destroy_multiple
*/
notificationsDismissMultiple: v.software === PLEROMA,
/**
* @see GET /api/v1/notifications
*/
notificationsExcludeVisibilities: v.software === PLEROMA,
/**
* Allows specifying notification types to include, rather than to exclude.
* @see GET /api/v1/notifications
*/
notificationsIncludeTypes: any([
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA && gte(v.version, '2.5.0'),
v.software === TAKAHE && gte(v.version, '0.6.2'),
v.software === GOTOSOCIAL,
]),
pleromaAdminAccounts: v.software === PLEROMA,
/**
* Ability to manage announcements by admins.
* @see GET /api/v1/pleroma/admin/announcements
* @see GET /api/v1/pleroma/admin/announcements/:id
* @see POST /api/v1/pleroma/admin/announcements
* @see PATCH /api/v1/pleroma/admin/announcements/:id
* @see DELETE /api/v1/pleroma/admin/announcements/:id
* @see {@link https://docs.pleroma.social/backend/development/API/admin_api/#get-apiv1pleromaadminannouncements}
*/
pleromaAdminAnnouncements: v.software === PLEROMA,
pleromaAdminModerationLog: v.software === PLEROMA,
pleromaAdminRelays: v.software === PLEROMA,
/**
* Ability to manage instance rules by admins.
* @see GET /api/v1/pleroma/admin/rules
* @see POST /api/v1/pleroma/admin/rules
* @see PATCH /api/v1/pleroma/admin/rules/:id
* @see DELETE /api/v1/pleroma/admin/rules/:id
*/
pleromaAdminRules: any([
v.software === PLEROMA && v.build === REBASED && gte(v.version, '2.5.0'),
v.software === PLEROMA && gte(v.version, '2.7.0'),
]),
pleromaAdminStatuses: v.software === PLEROMA,
/**
* Displays a form to follow a user when logged out.
* @see POST /main/ostatus
*/
pleromaRemoteFollow: v.software === PLEROMA,
/**
* Can add polls to statuses.
* @see POST /api/v1/statuses
*/
polls: any([
v.software === FIREFISH,
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA,
v.software === TAKAHE && gte(v.version, '0.8.0'),
v.software === GOTOSOCIAL,
]),
/**
* Can select a language for statuses.
* @see POST /api/v1/statuses
*/
postLanguages: any([
v.software === MASTODON,
v.software === PLEROMA && v.build === AKKOMA,
v.software === PLEROMA && v.build === PL,
v.software === PLEROMA && v.build === REBASED,
v.software === GOTOSOCIAL,
]),
/**
* @see GET /api/v1/akkoma/preferred_frontend/available
* @see PUT /api/v1/akkoma/preferred_frontend
*/
preferredFrontends: v.software === PLEROMA && v.build === AKKOMA,
/**
* Can set privacy scopes on statuses.
* @see POST /api/v1/statuses
*/
privacyScopes: true,
/**
* A directory of discoverable profiles from the instance.
* @see {@link https://docs.joinmastodon.org/methods/directory/}
*/
profileDirectory: any([
v.software === FRIENDICA,
v.software === MASTODON,
v.software === MITRA,
instance.api_versions['profile_directory.pleroma.pl-api'] >= 1,
]),
/**
* Ability to set custom profile fields.
* @see PATCH /api/v1/accounts/update_credentials
*/
profileFields: any([
v.software === MASTODON,
v.software === PLEROMA,
v.software === TAKAHE && gte(v.version, '0.7.0'),
v.software === MITRA,
v.software === GOTOSOCIAL,
]),
/**
* Returns favorites timeline of any user
* @see GET /api/v1/pleroma/accounts/:id/favourites
*/
publicFavourites: v.software === PLEROMA,
/**
* Can display a timeline of all known public statuses.
* Local and Fediverse timelines both use this feature.
* @see GET /api/v1/timelines/public
*/
publicTimeline: any([
v.software === DITTO,
v.software === FIREFISH,
v.software === FRIENDICA,
v.software === GOTOSOCIAL,
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA,
v.software === TAKAHE,
v.software === TOKI,
]),
/**
* Ability to quote posts in statuses.
* @see POST /api/v1/statuses
*/
quotePosts: any([
v.software === FRIENDICA && gte(v.version, '2023.3.0'),
v.software === PLEROMA && [REBASED, AKKOMA].includes(v.build!) && gte(v.version, '2.5.0'),
instance.api_versions['quote_posting.pleroma.pl-api'] >= 1,
instance?.feature_quote === true,
]),
/**
* Ability to boost a status to a selected scope.
* @see POST /api/v1/statuses/:id/reblog
*/
reblogVisibility: any([
v.software === MASTODON,
v.software === PLEROMA,
]),
/**
* Interact with statuses from another instance while logged-out.
* @see POST /api/v1/pleroma/remote_interaction
*/
remoteInteractions: v.software === PLEROMA && gte(v.version, '2.5.0'),
/**
* Ability to remove an account from your followers.
* @see POST /api/v1/accounts/:id/remove_from_followers
*/
removeFromFollowers: any([
v.software === MASTODON,
v.software === PLEROMA && gte(v.version, '2.5.0'),
v.software === PLEROMA && v.build === AKKOMA,
]),
/**
* Can request a password reset email through the API.
* @see POST /auth/password
*/
resetPassword: v.software === PLEROMA,
/**
* Ability to post statuses in Markdown, BBCode, and HTML.
* @see POST /api/v1/statuses
*/
richText: any([
v.software === MASTODON && v.build === GLITCH,
v.software === PLEROMA,
v.software === MITRA,
v.software === GOTOSOCIAL,
]),
/**
* Ability to follow account feeds using RSS.
*/
rssFeeds: any([
v.software === MASTODON,
v.software === PLEROMA,
v.software === GOTOSOCIAL,
]),
/**
* Can schedule statuses to be posted at a later time.
* @see POST /api/v1/statuses
* @see {@link https://docs.joinmastodon.org/methods/scheduled_statuses/}
*/
scheduledStatuses: any([
v.software === FRIENDICA,
v.software === MASTODON,
v.software === PLEROMA,
v.software === GOTOSOCIAL,
]),
/**
* Can create Listen activities
* @see GET /api/v1/pleroma/accounts/:id/scrobbles
* @see POST /api/v1/pleroma/scrobble
*/
scrobbles: v.software === PLEROMA && v.build !== AKKOMA,
/**
* Ability to search statuses from the given account.
* @see {@link https://docs.joinmastodon.org/methods/search/}
* @see POST /api/v2/search
*/
searchFromAccount: any([
v.software === ICESHRIMP,
v.software === MASTODON,
v.software === PLEROMA,
v.software === GOTOSOCIAL,
]),
/**
* @see POST /api/v1/user/email_change
*/
changeEmail: any([
v.software === GOTOSOCIAL && gte(v.version, '0.16.0'),
v.software === PLEROMA,
]),
/**
* @see POST /api/v1/user/password_change
* @see POST /api/v1/settings/change_password
* @see POST /api/pleroma/change_password
*/
changePassword: any([
v.software === GOTOSOCIAL,
v.software === MITRA,
v.software === PLEROMA,
]),
/**
* Ability to manage account sessions.
* @see GET /api/oauth_tokens.json
* @see DELETE /api/oauth_tokens/:id
*/
sessions: v.software === PLEROMA,
/**
* Can store client settings in the database.
* @see PATCH /api/v1/accounts/update_credentials
*/
settingsStore: v.software === PLEROMA,
/**
* Can set content warnings on statuses.
* @see POST /api/v1/statuses
*/
spoilers: true,
/**
* @see POST /api/friendica/statuses/:id/dislike
* @see POST /api/friendica/statuses/:id/undislike
* @see GET /api/friendica/statuses/:id/disliked_by
*/
statusDislikes: v.software === FRIENDICA && gte(v.version, '2023.3.0'),
/**
* Can display suggested accounts.
* @see {@link https://docs.joinmastodon.org/methods/suggestions/}
*/
suggestions: any([
v.software === FRIENDICA,
v.software === ICESHRIMP,
v.software === MASTODON,
instance.api_versions['v2_suggestions.pleroma.pl-api'] >= 1,
]),
/**
* Remove an account from follow suggestions
* @see DELETE /api/v1/suggestions/:account_id
*/
suggestionsDismiss: any([
v.software === MASTODON,
]),
/**
* Supports V2 suggested accounts.
* @see GET /api/v2/suggestions
*/
suggestionsV2: any([
v.software === FRIENDICA,
v.software === ICESHRIMP,
v.software === MASTODON,
instance.api_versions['v2_suggestions.pleroma.pl-api'] >= 1,
]),
/**
* Can translate statuses.
* @see POST /api/v1/statuses/:id/translate
*/
translations: any([
instance.api_versions['translation.pleroma.pl-api'] >= 1,
instance.api_versions['machine_translation.akkoma.pl-api'] >= 1,
instance?.configuration.translation.enabled,
]),
/**
* Trending links.
* @see GET /api/v1/trends/links
*/
trendingLinks: v.software === MASTODON && gte(v.compatVersion, '3.5.0'),
/**
* Trending statuses.
* @see GET /api/v1/trends/statuses
*/
trendingStatuses: any([
v.software === DITTO,
v.software === FRIENDICA && gte(v.version, '2022.12.0'),
v.software === ICESHRIMP,
v.software === MASTODON,
]),
/**
* Can display trending hashtags.
* @see GET /api/v1/trends
*/
trends: any([
v.software === DITTO,
v.software === FRIENDICA && gte(v.version, '2022.12.0'),
v.software === ICESHRIMP,
v.software === MASTODON,
]),
/**
* Whether the backend allows adding users you don't follow to lists.
* @see POST /api/v1/lists/:id/accounts
*/
unrestrictedLists: v.software === PLEROMA,
/**
* Ability to post statuses only to accounts with mutual relationship.
* @see POST /api/v1/statuses
*/
visibilityMutualsOnly: v.software === GOTOSOCIAL,
/**
* Ability to post statuses that don't federate.
* @see POST /api/v1/statuses
*/
visibilityLocalOnly: federation && any([
v.software === PLEROMA,
]),
};
};
/** Features available from a backend */
type Features = ReturnType<typeof getFeatures>;
/** Fediverse backend */
interface Backend {
/** Build name, if this software is a fork */
build: string | null;
/** Name of the software */
software: string | null;
/** API version number */
version: string;
/** Mastodon API version this backend is compatible with */
compatVersion: string;
}
/** Get information about the software from its version string */
const parseVersion = (version: string): Backend => {
const regex = /^([\w+.-]*)(?: \(compatible; ([\w]*) (.*)\))?$/;
const match = regex.exec(version.replace('/', ' '));
const semverString = match && (match[3] || match[1]);
const semver = match ? semverParse(semverString) || semverCoerce(semverString, {
loose: true,
}) : null;
const compat = match ? semverParse(match[1]) || semverCoerce(match[1]) : null;
if (match && semver && compat) {
return {
build: semver.build[0],
compatVersion: compat.version,
software: match[2] || MASTODON,
version: semver.version.split('-')[0],
};
} else {
// If we can't parse the version, this is a new and exotic backend.
// Fall back to minimal featureset.
return {
build: null,
compatVersion: '0.0.0',
software: null,
version: '0.0.0',
};
}
};
export {
DITTO,
FIREFISH,
FRIENDICA,
GOTOSOCIAL,
ICESHRIMP,
MASTODON,
MITRA,
PIXELFED,
PLEROMA,
TAKAHE,
TOKI,
AKKOMA,
GLITCH,
REBASED,
PL,
UNRELEASED,
type Features,
type Backend as BackendVersion,
getFeatures,
};