Merge branch 'master' into Comments
This commit is contained in:
commit
da2495e88f
69 changed files with 1903 additions and 645 deletions
90
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
Normal file
90
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
name: Bug Report
|
||||||
|
description: Helps us improve our product!
|
||||||
|
labels: "Needs triage, [Type] Bug"
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
### Thanks for contributing!
|
||||||
|
|
||||||
|
Please write a clear title, then fill in the fields below and submit.
|
||||||
|
|
||||||
|
Please **do not** link to image hosting services such as Cloudup, Droplr, Imgur, etc…
|
||||||
|
Instead, directly embed screenshot(s) or recording(s) in any of the text areas below: click, then drag and drop.
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
---
|
||||||
|
## Core Information
|
||||||
|
- type: textarea
|
||||||
|
id: summary
|
||||||
|
attributes:
|
||||||
|
label: Quick summary
|
||||||
|
- type: textarea
|
||||||
|
id: steps
|
||||||
|
attributes:
|
||||||
|
label: Steps to reproduce
|
||||||
|
placeholder: |
|
||||||
|
1. Start at `site-domain.com/blog`.
|
||||||
|
2. Click on any blog post.
|
||||||
|
3. ...
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: textarea
|
||||||
|
id: expected
|
||||||
|
attributes:
|
||||||
|
label: What you expected to happen
|
||||||
|
placeholder: |
|
||||||
|
e.g. The post should appear.
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: textarea
|
||||||
|
id: actual
|
||||||
|
attributes:
|
||||||
|
label: What actually happened
|
||||||
|
placeholder: |
|
||||||
|
e.g. The post did not appear.
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: dropdown
|
||||||
|
id: users-affected
|
||||||
|
attributes:
|
||||||
|
label: Impact
|
||||||
|
description: Approximately how many users are impacted?
|
||||||
|
options:
|
||||||
|
- One
|
||||||
|
- Some (< 50%)
|
||||||
|
- Most (> 50%)
|
||||||
|
- All
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: dropdown
|
||||||
|
id: workarounds
|
||||||
|
attributes:
|
||||||
|
label: Available workarounds?
|
||||||
|
options:
|
||||||
|
- No and the platform is unusable
|
||||||
|
- No but the platform is still usable
|
||||||
|
- Yes, difficult to implement
|
||||||
|
- Yes, easy to implement
|
||||||
|
- There is no user impact
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Optional Information
|
||||||
|
|
||||||
|
The following section is optional.
|
||||||
|
- type: textarea
|
||||||
|
id: logs
|
||||||
|
attributes:
|
||||||
|
label: Logs or notes
|
||||||
|
placeholder: |
|
||||||
|
Add any information that may be relevant, such as:
|
||||||
|
- Browser/Platform
|
||||||
|
- Theme
|
||||||
|
- Logs/Errors
|
34
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
Normal file
34
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
name: Feature Request
|
||||||
|
description: Suggest an idea for the ActivityPub plugin!
|
||||||
|
title: "Feature Request:"
|
||||||
|
labels: ["[Type] Feature Request"]
|
||||||
|
body:
|
||||||
|
- type: markdown
|
||||||
|
attributes:
|
||||||
|
value: |
|
||||||
|
Please, be as descriptive as possible. Issues lacking detail, or for any other reason than to request a feature, may be closed without action.
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: what
|
||||||
|
attributes:
|
||||||
|
label: What
|
||||||
|
description: Add a concise description of the feature being requested.
|
||||||
|
placeholder: eg. I would like a new dropdown at <xyz>...
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: why
|
||||||
|
attributes:
|
||||||
|
label: Why
|
||||||
|
description: Add a description of the problem this feature solves.
|
||||||
|
placeholder: |
|
||||||
|
eg. This will solve my accessibility needs.
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: how
|
||||||
|
attributes:
|
||||||
|
label: How
|
||||||
|
description: If applicable, add screenshots, mockup, animations and/or videos to help illustrate how the feature could be done.
|
17
.github/stale.yml
vendored
17
.github/stale.yml
vendored
|
@ -1,17 +0,0 @@
|
||||||
# Number of days of inactivity before an issue becomes stale
|
|
||||||
daysUntilStale: 120
|
|
||||||
# Number of days of inactivity before a stale issue is closed
|
|
||||||
daysUntilClose: 7
|
|
||||||
# Issues with these labels will never be considered stale
|
|
||||||
exemptLabels:
|
|
||||||
- pinned
|
|
||||||
- security
|
|
||||||
# Label to use when marking an issue as stale
|
|
||||||
staleLabel: wontfix
|
|
||||||
# Comment to post when marking an issue as stale. Set to `false` to disable
|
|
||||||
markComment: >
|
|
||||||
This issue has been automatically marked as stale because it has not had
|
|
||||||
recent activity. It will be closed if no further activity occurs. Thank you
|
|
||||||
for your contributions.
|
|
||||||
# Comment to post when closing a stale issue. Set to `false` to disable
|
|
||||||
closeComment: false
|
|
50
.github/workflows/gardening.yml
vendored
Normal file
50
.github/workflows/gardening.yml
vendored
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
# Repo gardening. Automate some of the triage tasks in the repo.
|
||||||
|
name: Repo Gardening
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request_target: # When a PR is opened, edited, updated, closed, or a label is added.
|
||||||
|
types: [opened, reopened, synchronize, edited, labeled, closed]
|
||||||
|
issues: # For auto-triage of issues.
|
||||||
|
types: [opened, labeled, reopened, edited, closed]
|
||||||
|
issue_comment: # To gather support references in issue comments.
|
||||||
|
types: [created]
|
||||||
|
concurrency:
|
||||||
|
# For pull_request_target, cancel any concurrent jobs with the same type (e.g. "opened", "labeled") and branch.
|
||||||
|
# Don't cancel any for other events, accomplished by grouping on the unique run_id.
|
||||||
|
group: gardening-${{ github.event_name }}-${{ github.event.action }}-${{ github.event_name == 'pull_request_target' && github.event.pull_request.head.ref || github.run_id }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
repo-gardening:
|
||||||
|
name: 'Automated repo gardening.'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: github.event_name == 'pull_request_target' || github.event.pull_request.head.repo.full_name == github.event.pull_request.base.repo.full_name
|
||||||
|
timeout-minutes: 10
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Setup Node
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: lts/*
|
||||||
|
|
||||||
|
- name: Wait for prior instances of the workflow to finish
|
||||||
|
uses: softprops/turnstyle@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: 'Automate triage (add labels, clean labels, ...).'
|
||||||
|
uses: automattic/action-repo-gardening@trunk
|
||||||
|
with:
|
||||||
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
slack_token: ${{ secrets.SLACK_TOKEN }}
|
||||||
|
slack_team_channel: ${{ secrets.SLACK_TEAM_CHANNEL }}
|
||||||
|
slack_he_triage_channel: ${{ secrets.SLACK_HE_TRIAGE_CHANNEL }}
|
||||||
|
slack_quality_channel: ${{ secrets.SLACK_QUALITY_CHANNEL }}
|
||||||
|
tasks: 'addLabels,cleanLabels,assignIssues,flagOss,gatherSupportReferences,replyToCustomersReminder'
|
||||||
|
add_labels: '[
|
||||||
|
{"path": "src/followers", "label": "[Block] Followers"},
|
||||||
|
{"path": "src/follow-me", "label": "[Block] Follow Me"}
|
||||||
|
]'
|
19
.github/workflows/stale.yml
vendored
Normal file
19
.github/workflows/stale.yml
vendored
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
name: 'Close stale issues and PRs'
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '30 1 * * *'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
stale:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/stale@v8
|
||||||
|
with:
|
||||||
|
stale-issue-message: 'This issue is stale because it has been open 120 days with no activity. Remove stale label or comment or this will be closed in 5 days.'
|
||||||
|
days-before-stale: 120
|
||||||
|
days-before-close: 7
|
||||||
|
exempt-all-pr-assignees: true
|
||||||
|
exempt-all-assignees: true
|
||||||
|
exempt-all-pr-milestones: true
|
||||||
|
exempt-all-issue-milestones: true
|
||||||
|
start-date: '2019-02-01T00:00:00Z'
|
15
.php_cs
15
.php_cs
|
@ -1,15 +0,0 @@
|
||||||
<?php
|
|
||||||
$finder = PhpCsFixer\Finder::create()
|
|
||||||
->exclude('vendor')
|
|
||||||
->exclude('node_modules')
|
|
||||||
->exclude('bin')
|
|
||||||
->in(__DIR__)
|
|
||||||
;
|
|
||||||
|
|
||||||
return PhpCsFixer\Config::create()
|
|
||||||
->setRules([
|
|
||||||
'native_function_invocation' => ['include' => ['@all']],
|
|
||||||
'native_constant_invocation' => true,
|
|
||||||
])
|
|
||||||
->setFinder($finder)
|
|
||||||
;
|
|
86
README.md
86
README.md
|
@ -1,9 +1,9 @@
|
||||||
# ActivityPub #
|
# ActivityPub #
|
||||||
**Contributors:** [automattic](https://profiles.wordpress.org/automattic/), [pfefferle](https://profiles.wordpress.org/pfefferle/), [mediaformat](https://profiles.wordpress.org/mediaformat/), [mattwiebe](https://profiles.wordpress.org/mattwiebe/), [akirk](https://profiles.wordpress.org/akirk/), [jeherve](https://profiles.wordpress.org/jeherve/), [nuriapena](https://profiles.wordpress.org/nuriapena/)
|
**Contributors:** [automattic](https://profiles.wordpress.org/automattic/), [pfefferle](https://profiles.wordpress.org/pfefferle/), [mediaformat](https://profiles.wordpress.org/mediaformat/), [mattwiebe](https://profiles.wordpress.org/mattwiebe/), [akirk](https://profiles.wordpress.org/akirk/), [jeherve](https://profiles.wordpress.org/jeherve/), [nuriapena](https://profiles.wordpress.org/nuriapena/), [cavalierlife](https://profiles.wordpress.org/cavalierlife/)
|
||||||
**Tags:** OStatus, fediverse, activitypub, activitystream
|
**Tags:** OStatus, fediverse, activitypub, activitystream
|
||||||
**Requires at least:** 4.7
|
**Requires at least:** 4.7
|
||||||
**Tested up to:** 6.3
|
**Tested up to:** 6.3
|
||||||
**Stable tag:** 1.0.0
|
**Stable tag:** 1.0.1
|
||||||
**Requires PHP:** 5.6
|
**Requires PHP:** 5.6
|
||||||
**License:** MIT
|
**License:** MIT
|
||||||
**License URI:** http://opensource.org/licenses/MIT
|
**License URI:** http://opensource.org/licenses/MIT
|
||||||
|
@ -12,39 +12,39 @@ The ActivityPub protocol is a decentralized social networking protocol based upo
|
||||||
|
|
||||||
## Description ##
|
## Description ##
|
||||||
|
|
||||||
This is BETA software, see the FAQ to see the current feature set or rather what is still planned.
|
Enter the fediverse with **ActivityPub**, broadcasting your blog to a wider audience! Attract followers, deliver updates, and receive comments from a diverse user base of **ActivityPub**\-compliant platforms.
|
||||||
|
|
||||||
The plugin implements the ActivityPub protocol for your blog, which means that your readers will be able to follow your blog posts on Mastodon and other federated platforms that support ActivityPub. In addition, replies to your posts on Mastodon and related platforms will automatically become comments on your blog post.
|
With the ActivityPub plugin installed, your WordPress blog itself function as a federated profile, along with profiles for each author. For instance, if your website is `example.com`, then the blog-wide profile can be found at `@example.com@example.com`, and authors like Jane and Bob would have their individual profiles at `@jane@example.com` and `@bobz@example.com`, respectively.
|
||||||
|
|
||||||
|
An example: I give you my Mastodon profile name: `@pfefferle@mastodon.social`. You search, see my profile, and hit follow. Now, any post I make appears in your Home feed. Similarly, with the ActivityPub plugin, you can find and follow Jane's profile at `@jane@example.com`.
|
||||||
|
|
||||||
|
Once you follow Jane's `@jane@example.com` profile, any blog post she crafts on `example.com` will land in your Home feed. Simultaneously, by following the blog-wide profile `@example.com@example.com`, you'll receive updates from all authors.
|
||||||
|
|
||||||
|
**Note**: if no one follows your author or blog instance, your posts remain unseen. The simplest method to verify the plugin's operation is by following your profile. If you possess a Mastodon profile, initiate by following your new one.
|
||||||
|
|
||||||
The plugin works with the following tested federated platforms, but there may be more that it works with as well:
|
The plugin works with the following tested federated platforms, but there may be more that it works with as well:
|
||||||
|
|
||||||
* [Mastodon](https://joinmastodon.org/)
|
* [Mastodon](https://joinmastodon.org/)
|
||||||
* [Pleroma](https://pleroma.social/)
|
* [Pleroma](https://pleroma.social/)/[Akkoma](https://akkoma.social/)
|
||||||
* [friendica](https://friendi.ca/)
|
* [friendica](https://friendi.ca/)
|
||||||
* [Hubzilla](https://hubzilla.org/)
|
* [Hubzilla](https://hubzilla.org/)
|
||||||
* [Pixelfed](https://pixelfed.org/)
|
* [Pixelfed](https://pixelfed.org/)
|
||||||
* [Socialhome](https://socialhome.network/)
|
* [Socialhome](https://socialhome.network/)
|
||||||
* [Misskey](https://join.misskey.page/)
|
* [Misskey](https://join.misskey.page/)
|
||||||
* [Calckey](https://calckey.org/)
|
* [Firefish](https://joinfirefish.org/) (rebrand of Calckey)
|
||||||
|
|
||||||
Here’s what that means and what you can expect.
|
|
||||||
|
|
||||||
Once the ActivityPub plugin is installed, each author’s page on your WordPress blog will become its own federated instance. In other words, if you have two authors, Jane and Bob, on your website, `example.com`, then your authors would have their own author pages at `example.com/author/jane` and `example.com/author/bob`. Each of those author pages would now be available to Mastodon users (and all other federated platform users) as a profile that can be followed. Let’s break that down further. Let’s say you have a friend on Mastodon who tells you to follow them and they give you their profile name `@janelivesheresomeofthetime@mastodon.social`. You search for her name, see her profile, and click the follow button, right? From then on, everything Jane posts on her profile shows up in your Home feed. Okay, similarly, now that Jane has installed the ActivityPub plugin on her `example.com` site, her friends can also follow her on Mastodon by searching for `@jane@example.com` and clicking the Follow button on that profile.
|
|
||||||
|
|
||||||
From now on, every blog post Jane publishes on example.com will show up on your Home feed because you follow her `@jane@example.com` profile.
|
|
||||||
Of course, if no one follows your author instance, then no one will ever see the posts - including you! So the easiest way to even know if the plugin is working is to follow your new profile yourself. If you already have a Mastodon profile, just follow your new one from there.
|
|
||||||
|
|
||||||
Some things to note:
|
Some things to note:
|
||||||
|
|
||||||
1. Many single-author blogs have chosen to turn off or redirect their author profile pages, usually via an SEO plugin like Yoast or Rank Math. This is usually done to avoid duplicate content with your blog’s home page. If your author page has been deactivated in this way, then ActivityPub won’t work for you. Instead, you can turn your author profile page back on, and then use the option in your SEO plugin to noindex the author page. This will enable the page to be live and ActivityPub will now work, but the live page won’t cause any duplicate content issues with search engines.
|
1. The blog-wide profile is only compatible with sites with rewrite rules enabled. If your site does not have rewrite rules enabled, the author-specific profiles may still work.
|
||||||
1. Once ActivityPub is installed, only new posts going forward will be available in the fediverse. Likewise, even if you’ve been using ActivityPub for a while, anyone who follows your site, will only see new posts you publish from that moment on. They will never see previously-published posts in their Home feed. This process is very similar to subscribing to a newsletter. If you subscribe to a newsletter, you will only receive future emails, but not the old archived ones. With ActivityPub, if someone follows your site, they will only receive new blog posts you publish from then on.
|
1. Many single-author blogs have chosen to turn off or redirect their author profile pages, usually via an SEO plugin like Yoast or Rank Math. This is usually done to avoid duplicate content with your blog’s home page. If your author page has been deactivated in this way, then ActivityPub author profiles won’t work for you. Instead, you can turn your author profile page back on, and then use the option in your SEO plugin to noindex the author page. This will duplicate content issues with search engines and will enable ActivityPub author profiles to work.
|
||||||
|
1. Once ActivityPub is installed, *only new posts going forward* will be available in the fediverse. Likewise, even if you’ve been using ActivityPub for a while, anyone who follows your site, will only see new posts you publish from that moment on. They will never see previously-published posts in their Home feed. This process is very similar to subscribing to a newsletter. If you subscribe to a newsletter, you will only receive future emails, but not the old archived ones. With ActivityPub, if someone follows your site, they will only receive new blog posts you publish from then on.
|
||||||
|
|
||||||
So what’s the process?
|
So what’s the process?
|
||||||
|
|
||||||
1. Install the ActivityPub plugin.
|
1. Install the ActivityPub plugin.
|
||||||
1. Go to the plugin’s settings page and adjust the settings to your liking. Click the Save button when ready.
|
1. Go to the plugin’s settings page and adjust the settings to your liking. Click the Save button when ready.
|
||||||
1. Make sure your blog’s author profile page is active.
|
1. Make sure your blog’s author profile page is active if you are using author profiles.
|
||||||
1. Go to Mastodon or any other federated platform, search for your author’s new federated profile, and follow it. Your new profile will be in the form of @yourauthorname@yourwebsite.com, so that is what you’ll search for.
|
1. Go to Mastodon or any other federated platform, and search for your profile, and follow it. Your new profile will be in the form of either `@your_username@example.com` or `@example.com@example.com`, so that is what you’ll search for.
|
||||||
1. On your blog, publish a new post.
|
1. On your blog, publish a new post.
|
||||||
1. From Mastodon, check to see if the new post appears in your Home feed.
|
1. From Mastodon, check to see if the new post appears in your Home feed.
|
||||||
|
|
||||||
|
@ -54,22 +54,14 @@ Please note that it may take up to 15 minutes or so for the new post to show up
|
||||||
|
|
||||||
### tl;dr ###
|
### tl;dr ###
|
||||||
|
|
||||||
This plugin connects your WordPress blog to popular social platforms like Mastodon, making your posts more accessible to a wider audience. Once installed, your blog's author pages can be followed by users on these platforms, allowing them to receive your new posts in their feeds.
|
This plugin connects your WordPress blog to popular social platforms like Mastodon, making your posts more accessible to a wider audience. Once installed, your blog can be followed by users on these platforms, allowing them to receive your new posts in their feeds.
|
||||||
|
|
||||||
Here's how it works:
|
|
||||||
|
|
||||||
1. Install the plugin and adjust settings as needed.
|
|
||||||
1. Ensure your blog's author profile page is active.
|
|
||||||
1. On Mastodon or other supported platforms, search for and follow your author's new profile (e.g., `@yourauthorname@yourwebsite.com`).
|
|
||||||
1. Publish a new post on your blog and check if it appears in your Mastodon feed.
|
|
||||||
|
|
||||||
Please note that it may take up to 15 minutes for a new post to appear in your feed, as messages are sent on a delay to avoid overwhelming your followers. Be patient and give it some time.
|
|
||||||
|
|
||||||
### What is the status of this plugin? ###
|
### What is the status of this plugin? ###
|
||||||
|
|
||||||
Implemented:
|
Implemented:
|
||||||
|
|
||||||
* profile pages (JSON representation)
|
* blog profile pages (JSON representation)
|
||||||
|
* author profile pages (JSON representation)
|
||||||
* custom links
|
* custom links
|
||||||
* functional inbox/outbox
|
* functional inbox/outbox
|
||||||
* follow (accept follows)
|
* follow (accept follows)
|
||||||
|
@ -79,8 +71,8 @@ Implemented:
|
||||||
|
|
||||||
To implement:
|
To implement:
|
||||||
|
|
||||||
* better configuration possibilities
|
|
||||||
* threaded comments support
|
* threaded comments support
|
||||||
|
* replace shortcodes with blocks for layout
|
||||||
|
|
||||||
### What is "ActivityPub for WordPress" ###
|
### What is "ActivityPub for WordPress" ###
|
||||||
|
|
||||||
|
@ -94,7 +86,7 @@ In order for webfinger to work, it must be mapped to the root directory of the U
|
||||||
|
|
||||||
Add the following to the .htaccess file in the root directory:
|
Add the following to the .htaccess file in the root directory:
|
||||||
|
|
||||||
RedirectMatch "^\/\.well-known/(webfinger|nodeinfo|x-nodeinfo2)(.*)$" "\/blog\/\.well-known$1$2"
|
RedirectMatch "^\/\.well-known/(webfinger|nodeinfo|x-nodeinfo2)(.*)$" /blog/.well-known/$1$2
|
||||||
|
|
||||||
Where 'blog' is the path to the subdirectory at which your blog resides.
|
Where 'blog' is the path to the subdirectory at which your blog resides.
|
||||||
|
|
||||||
|
@ -113,20 +105,36 @@ Where 'blog' is the path to the subdirectory at which your blog resides.
|
||||||
|
|
||||||
Project maintained on GitHub at [automattic/wordpress-activitypub](https://github.com/automattic/wordpress-activitypub).
|
Project maintained on GitHub at [automattic/wordpress-activitypub](https://github.com/automattic/wordpress-activitypub).
|
||||||
|
|
||||||
|
### 1.0.1 ###
|
||||||
|
|
||||||
|
* Update: improve image attachment detection using the block editor
|
||||||
|
* Update: better error code handling for API responses
|
||||||
|
* Update: use a tag stack instead of regex for protecting tags for Hashtags and @-Mentions
|
||||||
|
* Compatibility: better signature support for subpath-installations
|
||||||
|
* Compatibility: allow deactivating blocks registered by the plugin
|
||||||
|
* Compatibility: avoid Fatal Errors when using ClassicPress
|
||||||
|
* Compatibility: improve the Group-Actor to play nicely with existing implementations
|
||||||
|
* Fixed: truncate long blog titles and handles for the "Follow me" block
|
||||||
|
* Fixed: ensure that only a valid user can be selected for the "Follow me" block
|
||||||
|
* Fixed: fix a typo in a hook name
|
||||||
|
* Fixed: a problem with signatures when running WordPress in a sub-path
|
||||||
|
|
||||||
### 1.0.0 ###
|
### 1.0.0 ###
|
||||||
|
|
||||||
* Add: blog-wide Account (catchall, like `mydomain.com@mydomain.com`)
|
* Add: blog-wide Account (catchall, like `example.com@example.com`)
|
||||||
* Add: Signature Verification: https://docs.joinmastodon.org/spec/security/ .
|
* Add: a Follow Me block (help visitors to follow your Profile)
|
||||||
* Add: a Followers Block.
|
* Add: Signature Verification: https://docs.joinmastodon.org/spec/security/
|
||||||
|
* Add: a Followers Block (show off your Followers)
|
||||||
* Add: Simple caching
|
* Add: Simple caching
|
||||||
* Add: Collection endpoints for Featured Tags and Featured Posts
|
* Add: Collection endpoints for Featured Tags and Featured Posts
|
||||||
* Update: Complete rewrite of the Follower-System based on Custom Post Types.
|
* Add: Better handling of Hashtags in mobile apps
|
||||||
|
* Update: Complete rewrite of the Follower-System based on Custom Post Types
|
||||||
* Update: Improved linter (PHPCS)
|
* Update: Improved linter (PHPCS)
|
||||||
* Compatibility: Add a new conditional, `\Activitypub\is_activitypub_request()`, to allow third-party plugins to detect ActivityPub requests.
|
* Compatibility: Add a new conditional, `\Activitypub\is_activitypub_request()`, to allow third-party plugins to detect ActivityPub requests
|
||||||
* Compatibility: Add hooks to allow modifying images returned in ActivityPub requests.
|
* Compatibility: Add hooks to allow modifying images returned in ActivityPub requests
|
||||||
* Compatibility: Indicate that the plugin is compatible and has been tested with the latest version of WordPress, 6.3.
|
* Compatibility: Indicate that the plugin is compatible and has been tested with the latest version of WordPress, 6.3
|
||||||
* Compatibility: Avoid PHP notice on sites using PHP 8.2.
|
* Compatibility: Avoid PHP notice on sites using PHP 8.2
|
||||||
* Fixed: Load the plugin later in the WordPress code lifecycle to avoid errors in some requests.
|
* Fixed: Load the plugin later in the WordPress code lifecycle to avoid errors in some requests
|
||||||
* Fixed: Updating posts
|
* Fixed: Updating posts
|
||||||
* Fixed: Hashtag now support CamelCase and UTF-8
|
* Fixed: Hashtag now support CamelCase and UTF-8
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
* Plugin Name: ActivityPub
|
* Plugin Name: ActivityPub
|
||||||
* Plugin URI: https://github.com/pfefferle/wordpress-activitypub/
|
* Plugin URI: https://github.com/pfefferle/wordpress-activitypub/
|
||||||
* Description: The ActivityPub protocol is a decentralized social networking protocol based upon the ActivityStreams 2.0 data format.
|
* Description: The ActivityPub protocol is a decentralized social networking protocol based upon the ActivityStreams 2.0 data format.
|
||||||
* Version: 1.0.0
|
* Version: 1.0.1
|
||||||
* Author: Matthias Pfefferle & Automattic
|
* Author: Matthias Pfefferle & Automattic
|
||||||
* Author URI: https://automattic.com/
|
* Author URI: https://automattic.com/
|
||||||
* License: MIT
|
* License: MIT
|
||||||
|
@ -15,6 +15,8 @@
|
||||||
|
|
||||||
namespace Activitypub;
|
namespace Activitypub;
|
||||||
|
|
||||||
|
use function Activitypub\site_supports_blocks;
|
||||||
|
|
||||||
\defined( 'ACTIVITYPUB_REST_NAMESPACE' ) || \define( 'ACTIVITYPUB_REST_NAMESPACE', 'activitypub/1.0' );
|
\defined( 'ACTIVITYPUB_REST_NAMESPACE' ) || \define( 'ACTIVITYPUB_REST_NAMESPACE', 'activitypub/1.0' );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -53,10 +55,13 @@ function init() {
|
||||||
Admin::init();
|
Admin::init();
|
||||||
Hashtag::init();
|
Hashtag::init();
|
||||||
Shortcodes::init();
|
Shortcodes::init();
|
||||||
Blocks::init();
|
|
||||||
Mention::init();
|
Mention::init();
|
||||||
Health_Check::init();
|
Health_Check::init();
|
||||||
Scheduler::init();
|
Scheduler::init();
|
||||||
|
|
||||||
|
if ( site_supports_blocks() ) {
|
||||||
|
Blocks::init();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
\add_action( 'init', __NAMESPACE__ . '\init' );
|
\add_action( 'init', __NAMESPACE__ . '\init' );
|
||||||
|
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 3.3 KiB |
BIN
assets/img/wp-logo.png
Normal file
BIN
assets/img/wp-logo.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 12 KiB |
47
build/follow-me/block.json
Normal file
47
build/follow-me/block.json
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
{
|
||||||
|
"$schema": "https://schemas.wp.org/trunk/block.json",
|
||||||
|
"name": "activitypub/follow-me",
|
||||||
|
"apiVersion": 3,
|
||||||
|
"version": "1.0.0",
|
||||||
|
"title": "Follow me on the Fediverse",
|
||||||
|
"category": "widgets",
|
||||||
|
"description": "Display your Fediverse profile so that visitors can follow you.",
|
||||||
|
"textdomain": "activitypub",
|
||||||
|
"icon": "groups",
|
||||||
|
"supports": {
|
||||||
|
"html": false,
|
||||||
|
"color": {
|
||||||
|
"gradients": true,
|
||||||
|
"link": true,
|
||||||
|
"__experimentalDefaultControls": {
|
||||||
|
"background": true,
|
||||||
|
"text": true,
|
||||||
|
"link": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"__experimentalBorder": {
|
||||||
|
"radius": true,
|
||||||
|
"width": true,
|
||||||
|
"color": true,
|
||||||
|
"style": true
|
||||||
|
},
|
||||||
|
"typography": {
|
||||||
|
"fontSize": true,
|
||||||
|
"__experimentalDefaultControls": {
|
||||||
|
"fontSize": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"attributes": {
|
||||||
|
"selectedUser": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "site"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"editorScript": "file:./index.js",
|
||||||
|
"viewScript": "file:./view.js",
|
||||||
|
"style": [
|
||||||
|
"file:./style-index.css",
|
||||||
|
"wp-components"
|
||||||
|
]
|
||||||
|
}
|
1
build/follow-me/index.asset.php
Normal file
1
build/follow-me/index.asset.php
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<?php return array('dependencies' => array('wp-api-fetch', 'wp-block-editor', 'wp-blocks', 'wp-components', 'wp-compose', 'wp-data', 'wp-element', 'wp-i18n', 'wp-primitives'), 'version' => '5e11f2fcd4dc9ac6d7ee');
|
1
build/follow-me/index.js
Normal file
1
build/follow-me/index.js
Normal file
File diff suppressed because one or more lines are too long
1
build/follow-me/style-index.css
Normal file
1
build/follow-me/style-index.css
Normal file
|
@ -0,0 +1 @@
|
||||||
|
.activitypub-follow-me-block-wrapper .activitypub-profile,.editor-styles-wrapper .activitypub-profile{align-items:self-start;display:flex;padding:1rem}.activitypub-follow-me-block-wrapper .activitypub-profile .activitypub-profile__avatar,.editor-styles-wrapper .activitypub-profile .activitypub-profile__avatar{border-radius:50%;height:75px;margin-right:1rem;width:75px}.activitypub-follow-me-block-wrapper .activitypub-profile .activitypub-profile__content,.editor-styles-wrapper .activitypub-profile .activitypub-profile__content{flex:1;min-width:0}.activitypub-follow-me-block-wrapper .activitypub-profile .activitypub-profile__handle,.activitypub-follow-me-block-wrapper .activitypub-profile .activitypub-profile__name,.editor-styles-wrapper .activitypub-profile .activitypub-profile__handle,.editor-styles-wrapper .activitypub-profile .activitypub-profile__name{line-height:1.2;margin:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.activitypub-follow-me-block-wrapper .activitypub-profile .activitypub-profile__name,.editor-styles-wrapper .activitypub-profile .activitypub-profile__name{font-size:1.25em}.activitypub-follow-me-block-wrapper .activitypub-profile .activitypub-profile__follow,.editor-styles-wrapper .activitypub-profile .activitypub-profile__follow{align-self:center;background-color:var(--wp--preset--color--black);color:var(--wp--preset--color--white);margin-left:1rem}.activitypub-follow-me__dialog{max-width:30em}.activitypub-follow-me__dialog h4{line-height:1;margin:0}.activitypub-follow-me__dialog .apmfd__section{margin-bottom:2em}.activitypub-follow-me__dialog .apfmd-description{font-size:var(--wp--preset--font-size--normal,.75rem);margin:.33em 0 1em}.activitypub-follow-me__dialog .apfmd__button-group{display:flex;justify-content:flex-end}.activitypub-follow-me__dialog .apfmd__button-group svg{height:21px;margin-right:.5em;width:21px}.activitypub-follow-me__dialog .apfmd__button-group input{flex:1;padding-left:1em;padding-right:1em}
|
1
build/follow-me/view.asset.php
Normal file
1
build/follow-me/view.asset.php
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<?php return array('dependencies' => array('wp-api-fetch', 'wp-components', 'wp-compose', 'wp-dom-ready', 'wp-element', 'wp-i18n', 'wp-primitives'), 'version' => '06a922cf2e58c94b6431');
|
1
build/follow-me/view.js
Normal file
1
build/follow-me/view.js
Normal file
File diff suppressed because one or more lines are too long
|
@ -1 +1 @@
|
||||||
<?php return array('dependencies' => array('react', 'wp-api-fetch', 'wp-block-editor', 'wp-blocks', 'wp-components', 'wp-data', 'wp-element', 'wp-i18n', 'wp-primitives', 'wp-url'), 'version' => '284dffd27ea0242085be');
|
<?php return array('dependencies' => array('react', 'wp-api-fetch', 'wp-block-editor', 'wp-blocks', 'wp-components', 'wp-data', 'wp-element', 'wp-i18n', 'wp-primitives', 'wp-url'), 'version' => '423cdeefa42e7c7cabd1');
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
(()=>{var e={184:(e,t)=>{var a;!function(){"use strict";var n={}.hasOwnProperty;function l(){for(var e=[],t=0;t<arguments.length;t++){var a=arguments[t];if(a){var r=typeof a;if("string"===r||"number"===r)e.push(a);else if(Array.isArray(a)){if(a.length){var o=l.apply(null,a);o&&e.push(o)}}else if("object"===r){if(a.toString!==Object.prototype.toString&&!a.toString.toString().includes("[native code]")){e.push(a.toString());continue}for(var i in a)n.call(a,i)&&a[i]&&e.push(i)}}}return e.join(" ")}e.exports?(l.default=l,e.exports=l):void 0===(a=function(){return l}.apply(t,[]))||(e.exports=a)}()}},t={};function a(n){var l=t[n];if(void 0!==l)return l.exports;var r=t[n]={exports:{}};return e[n](r,r.exports,a),r.exports}a.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return a.d(t,{a:t}),t},a.d=(e,t)=>{for(var n in t)a.o(t,n)&&!a.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:t[n]})},a.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),(()=>{"use strict";const e=window.wp.blocks,t=window.wp.element,n=window.wp.primitives,l=(0,t.createElement)(n.SVG,{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24"},(0,t.createElement)(n.Path,{d:"M15.5 9.5a1 1 0 100-2 1 1 0 000 2zm0 1.5a2.5 2.5 0 100-5 2.5 2.5 0 000 5zm-2.25 6v-2a2.75 2.75 0 00-2.75-2.75h-4A2.75 2.75 0 003.75 15v2h1.5v-2c0-.69.56-1.25 1.25-1.25h4c.69 0 1.25.56 1.25 1.25v2h1.5zm7-2v2h-1.5v-2c0-.69-.56-1.25-1.25-1.25H15v-1.5h2.5A2.75 2.75 0 0120.25 15zM9.5 8.5a1 1 0 11-2 0 1 1 0 012 0zm1.5 0a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z",fillRule:"evenodd"}));function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var a=arguments[t];for(var n in a)Object.prototype.hasOwnProperty.call(a,n)&&(e[n]=a[n])}return e},r.apply(this,arguments)}const o=window.wp.components,i=window.wp.data,c=window.wp.blockEditor,s=window.wp.i18n,p=window.React,u=window.wp.apiFetch;var v=a.n(u);const m=window.wp.url;var b=a(184),w=a.n(b);function d(e){let{active:a,children:n,page:l,pageClick:r,className:o}=e;const i=w()("wp-block activitypub-pager",o,{current:a});return(0,t.createElement)("a",{className:i,onClick:e=>{e.preventDefault(),!a&&r(l)}},n)}const g={outlined:"outlined",minimal:"minimal"};function f(e){let{compact:a,nextLabel:n,page:l,pageClick:r,perPage:o,prevLabel:i,total:c,variant:s=g.outlined}=e;const p=((e,t)=>{let a=[1,e-2,e-1,e,e+1,e+2,t];a.sort(((e,t)=>e-t)),a=a.filter(((e,a,n)=>e>=1&&e<=t&&n.lastIndexOf(e)===a));for(let e=a.length-2;e>=0;e--)a[e]===a[e+1]&&a.splice(e+1,1);return a})(l,Math.ceil(c/o)),u=w()("alignwide wp-block-query-pagination is-content-justification-space-between is-layout-flex wp-block-query-pagination-is-layout-flex",`is-${s}`,{"is-compact":a});return(0,t.createElement)("nav",{className:u},i&&(0,t.createElement)(d,{key:"prev",page:l-1,pageClick:r,active:1===l,"aria-label":i,className:"wp-block-query-pagination-previous block-editor-block-list__block"},i),!a&&(0,t.createElement)("div",{className:"block-editor-block-list__block wp-block wp-block-query-pagination-numbers"},p.map((e=>(0,t.createElement)(d,{key:e,page:e,pageClick:r,active:e===l,className:"page-numbers"},e)))),n&&(0,t.createElement)(d,{key:"next",page:l+1,pageClick:r,active:l===Math.ceil(c/o),"aria-label":n,className:"wp-block-query-pagination-next block-editor-block-list__block"},n))}const{namespace:y}=window._activityPubOptions;function h(e){let{selectedUser:a,per_page:n,order:l,title:o,page:i,setPage:c,className:u="",followLinks:b=!0}=e;const w="site"===a?0:a,[d,g]=(0,p.useState)([]),[h,_]=(0,p.useState)(0),[E,x]=(0,p.useState)(0),[C,S]=function(){const[e,t]=(0,p.useState)(1);return[e,t]}(),O=i||C,N=c||S,P=(0,t.createInterpolateElement)(/* translators: arrow for previous followers link */
|
(()=>{var e={184:(e,t)=>{var a;!function(){"use strict";var n={}.hasOwnProperty;function l(){for(var e=[],t=0;t<arguments.length;t++){var a=arguments[t];if(a){var r=typeof a;if("string"===r||"number"===r)e.push(a);else if(Array.isArray(a)){if(a.length){var o=l.apply(null,a);o&&e.push(o)}}else if("object"===r){if(a.toString!==Object.prototype.toString&&!a.toString.toString().includes("[native code]")){e.push(a.toString());continue}for(var i in a)n.call(a,i)&&a[i]&&e.push(i)}}}return e.join(" ")}e.exports?(l.default=l,e.exports=l):void 0===(a=function(){return l}.apply(t,[]))||(e.exports=a)}()}},t={};function a(n){var l=t[n];if(void 0!==l)return l.exports;var r=t[n]={exports:{}};return e[n](r,r.exports,a),r.exports}a.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return a.d(t,{a:t}),t},a.d=(e,t)=>{for(var n in t)a.o(t,n)&&!a.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:t[n]})},a.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),(()=>{"use strict";const e=window.wp.blocks,t=window.wp.element,n=window.wp.primitives,l=(0,t.createElement)(n.SVG,{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24"},(0,t.createElement)(n.Path,{d:"M15.5 9.5a1 1 0 100-2 1 1 0 000 2zm0 1.5a2.5 2.5 0 100-5 2.5 2.5 0 000 5zm-2.25 6v-2a2.75 2.75 0 00-2.75-2.75h-4A2.75 2.75 0 003.75 15v2h1.5v-2c0-.69.56-1.25 1.25-1.25h4c.69 0 1.25.56 1.25 1.25v2h1.5zm7-2v2h-1.5v-2c0-.69-.56-1.25-1.25-1.25H15v-1.5h2.5A2.75 2.75 0 0120.25 15zM9.5 8.5a1 1 0 11-2 0 1 1 0 012 0zm1.5 0a2.5 2.5 0 11-5 0 2.5 2.5 0 015 0z",fillRule:"evenodd"}));function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var a=arguments[t];for(var n in a)Object.prototype.hasOwnProperty.call(a,n)&&(e[n]=a[n])}return e},r.apply(this,arguments)}const o=window.wp.components,i=window.wp.blockEditor,c=window.wp.i18n,s=window.React,p=window.wp.apiFetch;var u=a.n(p);const v=window.wp.url;var m=a(184),b=a.n(m);function w(e){let{active:a,children:n,page:l,pageClick:r,className:o}=e;const i=b()("wp-block activitypub-pager",o,{current:a});return(0,t.createElement)("a",{className:i,onClick:e=>{e.preventDefault(),!a&&r(l)}},n)}const d={outlined:"outlined",minimal:"minimal"};function g(e){let{compact:a,nextLabel:n,page:l,pageClick:r,perPage:o,prevLabel:i,total:c,variant:s=d.outlined}=e;const p=((e,t)=>{let a=[1,e-2,e-1,e,e+1,e+2,t];a.sort(((e,t)=>e-t)),a=a.filter(((e,a,n)=>e>=1&&e<=t&&n.lastIndexOf(e)===a));for(let e=a.length-2;e>=0;e--)a[e]===a[e+1]&&a.splice(e+1,1);return a})(l,Math.ceil(c/o)),u=b()("alignwide wp-block-query-pagination is-content-justification-space-between is-layout-flex wp-block-query-pagination-is-layout-flex",`is-${s}`,{"is-compact":a});return(0,t.createElement)("nav",{className:u},i&&(0,t.createElement)(w,{key:"prev",page:l-1,pageClick:r,active:1===l,"aria-label":i,className:"wp-block-query-pagination-previous block-editor-block-list__block"},i),!a&&(0,t.createElement)("div",{className:"block-editor-block-list__block wp-block wp-block-query-pagination-numbers"},p.map((e=>(0,t.createElement)(w,{key:e,page:e,pageClick:r,active:e===l,className:"page-numbers"},e)))),n&&(0,t.createElement)(w,{key:"next",page:l+1,pageClick:r,active:l===Math.ceil(c/o),"aria-label":n,className:"wp-block-query-pagination-next block-editor-block-list__block"},n))}const{namespace:f}=window._activityPubOptions;function y(e){let{selectedUser:a,per_page:n,order:l,title:o,page:i,setPage:p,className:m="",followLinks:b=!0}=e;const w="site"===a?0:a,[d,y]=(0,s.useState)([]),[k,E]=(0,s.useState)(0),[_,x]=(0,s.useState)(0),[C,S]=function(){const[e,t]=(0,s.useState)(1);return[e,t]}(),O=i||C,N=p||S,P=(0,t.createInterpolateElement)(/* translators: arrow for previous followers link */
|
||||||
(0,s.__)("<span>←</span> Less","activitypub"),{span:(0,t.createElement)("span",{class:"wp-block-query-pagination-previous-arrow is-arrow-arrow","aria-hidden":"true"})}),L=(0,t.createInterpolateElement)(/* translators: arrow for next followers link */
|
(0,c.__)("<span>←</span> Less","activitypub"),{span:(0,t.createElement)("span",{class:"wp-block-query-pagination-previous-arrow is-arrow-arrow","aria-hidden":"true"})}),L=(0,t.createInterpolateElement)(/* translators: arrow for next followers link */
|
||||||
(0,s.__)("More <span>→</span>","activitypub"),{span:(0,t.createElement)("span",{class:"wp-block-query-pagination-next-arrow is-arrow-arrow","aria-hidden":"true"})});return(0,p.useEffect)((()=>{const e=function(e,t,a,n){const l=`/${y}/users/${e}/followers`,r={per_page:t,order:a,page:n,context:"full"};return(0,m.addQueryArgs)(l,r)}(w,n,l,O);v()({path:e}).then((e=>{_(Math.ceil(e.totalItems/n)),x(e.totalItems),g(e.orderedItems)})).catch((e=>console.error(e)))}),[w,n,l,O]),(0,t.createElement)("div",{className:"activitypub-follower-block "+u},(0,t.createElement)("h3",null,o),(0,t.createElement)("ul",null,d&&d.map((e=>(0,t.createElement)("li",{key:e.url},(0,t.createElement)(k,r({},e,{followLinks:b})))))),h>1&&(0,t.createElement)(f,{page:O,perPage:n,total:E,pageClick:N,nextLabel:L,prevLabel:P,compact:"is-style-compact"===u}))}function k(e){let{name:a,icon:n,url:l,preferredUsername:i,followLinks:c=!0}=e;const s=`@${i}`,p={};return c||(p.onClick=e=>e.preventDefault()),(0,t.createElement)(o.ExternalLink,r({className:"activitypub-link",href:l,title:s},p),(0,t.createElement)("img",{width:"40",height:"40",src:n.url,class:"avatar activitypub-avatar"}),(0,t.createElement)("span",{class:"activitypub-actor"},(0,t.createElement)("strong",{className:"activitypub-name"},a),(0,t.createElement)("span",{class:"sep"},"/"),(0,t.createElement)("span",{class:"activitypub-handle"},s)))}const _=window._activityPubOptions?.enabled;(0,e.registerBlockType)("activitypub/followers",{edit:function(e){let{attributes:a,setAttributes:n}=e;const{order:l,per_page:p,selectedUser:u,title:v}=a,m=(0,c.useBlockProps)(),[b,w]=(0,t.useState)(1),d=[{label:(0,s.__)("New to old","activitypub"),value:"desc"},{label:(0,s.__)("Old to new","activitypub"),value:"asc"}],g=function(){const e=_?.users?(0,i.useSelect)((e=>e("core").getUsers({who:"authors"}))):[];return(0,t.useMemo)((()=>{if(!e)return[];const t=_?.site?[{label:(0,s.__)("Whole Site","activitypub"),value:"site"}]:[];return e.reduce(((e,t)=>(e.push({label:t.name,value:t.id}),e)),t)}),[e])}(),f=e=>t=>{w(1),n({[e]:t})};return(0,t.createElement)("div",m,(0,t.createElement)(c.InspectorControls,{key:"setting"},(0,t.createElement)(o.PanelBody,{title:(0,s.__)("Followers Options","activitypub")},(0,t.createElement)(o.TextControl,{label:(0,s.__)("Title","activitypub"),help:(0,s.__)("Title to display above the list of followers. Blank for none.","activitypub"),value:v,onChange:e=>n({title:e})}),(0,t.createElement)(o.SelectControl,{label:(0,s.__)("Select User","activitypub"),value:u,options:g,onChange:f("selectedUser")}),(0,t.createElement)(o.SelectControl,{label:(0,s.__)("Sort","activitypub"),value:l,options:d,onChange:f("order")}),(0,t.createElement)(o.RangeControl,{label:(0,s.__)("Number of Followers","activitypub"),value:p,onChange:f("per_page"),min:1,max:10}))),(0,t.createElement)(h,r({},a,{page:b,setPage:w,followLinks:!1})))},save:()=>null,icon:l})})()})();
|
(0,c.__)("More <span>→</span>","activitypub"),{span:(0,t.createElement)("span",{class:"wp-block-query-pagination-next-arrow is-arrow-arrow","aria-hidden":"true"})});return(0,s.useEffect)((()=>{const e=function(e,t,a,n){const l=`/${f}/users/${e}/followers`,r={per_page:t,order:a,page:n,context:"full"};return(0,v.addQueryArgs)(l,r)}(w,n,l,O);u()({path:e}).then((e=>{E(Math.ceil(e.totalItems/n)),x(e.totalItems),y(e.orderedItems)})).catch((()=>{}))}),[w,n,l,O]),(0,t.createElement)("div",{className:"activitypub-follower-block "+m},(0,t.createElement)("h3",null,o),(0,t.createElement)("ul",null,d&&d.map((e=>(0,t.createElement)("li",{key:e.url},(0,t.createElement)(h,r({},e,{followLinks:b})))))),k>1&&(0,t.createElement)(g,{page:O,perPage:n,total:_,pageClick:N,nextLabel:L,prevLabel:P,compact:"is-style-compact"===m}))}function h(e){let{name:a,icon:n,url:l,preferredUsername:i,followLinks:c=!0}=e;const s=`@${i}`,p={};return c||(p.onClick=e=>e.preventDefault()),(0,t.createElement)(o.ExternalLink,r({className:"activitypub-link",href:l,title:s},p),(0,t.createElement)("img",{width:"40",height:"40",src:n.url,class:"avatar activitypub-avatar"}),(0,t.createElement)("span",{class:"activitypub-actor"},(0,t.createElement)("strong",{className:"activitypub-name"},a),(0,t.createElement)("span",{class:"sep"},"/"),(0,t.createElement)("span",{class:"activitypub-handle"},s)))}const k=window.wp.data,E=window._activityPubOptions?.enabled;(0,e.registerBlockType)("activitypub/followers",{edit:function(e){let{attributes:a,setAttributes:n}=e;const{order:l,per_page:s,selectedUser:p,title:u}=a,v=(0,i.useBlockProps)(),[m,b]=(0,t.useState)(1),w=[{label:(0,c.__)("New to old","activitypub"),value:"desc"},{label:(0,c.__)("Old to new","activitypub"),value:"asc"}],d=function(){const e=E?.users?(0,k.useSelect)((e=>e("core").getUsers({who:"authors"}))):[];return(0,t.useMemo)((()=>{if(!e)return[];const t=E?.site?[{label:(0,c.__)("Whole Site","activitypub"),value:"site"}]:[];return e.reduce(((e,t)=>(e.push({label:t.name,value:`${t.id}`}),e)),t)}),[e])}(),g=e=>t=>{b(1),n({[e]:t})};return(0,t.useEffect)((()=>{d.length&&(d.find((e=>{let{value:t}=e;return t===p}))||n({selectedUser:d[0].value}))}),[p,d]),(0,t.createElement)("div",v,(0,t.createElement)(i.InspectorControls,{key:"setting"},(0,t.createElement)(o.PanelBody,{title:(0,c.__)("Followers Options","activitypub")},(0,t.createElement)(o.TextControl,{label:(0,c.__)("Title","activitypub"),help:(0,c.__)("Title to display above the list of followers. Blank for none.","activitypub"),value:u,onChange:e=>n({title:e})}),(0,t.createElement)(o.SelectControl,{label:(0,c.__)("Select User","activitypub"),value:p,options:d,onChange:g("selectedUser")}),(0,t.createElement)(o.SelectControl,{label:(0,c.__)("Sort","activitypub"),value:l,options:w,onChange:g("order")}),(0,t.createElement)(o.RangeControl,{label:(0,c.__)("Number of Followers","activitypub"),value:s,onChange:g("per_page"),min:1,max:10}))),(0,t.createElement)(y,r({},a,{page:m,setPage:b,followLinks:!1})))},save:()=>null,icon:l})})()})();
|
|
@ -1 +1 @@
|
||||||
<?php return array('dependencies' => array('react', 'wp-api-fetch', 'wp-components', 'wp-dom-ready', 'wp-element', 'wp-i18n', 'wp-url'), 'version' => 'd645fd4aa610b479e8f4');
|
<?php return array('dependencies' => array('react', 'wp-api-fetch', 'wp-components', 'wp-dom-ready', 'wp-element', 'wp-i18n', 'wp-url'), 'version' => '6384e801c2802d2fecee');
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
(()=>{var e,t={189:(e,t,a)=>{"use strict";const r=window.wp.element;function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var a=arguments[t];for(var r in a)Object.prototype.hasOwnProperty.call(a,r)&&(e[r]=a[r])}return e},n.apply(this,arguments)}const l=window.React,o=window.wp.apiFetch;var i=a.n(o);const c=window.wp.url,s=window.wp.i18n;var p=a(184),u=a.n(p);function m(e){let{active:t,children:a,page:n,pageClick:l,className:o}=e;const i=u()("wp-block activitypub-pager",o,{current:t});return(0,r.createElement)("a",{className:i,onClick:e=>{e.preventDefault(),!t&&l(n)}},a)}const v={outlined:"outlined",minimal:"minimal"};function b(e){let{compact:t,nextLabel:a,page:n,pageClick:l,perPage:o,prevLabel:i,total:c,variant:s=v.outlined}=e;const p=((e,t)=>{let a=[1,e-2,e-1,e,e+1,e+2,t];a.sort(((e,t)=>e-t)),a=a.filter(((e,a,r)=>e>=1&&e<=t&&r.lastIndexOf(e)===a));for(let e=a.length-2;e>=0;e--)a[e]===a[e+1]&&a.splice(e+1,1);return a})(n,Math.ceil(c/o)),b=u()("alignwide wp-block-query-pagination is-content-justification-space-between is-layout-flex wp-block-query-pagination-is-layout-flex",`is-${s}`,{"is-compact":t});return(0,r.createElement)("nav",{className:b},i&&(0,r.createElement)(m,{key:"prev",page:n-1,pageClick:l,active:1===n,"aria-label":i,className:"wp-block-query-pagination-previous block-editor-block-list__block"},i),!t&&(0,r.createElement)("div",{className:"block-editor-block-list__block wp-block wp-block-query-pagination-numbers"},p.map((e=>(0,r.createElement)(m,{key:e,page:e,pageClick:l,active:e===n,className:"page-numbers"},e)))),a&&(0,r.createElement)(m,{key:"next",page:n+1,pageClick:l,active:n===Math.ceil(c/o),"aria-label":a,className:"wp-block-query-pagination-next block-editor-block-list__block"},a))}const f=window.wp.components,{namespace:d}=window._activityPubOptions;function w(e){let{selectedUser:t,per_page:a,order:o,title:p,page:u,setPage:m,className:v="",followLinks:f=!0}=e;const w="site"===t?0:t,[y,k]=(0,l.useState)([]),[h,E]=(0,l.useState)(0),[O,x]=(0,l.useState)(0),[_,N]=function(){const[e,t]=(0,l.useState)(1);return[e,t]}(),j=u||_,S=m||N,C=(0,r.createInterpolateElement)(/* translators: arrow for previous followers link */
|
(()=>{var e,t={189:(e,t,a)=>{"use strict";const r=window.wp.element;function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var a=arguments[t];for(var r in a)Object.prototype.hasOwnProperty.call(a,r)&&(e[r]=a[r])}return e},n.apply(this,arguments)}const l=window.React,i=window.wp.apiFetch;var o=a.n(i);const c=window.wp.url,s=window.wp.i18n;var p=a(184),u=a.n(p);function m(e){let{active:t,children:a,page:n,pageClick:l,className:i}=e;const o=u()("wp-block activitypub-pager",i,{current:t});return(0,r.createElement)("a",{className:o,onClick:e=>{e.preventDefault(),!t&&l(n)}},a)}const v={outlined:"outlined",minimal:"minimal"};function b(e){let{compact:t,nextLabel:a,page:n,pageClick:l,perPage:i,prevLabel:o,total:c,variant:s=v.outlined}=e;const p=((e,t)=>{let a=[1,e-2,e-1,e,e+1,e+2,t];a.sort(((e,t)=>e-t)),a=a.filter(((e,a,r)=>e>=1&&e<=t&&r.lastIndexOf(e)===a));for(let e=a.length-2;e>=0;e--)a[e]===a[e+1]&&a.splice(e+1,1);return a})(n,Math.ceil(c/i)),b=u()("alignwide wp-block-query-pagination is-content-justification-space-between is-layout-flex wp-block-query-pagination-is-layout-flex",`is-${s}`,{"is-compact":t});return(0,r.createElement)("nav",{className:b},o&&(0,r.createElement)(m,{key:"prev",page:n-1,pageClick:l,active:1===n,"aria-label":o,className:"wp-block-query-pagination-previous block-editor-block-list__block"},o),!t&&(0,r.createElement)("div",{className:"block-editor-block-list__block wp-block wp-block-query-pagination-numbers"},p.map((e=>(0,r.createElement)(m,{key:e,page:e,pageClick:l,active:e===n,className:"page-numbers"},e)))),a&&(0,r.createElement)(m,{key:"next",page:n+1,pageClick:l,active:n===Math.ceil(c/i),"aria-label":a,className:"wp-block-query-pagination-next block-editor-block-list__block"},a))}const f=window.wp.components,{namespace:d}=window._activityPubOptions;function w(e){let{selectedUser:t,per_page:a,order:i,title:p,page:u,setPage:m,className:v="",followLinks:f=!0}=e;const w="site"===t?0:t,[y,k]=(0,l.useState)([]),[h,E]=(0,l.useState)(0),[O,x]=(0,l.useState)(0),[_,N]=function(){const[e,t]=(0,l.useState)(1);return[e,t]}(),j=u||_,S=m||N,C=(0,r.createInterpolateElement)(/* translators: arrow for previous followers link */
|
||||||
(0,s.__)("<span>←</span> Less","activitypub"),{span:(0,r.createElement)("span",{class:"wp-block-query-pagination-previous-arrow is-arrow-arrow","aria-hidden":"true"})}),L=(0,r.createInterpolateElement)(/* translators: arrow for next followers link */
|
(0,s.__)("<span>←</span> Less","activitypub"),{span:(0,r.createElement)("span",{class:"wp-block-query-pagination-previous-arrow is-arrow-arrow","aria-hidden":"true"})}),L=(0,r.createInterpolateElement)(/* translators: arrow for next followers link */
|
||||||
(0,s.__)("More <span>→</span>","activitypub"),{span:(0,r.createElement)("span",{class:"wp-block-query-pagination-next-arrow is-arrow-arrow","aria-hidden":"true"})});return(0,l.useEffect)((()=>{const e=function(e,t,a,r){const n=`/${d}/users/${e}/followers`,l={per_page:t,order:a,page:r,context:"full"};return(0,c.addQueryArgs)(n,l)}(w,a,o,j);i()({path:e}).then((e=>{E(Math.ceil(e.totalItems/a)),x(e.totalItems),k(e.orderedItems)})).catch((e=>console.error(e)))}),[w,a,o,j]),(0,r.createElement)("div",{className:"activitypub-follower-block "+v},(0,r.createElement)("h3",null,p),(0,r.createElement)("ul",null,y&&y.map((e=>(0,r.createElement)("li",{key:e.url},(0,r.createElement)(g,n({},e,{followLinks:f})))))),h>1&&(0,r.createElement)(b,{page:j,perPage:a,total:O,pageClick:S,nextLabel:L,prevLabel:C,compact:"is-style-compact"===v}))}function g(e){let{name:t,icon:a,url:l,preferredUsername:o,followLinks:i=!0}=e;const c=`@${o}`,s={};return i||(s.onClick=e=>e.preventDefault()),(0,r.createElement)(f.ExternalLink,n({className:"activitypub-link",href:l,title:c},s),(0,r.createElement)("img",{width:"40",height:"40",src:a.url,class:"avatar activitypub-avatar"}),(0,r.createElement)("span",{class:"activitypub-actor"},(0,r.createElement)("strong",{className:"activitypub-name"},t),(0,r.createElement)("span",{class:"sep"},"/"),(0,r.createElement)("span",{class:"activitypub-handle"},c)))}const y=window.wp.domReady;a.n(y)()((()=>{[].forEach.call(document.querySelectorAll(".activitypub-follower-block"),(e=>{const t=JSON.parse(e.dataset.attrs);(0,r.render)((0,r.createElement)(w,t),e)}))}))},184:(e,t)=>{var a;!function(){"use strict";var r={}.hasOwnProperty;function n(){for(var e=[],t=0;t<arguments.length;t++){var a=arguments[t];if(a){var l=typeof a;if("string"===l||"number"===l)e.push(a);else if(Array.isArray(a)){if(a.length){var o=n.apply(null,a);o&&e.push(o)}}else if("object"===l){if(a.toString!==Object.prototype.toString&&!a.toString.toString().includes("[native code]")){e.push(a.toString());continue}for(var i in a)r.call(a,i)&&a[i]&&e.push(i)}}}return e.join(" ")}e.exports?(n.default=n,e.exports=n):void 0===(a=function(){return n}.apply(t,[]))||(e.exports=a)}()}},a={};function r(e){var n=a[e];if(void 0!==n)return n.exports;var l=a[e]={exports:{}};return t[e](l,l.exports,r),l.exports}r.m=t,e=[],r.O=(t,a,n,l)=>{if(!a){var o=1/0;for(p=0;p<e.length;p++){for(var[a,n,l]=e[p],i=!0,c=0;c<a.length;c++)(!1&l||o>=l)&&Object.keys(r.O).every((e=>r.O[e](a[c])))?a.splice(c--,1):(i=!1,l<o&&(o=l));if(i){e.splice(p--,1);var s=n();void 0!==s&&(t=s)}}return t}l=l||0;for(var p=e.length;p>0&&e[p-1][2]>l;p--)e[p]=e[p-1];e[p]=[a,n,l]},r.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return r.d(t,{a:t}),t},r.d=(e,t)=>{for(var a in t)r.o(t,a)&&!r.o(e,a)&&Object.defineProperty(e,a,{enumerable:!0,get:t[a]})},r.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),(()=>{var e={638:0,962:0};r.O.j=t=>0===e[t];var t=(t,a)=>{var n,l,[o,i,c]=a,s=0;if(o.some((t=>0!==e[t]))){for(n in i)r.o(i,n)&&(r.m[n]=i[n]);if(c)var p=c(r)}for(t&&t(a);s<o.length;s++)l=o[s],r.o(e,l)&&e[l]&&e[l][0](),e[l]=0;return r.O(p)},a=globalThis.webpackChunkwordpress_activitypub=globalThis.webpackChunkwordpress_activitypub||[];a.forEach(t.bind(null,0)),a.push=t.bind(null,a.push.bind(a))})();var n=r.O(void 0,[962],(()=>r(189)));n=r.O(n)})();
|
(0,s.__)("More <span>→</span>","activitypub"),{span:(0,r.createElement)("span",{class:"wp-block-query-pagination-next-arrow is-arrow-arrow","aria-hidden":"true"})});return(0,l.useEffect)((()=>{const e=function(e,t,a,r){const n=`/${d}/users/${e}/followers`,l={per_page:t,order:a,page:r,context:"full"};return(0,c.addQueryArgs)(n,l)}(w,a,i,j);o()({path:e}).then((e=>{E(Math.ceil(e.totalItems/a)),x(e.totalItems),k(e.orderedItems)})).catch((()=>{}))}),[w,a,i,j]),(0,r.createElement)("div",{className:"activitypub-follower-block "+v},(0,r.createElement)("h3",null,p),(0,r.createElement)("ul",null,y&&y.map((e=>(0,r.createElement)("li",{key:e.url},(0,r.createElement)(g,n({},e,{followLinks:f})))))),h>1&&(0,r.createElement)(b,{page:j,perPage:a,total:O,pageClick:S,nextLabel:L,prevLabel:C,compact:"is-style-compact"===v}))}function g(e){let{name:t,icon:a,url:l,preferredUsername:i,followLinks:o=!0}=e;const c=`@${i}`,s={};return o||(s.onClick=e=>e.preventDefault()),(0,r.createElement)(f.ExternalLink,n({className:"activitypub-link",href:l,title:c},s),(0,r.createElement)("img",{width:"40",height:"40",src:a.url,class:"avatar activitypub-avatar"}),(0,r.createElement)("span",{class:"activitypub-actor"},(0,r.createElement)("strong",{className:"activitypub-name"},t),(0,r.createElement)("span",{class:"sep"},"/"),(0,r.createElement)("span",{class:"activitypub-handle"},c)))}const y=window.wp.domReady;a.n(y)()((()=>{[].forEach.call(document.querySelectorAll(".activitypub-follower-block"),(e=>{const t=JSON.parse(e.dataset.attrs);(0,r.render)((0,r.createElement)(w,t),e)}))}))},184:(e,t)=>{var a;!function(){"use strict";var r={}.hasOwnProperty;function n(){for(var e=[],t=0;t<arguments.length;t++){var a=arguments[t];if(a){var l=typeof a;if("string"===l||"number"===l)e.push(a);else if(Array.isArray(a)){if(a.length){var i=n.apply(null,a);i&&e.push(i)}}else if("object"===l){if(a.toString!==Object.prototype.toString&&!a.toString.toString().includes("[native code]")){e.push(a.toString());continue}for(var o in a)r.call(a,o)&&a[o]&&e.push(o)}}}return e.join(" ")}e.exports?(n.default=n,e.exports=n):void 0===(a=function(){return n}.apply(t,[]))||(e.exports=a)}()}},a={};function r(e){var n=a[e];if(void 0!==n)return n.exports;var l=a[e]={exports:{}};return t[e](l,l.exports,r),l.exports}r.m=t,e=[],r.O=(t,a,n,l)=>{if(!a){var i=1/0;for(p=0;p<e.length;p++){for(var[a,n,l]=e[p],o=!0,c=0;c<a.length;c++)(!1&l||i>=l)&&Object.keys(r.O).every((e=>r.O[e](a[c])))?a.splice(c--,1):(o=!1,l<i&&(i=l));if(o){e.splice(p--,1);var s=n();void 0!==s&&(t=s)}}return t}l=l||0;for(var p=e.length;p>0&&e[p-1][2]>l;p--)e[p]=e[p-1];e[p]=[a,n,l]},r.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return r.d(t,{a:t}),t},r.d=(e,t)=>{for(var a in t)r.o(t,a)&&!r.o(e,a)&&Object.defineProperty(e,a,{enumerable:!0,get:t[a]})},r.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),(()=>{var e={638:0,962:0};r.O.j=t=>0===e[t];var t=(t,a)=>{var n,l,[i,o,c]=a,s=0;if(i.some((t=>0!==e[t]))){for(n in o)r.o(o,n)&&(r.m[n]=o[n]);if(c)var p=c(r)}for(t&&t(a);s<i.length;s++)l=i[s],r.o(e,l)&&e[l]&&e[l][0](),e[l]=0;return r.O(p)},a=globalThis.webpackChunkwordpress_activitypub=globalThis.webpackChunkwordpress_activitypub||[];a.forEach(t.bind(null,0)),a.push=t.bind(null,a.push.bind(a))})();var n=r.O(void 0,[962],(()=>r(189)));n=r.O(n)})();
|
|
@ -41,6 +41,9 @@
|
||||||
],
|
],
|
||||||
"lint": [
|
"lint": [
|
||||||
"vendor/bin/phpcs -n -q"
|
"vendor/bin/phpcs -n -q"
|
||||||
|
],
|
||||||
|
"lint:fix": [
|
||||||
|
"vendor/bin/phpcbf"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,9 @@ class Activity extends Base_Object {
|
||||||
'schema' => 'http://schema.org#',
|
'schema' => 'http://schema.org#',
|
||||||
'pt' => 'https://joinpeertube.org/ns#',
|
'pt' => 'https://joinpeertube.org/ns#',
|
||||||
'toot' => 'http://joinmastodon.org/ns#',
|
'toot' => 'http://joinmastodon.org/ns#',
|
||||||
|
'webfinger' => 'https://webfinger.net/#',
|
||||||
'litepub' => 'http://litepub.social/ns#',
|
'litepub' => 'http://litepub.social/ns#',
|
||||||
|
'lemmy' => 'https://join-lemmy.org/ns#',
|
||||||
'value' => 'schema:value',
|
'value' => 'schema:value',
|
||||||
'Hashtag' => 'as:Hashtag',
|
'Hashtag' => 'as:Hashtag',
|
||||||
'featured' => array(
|
'featured' => array(
|
||||||
|
@ -37,8 +39,19 @@ class Activity extends Base_Object {
|
||||||
'@id' => 'toot:featuredTags',
|
'@id' => 'toot:featuredTags',
|
||||||
'@type' => '@id',
|
'@type' => '@id',
|
||||||
),
|
),
|
||||||
|
'alsoKnownAs' => array(
|
||||||
|
'@id' => 'as:alsoKnownAs',
|
||||||
|
'@type' => '@id',
|
||||||
|
),
|
||||||
|
'moderators' => array(
|
||||||
|
'@id' => 'lemmy:moderators',
|
||||||
|
'@type' => '@id',
|
||||||
|
),
|
||||||
|
'postingRestrictedToMods' => 'lemmy:postingRestrictedToMods',
|
||||||
'discoverable' => 'toot:discoverable',
|
'discoverable' => 'toot:discoverable',
|
||||||
|
'indexable' => 'toot:indexable',
|
||||||
'sensitive' => 'as:sensitive',
|
'sensitive' => 'as:sensitive',
|
||||||
|
'resource' => 'webfinger:resource',
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -450,7 +450,7 @@ class Base_Object {
|
||||||
|
|
||||||
if ( \strncasecmp( $method, 'get', 3 ) === 0 ) {
|
if ( \strncasecmp( $method, 'get', 3 ) === 0 ) {
|
||||||
if ( ! $this->has( $var ) ) {
|
if ( ! $this->has( $var ) ) {
|
||||||
return new WP_Error( 'invalid_key', 'Invalid key' );
|
return new WP_Error( 'invalid_key', __( 'Invalid key', 'activitypub' ), array( 'status' => 404 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->$var;
|
return $this->$var;
|
||||||
|
@ -492,7 +492,7 @@ class Base_Object {
|
||||||
*/
|
*/
|
||||||
public function get( $key ) {
|
public function get( $key ) {
|
||||||
if ( ! $this->has( $key ) ) {
|
if ( ! $this->has( $key ) ) {
|
||||||
return new WP_Error( 'invalid_key', 'Invalid key' );
|
return new WP_Error( 'invalid_key', __( 'Invalid key', 'activitypub' ), array( 'status' => 404 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
return call_user_func( array( $this, 'get_' . $key ) );
|
return call_user_func( array( $this, 'get_' . $key ) );
|
||||||
|
@ -519,7 +519,7 @@ class Base_Object {
|
||||||
*/
|
*/
|
||||||
public function set( $key, $value ) {
|
public function set( $key, $value ) {
|
||||||
if ( ! $this->has( $key ) ) {
|
if ( ! $this->has( $key ) ) {
|
||||||
return new WP_Error( 'invalid_key', 'Invalid key' );
|
return new WP_Error( 'invalid_key', __( 'Invalid key', 'activitypub' ), array( 'status' => 404 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->$key = $value;
|
$this->$key = $value;
|
||||||
|
@ -537,7 +537,7 @@ class Base_Object {
|
||||||
*/
|
*/
|
||||||
public function add( $key, $value ) {
|
public function add( $key, $value ) {
|
||||||
if ( ! $this->has( $key ) ) {
|
if ( ! $this->has( $key ) ) {
|
||||||
return new WP_Error( 'invalid_key', 'Invalid key' );
|
return new WP_Error( 'invalid_key', __( 'Invalid key', 'activitypub' ), array( 'status' => 404 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! isset( $this->$key ) ) {
|
if ( ! isset( $this->$key ) ) {
|
||||||
|
@ -562,6 +562,10 @@ class Base_Object {
|
||||||
public static function init_from_json( $json ) {
|
public static function init_from_json( $json ) {
|
||||||
$array = \json_decode( $json, true );
|
$array = \json_decode( $json, true );
|
||||||
|
|
||||||
|
if ( ! is_array( $array ) ) {
|
||||||
|
$array = array();
|
||||||
|
}
|
||||||
|
|
||||||
return self::init_from_array( $array );
|
return self::init_from_array( $array );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -573,6 +577,10 @@ class Base_Object {
|
||||||
* @return \Activitypub\Activity\Base_Object An Object built from the JSON string.
|
* @return \Activitypub\Activity\Base_Object An Object built from the JSON string.
|
||||||
*/
|
*/
|
||||||
public static function init_from_array( $array ) {
|
public static function init_from_array( $array ) {
|
||||||
|
if ( ! is_array( $array ) ) {
|
||||||
|
return new WP_Error( 'invalid_array', __( 'Invalid array', 'activitypub' ), array( 'status' => 404 ) );
|
||||||
|
}
|
||||||
|
|
||||||
$object = new static();
|
$object = new static();
|
||||||
|
|
||||||
foreach ( $array as $key => $value ) {
|
foreach ( $array as $key => $value ) {
|
||||||
|
@ -636,7 +644,7 @@ class Base_Object {
|
||||||
}
|
}
|
||||||
|
|
||||||
// if value is still empty, ignore it for the array and continue.
|
// if value is still empty, ignore it for the array and continue.
|
||||||
if ( $value ) {
|
if ( isset( $value ) ) {
|
||||||
$array[ snake_to_camel_case( $key ) ] = $value;
|
$array[ snake_to_camel_case( $key ) ] = $value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,6 +17,7 @@ class Activitypub {
|
||||||
\add_filter( 'template_include', array( self::class, 'render_json_template' ), 99 );
|
\add_filter( 'template_include', array( self::class, 'render_json_template' ), 99 );
|
||||||
\add_filter( 'query_vars', array( self::class, 'add_query_vars' ) );
|
\add_filter( 'query_vars', array( self::class, 'add_query_vars' ) );
|
||||||
\add_filter( 'pre_get_avatar_data', array( self::class, 'pre_get_avatar_data' ), 11, 2 );
|
\add_filter( 'pre_get_avatar_data', array( self::class, 'pre_get_avatar_data' ), 11, 2 );
|
||||||
|
\add_filter( 'get_comment_link', array( self::class, 'remote_comment_link' ), 11, 3 );
|
||||||
|
|
||||||
// Add support for ActivityPub to custom post types
|
// Add support for ActivityPub to custom post types
|
||||||
$post_types = \get_option( 'activitypub_support_post_types', array( 'post', 'page' ) ) ? \get_option( 'activitypub_support_post_types', array( 'post', 'page' ) ) : array();
|
$post_types = \get_option( 'activitypub_support_post_types', array( 'post', 'page' ) ) ? \get_option( 'activitypub_support_post_types', array( 'post', 'page' ) ) : array();
|
||||||
|
@ -188,6 +189,22 @@ class Activitypub {
|
||||||
return \get_comment_meta( $comment->comment_ID, 'avatar_url', true );
|
return \get_comment_meta( $comment->comment_ID, 'avatar_url', true );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Link remote comments to source url.
|
||||||
|
*
|
||||||
|
* @param string $comment_link
|
||||||
|
* @param object|WP_Comment $comment
|
||||||
|
*
|
||||||
|
* @return string $url
|
||||||
|
*/
|
||||||
|
public static function remote_comment_link( $comment_link, $comment ) {
|
||||||
|
$remote_comment_link = get_comment_meta( $comment->comment_ID, 'source_url', true );
|
||||||
|
if ( $remote_comment_link ) {
|
||||||
|
$comment_link = esc_url( $remote_comment_link );
|
||||||
|
}
|
||||||
|
return $comment_link;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Store permalink in meta, to send delete Activity.
|
* Store permalink in meta, to send delete Activity.
|
||||||
*
|
*
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
<?php
|
<?php
|
||||||
namespace Activitypub;
|
namespace Activitypub;
|
||||||
|
|
||||||
|
use WP_User_Query;
|
||||||
|
use Activitypub\Model\Blog_User;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ActivityPub Admin Class
|
* ActivityPub Admin Class
|
||||||
*
|
*
|
||||||
|
@ -169,7 +172,7 @@ class Admin {
|
||||||
'type' => 'string',
|
'type' => 'string',
|
||||||
'description' => \esc_html__( 'The Identifier of the Blog-User', 'activitypub' ),
|
'description' => \esc_html__( 'The Identifier of the Blog-User', 'activitypub' ),
|
||||||
'show_in_rest' => true,
|
'show_in_rest' => true,
|
||||||
'default' => \Activitypub\Model\Blog_User::get_default_username(),
|
'default' => Blog_User::get_default_username(),
|
||||||
'sanitize_callback' => function( $value ) {
|
'sanitize_callback' => function( $value ) {
|
||||||
// hack to allow dots in the username
|
// hack to allow dots in the username
|
||||||
$parts = explode( '.', $value );
|
$parts = explode( '.', $value );
|
||||||
|
@ -179,7 +182,31 @@ class Admin {
|
||||||
$sanitized[] = \sanitize_title( $part );
|
$sanitized[] = \sanitize_title( $part );
|
||||||
}
|
}
|
||||||
|
|
||||||
return implode( '.', $sanitized );
|
$sanitized = implode( '.', $sanitized );
|
||||||
|
|
||||||
|
// check for login or nicename.
|
||||||
|
$user = new WP_User_Query(
|
||||||
|
array(
|
||||||
|
'search' => $sanitized,
|
||||||
|
'search_columns' => array( 'user_login', 'user_nicename' ),
|
||||||
|
'number' => 1,
|
||||||
|
'hide_empty' => true,
|
||||||
|
'fields' => 'ID',
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
if ( $user->results ) {
|
||||||
|
add_settings_error(
|
||||||
|
'activitypub_blog_user_identifier',
|
||||||
|
'activitypub_blog_user_identifier',
|
||||||
|
\esc_html__( 'You cannot use an existing author\'s name for the blog profile ID.', 'activitypub' ),
|
||||||
|
'error'
|
||||||
|
);
|
||||||
|
|
||||||
|
return Blog_User::get_default_username();
|
||||||
|
}
|
||||||
|
|
||||||
|
return $sanitized;
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
|
@ -13,7 +13,9 @@ class Blocks {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function add_data() {
|
public static function add_data() {
|
||||||
$handle = is_admin() ? 'activitypub-followers-editor-script' : 'activitypub-followers-view-script';
|
$context = is_admin() ? 'editor' : 'view';
|
||||||
|
$followers_handle = 'activitypub-followers-' . $context . '-script';
|
||||||
|
$follow_me_handle = 'activitypub-follow-me-' . $context . '-script';
|
||||||
$data = array(
|
$data = array(
|
||||||
'namespace' => ACTIVITYPUB_REST_NAMESPACE,
|
'namespace' => ACTIVITYPUB_REST_NAMESPACE,
|
||||||
'enabled' => array(
|
'enabled' => array(
|
||||||
|
@ -22,7 +24,8 @@ class Blocks {
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
$js = sprintf( 'var _activityPubOptions = %s;', wp_json_encode( $data ) );
|
$js = sprintf( 'var _activityPubOptions = %s;', wp_json_encode( $data ) );
|
||||||
\wp_add_inline_script( $handle, $js, 'before' );
|
\wp_add_inline_script( $followers_handle, $js, 'before' );
|
||||||
|
\wp_add_inline_script( $follow_me_handle, $js, 'before' );
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function register_blocks() {
|
public static function register_blocks() {
|
||||||
|
@ -32,6 +35,12 @@ class Blocks {
|
||||||
'render_callback' => array( self::class, 'render_follower_block' ),
|
'render_callback' => array( self::class, 'render_follower_block' ),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
\register_block_type_from_metadata(
|
||||||
|
ACTIVITYPUB_PLUGIN_DIR . '/build/follow-me',
|
||||||
|
array(
|
||||||
|
'render_callback' => array( self::class, 'render_follow_me_block' ),
|
||||||
|
)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static function get_user_id( $user_string ) {
|
private static function get_user_id( $user_string ) {
|
||||||
|
@ -42,7 +51,24 @@ class Blocks {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function render_follower_block( $attrs, $content, $block ) {
|
/**
|
||||||
|
* Render the follow me block.
|
||||||
|
* @param array $attrs The block attributes.
|
||||||
|
* @return string The HTML to render.
|
||||||
|
*/
|
||||||
|
public static function render_follow_me_block( $attrs ) {
|
||||||
|
$wrapper_attributes = get_block_wrapper_attributes(
|
||||||
|
array(
|
||||||
|
'aria-label' => __( 'Follow me on the Fediverse', 'activitypub' ),
|
||||||
|
'class' => 'activitypub-follow-me-block-wrapper',
|
||||||
|
'data-attrs' => wp_json_encode( $attrs ),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
// todo: render more than an empty div?
|
||||||
|
return '<div ' . $wrapper_attributes . '></div>';
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function render_follower_block( $attrs ) {
|
||||||
$followee_user_id = self::get_user_id( $attrs['selectedUser'] );
|
$followee_user_id = self::get_user_id( $attrs['selectedUser'] );
|
||||||
$per_page = absint( $attrs['per_page'] );
|
$per_page = absint( $attrs['per_page'] );
|
||||||
$followers = Followers::get_followers( $followee_user_id, $per_page );
|
$followers = Followers::get_followers( $followee_user_id, $per_page );
|
||||||
|
@ -55,7 +81,7 @@ class Blocks {
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
$html = '<div class="activitypub-follower-block" ' . $wrapper_attributes . '>';
|
$html = '<div ' . $wrapper_attributes . '>';
|
||||||
if ( $title ) {
|
if ( $title ) {
|
||||||
$html .= '<h3>' . $title . '</h3>';
|
$html .= '<h3>' . $title . '</h3>';
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,38 +43,56 @@ class Hashtag {
|
||||||
* @return string the filtered post-content
|
* @return string the filtered post-content
|
||||||
*/
|
*/
|
||||||
public static function the_content( $the_content ) {
|
public static function the_content( $the_content ) {
|
||||||
$protected_tags = array();
|
$tag_stack = array();
|
||||||
$protect = function( $m ) use ( &$protected_tags ) {
|
$protected_tags = array(
|
||||||
$c = \wp_rand( 100000, 999999 );
|
'pre',
|
||||||
$protect = '!#!#PROTECT' . $c . '#!#!';
|
'code',
|
||||||
while ( isset( $protected_tags[ $protect ] ) ) {
|
'textarea',
|
||||||
$c = \wp_rand( 100000, 999999 );
|
'style',
|
||||||
$protect = '!#!#PROTECT' . $c . '#!#!';
|
'a',
|
||||||
|
);
|
||||||
|
$content_with_links = '';
|
||||||
|
$in_protected_tag = false;
|
||||||
|
foreach ( wp_html_split( $the_content ) as $chunk ) {
|
||||||
|
if ( preg_match( '#^<!--[\s\S]*-->$#i', $chunk, $m ) ) {
|
||||||
|
$content_with_links .= $chunk;
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
$protected_tags[ $protect ] = $m[0];
|
|
||||||
return $protect;
|
|
||||||
};
|
|
||||||
$the_content = preg_replace_callback(
|
|
||||||
'#<!\[CDATA\[.*?\]\]>#is',
|
|
||||||
$protect,
|
|
||||||
$the_content
|
|
||||||
);
|
|
||||||
$the_content = preg_replace_callback(
|
|
||||||
'#<(pre|code|textarea|style)\b[^>]*>.*?</\1[^>]*>#is',
|
|
||||||
$protect,
|
|
||||||
$the_content
|
|
||||||
);
|
|
||||||
$the_content = preg_replace_callback(
|
|
||||||
'#<[^>]+>#i',
|
|
||||||
$protect,
|
|
||||||
$the_content
|
|
||||||
);
|
|
||||||
|
|
||||||
$the_content = \preg_replace_callback( '/' . ACTIVITYPUB_HASHTAGS_REGEXP . '/i', array( '\Activitypub\Hashtag', 'replace_with_links' ), $the_content );
|
if ( preg_match( '#^<(/)?([a-z-]+)\b[^>]*>$#i', $chunk, $m ) ) {
|
||||||
|
$tag = strtolower( $m[2] );
|
||||||
|
if ( '/' === $m[1] ) {
|
||||||
|
// Closing tag.
|
||||||
|
$i = array_search( $tag, $tag_stack );
|
||||||
|
// We can only remove the tag from the stack if it is in the stack.
|
||||||
|
if ( false !== $i ) {
|
||||||
|
$tag_stack = array_slice( $tag_stack, 0, $i );
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Opening tag, add it to the stack.
|
||||||
|
$tag_stack[] = $tag;
|
||||||
|
}
|
||||||
|
|
||||||
$the_content = str_replace( array_reverse( array_keys( $protected_tags ) ), array_reverse( array_values( $protected_tags ) ), $the_content );
|
// If we're in a protected tag, the tag_stack contains at least one protected tag string.
|
||||||
|
// The protected tag state can only change when we encounter a start or end tag.
|
||||||
|
$in_protected_tag = array_intersect( $tag_stack, $protected_tags );
|
||||||
|
|
||||||
return $the_content;
|
// Never inspect tags.
|
||||||
|
$content_with_links .= $chunk;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( $in_protected_tag ) {
|
||||||
|
// Don't inspect a chunk inside an inspected tag.
|
||||||
|
$content_with_links .= $chunk;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only reachable when there is no protected tag in the stack.
|
||||||
|
$content_with_links .= \preg_replace_callback( '/' . ACTIVITYPUB_HASHTAGS_REGEXP . '/i', array( '\Activitypub\Hashtag', 'replace_with_links' ), $chunk );
|
||||||
|
}
|
||||||
|
|
||||||
|
return $content_with_links;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -89,7 +107,7 @@ class Hashtag {
|
||||||
|
|
||||||
if ( $tag_object ) {
|
if ( $tag_object ) {
|
||||||
$link = \get_term_link( $tag_object, 'post_tag' );
|
$link = \get_term_link( $tag_object, 'post_tag' );
|
||||||
return \sprintf( '<a rel="tag" class="u-tag u-category" href="%s">#%s</a>', $link, $tag );
|
return \sprintf( '<a rel="tag" class="hashtag u-tag u-category" href="%s">#%s</a>', $link, $tag );
|
||||||
}
|
}
|
||||||
|
|
||||||
return '#' . $tag;
|
return '#' . $tag;
|
||||||
|
|
|
@ -1,6 +1,12 @@
|
||||||
<?php
|
<?php
|
||||||
namespace Activitypub;
|
namespace Activitypub;
|
||||||
|
|
||||||
|
use WP_Error;
|
||||||
|
use Activitypub\Webfinger;
|
||||||
|
|
||||||
|
use function Activitypub\get_plugin_version;
|
||||||
|
use function Activitypub\get_webfinger_resource;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ActivityPub Health_Check Class
|
* ActivityPub Health_Check Class
|
||||||
*
|
*
|
||||||
|
@ -29,6 +35,11 @@ class Health_Check {
|
||||||
'test' => array( self::class, 'test_webfinger' ),
|
'test' => array( self::class, 'test_webfinger' ),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
$tests['direct']['activitypub_test_system_cron'] = array(
|
||||||
|
'label' => __( 'System Cron Test', 'activitypub' ),
|
||||||
|
'test' => array( self::class, 'test_system_cron' ),
|
||||||
|
);
|
||||||
|
|
||||||
return $tests;
|
return $tests;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -70,6 +81,49 @@ class Health_Check {
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* System Cron tests
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public static function test_system_cron() {
|
||||||
|
$result = array(
|
||||||
|
'label' => \__( 'System Task Scheduler configured', 'activitypub' ),
|
||||||
|
'status' => 'good',
|
||||||
|
'badge' => array(
|
||||||
|
'label' => \__( 'ActivityPub', 'activitypub' ),
|
||||||
|
'color' => 'green',
|
||||||
|
),
|
||||||
|
'description' => \sprintf(
|
||||||
|
'<p>%s</p>',
|
||||||
|
\esc_html__( 'You seem to use the System Task Scheduler to process WP_Cron tasks.', 'activitypub' )
|
||||||
|
),
|
||||||
|
'actions' => '',
|
||||||
|
'test' => 'test_system_cron',
|
||||||
|
);
|
||||||
|
|
||||||
|
if ( defined( 'DISABLE_WP_CRON' ) && DISABLE_WP_CRON ) {
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
$result['status'] = 'recommended';
|
||||||
|
$result['label'] = \__( 'System Task Scheduler not configured', 'activitypub' );
|
||||||
|
$result['badge']['color'] = 'orange';
|
||||||
|
$result['description'] = \sprintf(
|
||||||
|
'<p>%s</p>',
|
||||||
|
\__( 'Enhance your WordPress site’s performance and mitigate potential heavy loads caused by plugins like ActivityPub by setting up a system cron job to run WP Cron. This ensures scheduled tasks are executed consistently and reduces the reliance on website traffic for trigger events.', 'activitypub' )
|
||||||
|
);
|
||||||
|
$result['actions'] .= sprintf(
|
||||||
|
'<p><a href="%s" target="_blank" rel="noopener">%s<span class="screen-reader-text"> %s</span><span aria-hidden="true" class="dashicons dashicons-external"></span></a></p>',
|
||||||
|
__( 'https://developer.wordpress.org/plugins/cron/hooking-wp-cron-into-the-system-task-scheduler/', 'activitypub' ),
|
||||||
|
__( 'Learn how to hook the WP-Cron into the System Task Scheduler.', 'activitypub' ),
|
||||||
|
/* translators: Hidden accessibility text. */
|
||||||
|
__( '(opens in a new tab)', 'activitypub' )
|
||||||
|
);
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WebFinger tests
|
* WebFinger tests
|
||||||
*
|
*
|
||||||
|
@ -111,7 +165,7 @@ class Health_Check {
|
||||||
/**
|
/**
|
||||||
* Check if `author_posts_url` is accessible and that request returns correct JSON
|
* Check if `author_posts_url` is accessible and that request returns correct JSON
|
||||||
*
|
*
|
||||||
* @return boolean|\WP_Error
|
* @return boolean|WP_Error
|
||||||
*/
|
*/
|
||||||
public static function is_author_url_accessible() {
|
public static function is_author_url_accessible() {
|
||||||
$user = \wp_get_current_user();
|
$user = \wp_get_current_user();
|
||||||
|
@ -120,7 +174,7 @@ class Health_Check {
|
||||||
|
|
||||||
// check for "author" in URL
|
// check for "author" in URL
|
||||||
if ( $author_url !== $reference_author_url ) {
|
if ( $author_url !== $reference_author_url ) {
|
||||||
return new \WP_Error(
|
return new WP_Error(
|
||||||
'author_url_not_accessible',
|
'author_url_not_accessible',
|
||||||
\sprintf(
|
\sprintf(
|
||||||
// translators: %s: Author URL
|
// translators: %s: Author URL
|
||||||
|
@ -143,7 +197,7 @@ class Health_Check {
|
||||||
);
|
);
|
||||||
|
|
||||||
if ( \is_wp_error( $response ) ) {
|
if ( \is_wp_error( $response ) ) {
|
||||||
return new \WP_Error(
|
return new WP_Error(
|
||||||
'author_url_not_accessible',
|
'author_url_not_accessible',
|
||||||
\sprintf(
|
\sprintf(
|
||||||
// translators: %s: Author URL
|
// translators: %s: Author URL
|
||||||
|
@ -160,7 +214,7 @@ class Health_Check {
|
||||||
|
|
||||||
// check for redirects
|
// check for redirects
|
||||||
if ( \in_array( $response_code, array( 301, 302, 307, 308 ), true ) ) {
|
if ( \in_array( $response_code, array( 301, 302, 307, 308 ), true ) ) {
|
||||||
return new \WP_Error(
|
return new WP_Error(
|
||||||
'author_url_not_accessible',
|
'author_url_not_accessible',
|
||||||
\sprintf(
|
\sprintf(
|
||||||
// translators: %s: Author URL
|
// translators: %s: Author URL
|
||||||
|
@ -177,7 +231,7 @@ class Health_Check {
|
||||||
$body = \wp_remote_retrieve_body( $response );
|
$body = \wp_remote_retrieve_body( $response );
|
||||||
|
|
||||||
if ( ! \is_string( $body ) || ! \is_array( \json_decode( $body, true ) ) ) {
|
if ( ! \is_string( $body ) || ! \is_array( \json_decode( $body, true ) ) ) {
|
||||||
return new \WP_Error(
|
return new WP_Error(
|
||||||
'author_url_not_accessible',
|
'author_url_not_accessible',
|
||||||
\sprintf(
|
\sprintf(
|
||||||
// translators: %s: Author URL
|
// translators: %s: Author URL
|
||||||
|
@ -196,13 +250,13 @@ class Health_Check {
|
||||||
/**
|
/**
|
||||||
* Check if WebFinger endpoint is accessible and profile request returns correct JSON
|
* Check if WebFinger endpoint is accessible and profile request returns correct JSON
|
||||||
*
|
*
|
||||||
* @return boolean|\WP_Error
|
* @return boolean|WP_Error
|
||||||
*/
|
*/
|
||||||
public static function is_webfinger_endpoint_accessible() {
|
public static function is_webfinger_endpoint_accessible() {
|
||||||
$user = \wp_get_current_user();
|
$user = \wp_get_current_user();
|
||||||
$account = \Activitypub\get_webfinger_resource( $user->ID );
|
$account = get_webfinger_resource( $user->ID );
|
||||||
|
|
||||||
$url = \Activitypub\Webfinger::resolve( $account );
|
$url = Webfinger::resolve( $account );
|
||||||
if ( \is_wp_error( $url ) ) {
|
if ( \is_wp_error( $url ) ) {
|
||||||
$allowed = array( 'code' => array() );
|
$allowed = array( 'code' => array() );
|
||||||
$not_accessible = wp_kses(
|
$not_accessible = wp_kses(
|
||||||
|
@ -237,7 +291,7 @@ class Health_Check {
|
||||||
if ( isset( $health_messages[ $url->get_error_code() ] ) ) {
|
if ( isset( $health_messages[ $url->get_error_code() ] ) ) {
|
||||||
$message = $health_messages[ $url->get_error_code() ];
|
$message = $health_messages[ $url->get_error_code() ];
|
||||||
}
|
}
|
||||||
return new \WP_Error(
|
return new WP_Error(
|
||||||
$url->get_error_code(),
|
$url->get_error_code(),
|
||||||
$message,
|
$message,
|
||||||
$url->get_error_data()
|
$url->get_error_data()
|
||||||
|
@ -291,7 +345,7 @@ class Health_Check {
|
||||||
'fields' => array(
|
'fields' => array(
|
||||||
'webfinger' => array(
|
'webfinger' => array(
|
||||||
'label' => __( 'WebFinger Resource', 'activitypub' ),
|
'label' => __( 'WebFinger Resource', 'activitypub' ),
|
||||||
'value' => \Activitypub\Webfinger::get_user_resource( wp_get_current_user()->ID ),
|
'value' => Webfinger::get_user_resource( wp_get_current_user()->ID ),
|
||||||
'private' => true,
|
'private' => true,
|
||||||
),
|
),
|
||||||
'author_url' => array(
|
'author_url' => array(
|
||||||
|
@ -299,6 +353,11 @@ class Health_Check {
|
||||||
'value' => get_author_posts_url( wp_get_current_user()->ID ),
|
'value' => get_author_posts_url( wp_get_current_user()->ID ),
|
||||||
'private' => true,
|
'private' => true,
|
||||||
),
|
),
|
||||||
|
'plugin_version' => array(
|
||||||
|
'label' => __( 'Plugin Version', 'activitypub' ),
|
||||||
|
'value' => get_plugin_version(),
|
||||||
|
'private' => true,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -53,7 +53,7 @@ class Http {
|
||||||
$code = \wp_remote_retrieve_response_code( $response );
|
$code = \wp_remote_retrieve_response_code( $response );
|
||||||
|
|
||||||
if ( $code >= 400 ) {
|
if ( $code >= 400 ) {
|
||||||
$response = new WP_Error( $code, __( 'Failed HTTP Request', 'activitypub' ) );
|
$response = new WP_Error( $code, __( 'Failed HTTP Request', 'activitypub' ), array( 'status' => $code ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
\do_action( 'activitypub_safe_remote_post_response', $response, $url, $body, $user_id );
|
\do_action( 'activitypub_safe_remote_post_response', $response, $url, $body, $user_id );
|
||||||
|
@ -101,7 +101,7 @@ class Http {
|
||||||
$code = \wp_remote_retrieve_response_code( $response );
|
$code = \wp_remote_retrieve_response_code( $response );
|
||||||
|
|
||||||
if ( $code >= 400 ) {
|
if ( $code >= 400 ) {
|
||||||
$response = new WP_Error( $code, __( 'Failed HTTP Request', 'activitypub' ) );
|
$response = new WP_Error( $code, __( 'Failed HTTP Request', 'activitypub' ), array( 'status' => $code ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
\do_action( 'activitypub_safe_remote_get_response', $response, $url );
|
\do_action( 'activitypub_safe_remote_get_response', $response, $url );
|
||||||
|
|
|
@ -25,43 +25,56 @@ class Mention {
|
||||||
* @return string the filtered post-content
|
* @return string the filtered post-content
|
||||||
*/
|
*/
|
||||||
public static function the_content( $the_content ) {
|
public static function the_content( $the_content ) {
|
||||||
$protected_tags = array();
|
$tag_stack = array();
|
||||||
$protect = function( $m ) use ( &$protected_tags ) {
|
$protected_tags = array(
|
||||||
$c = \wp_rand( 100000, 999999 );
|
'pre',
|
||||||
$protect = '!#!#PROTECT' . $c . '#!#!';
|
'code',
|
||||||
while ( isset( $protected_tags[ $protect ] ) ) {
|
'textarea',
|
||||||
$c = \wp_rand( 100000, 999999 );
|
'style',
|
||||||
$protect = '!#!#PROTECT' . $c . '#!#!';
|
'a',
|
||||||
|
);
|
||||||
|
$content_with_links = '';
|
||||||
|
$in_protected_tag = false;
|
||||||
|
foreach ( wp_html_split( $the_content ) as $chunk ) {
|
||||||
|
if ( preg_match( '#^<!--[\s\S]*-->$#i', $chunk, $m ) ) {
|
||||||
|
$content_with_links .= $chunk;
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
$protected_tags[ $protect ] = $m[0];
|
|
||||||
return $protect;
|
|
||||||
};
|
|
||||||
$the_content = preg_replace_callback(
|
|
||||||
'#<!\[CDATA\[.*?\]\]>#is',
|
|
||||||
$protect,
|
|
||||||
$the_content
|
|
||||||
);
|
|
||||||
$the_content = preg_replace_callback(
|
|
||||||
'#<(pre|code|textarea|style)\b[^>]*>.*?</\1[^>]*>#is',
|
|
||||||
$protect,
|
|
||||||
$the_content
|
|
||||||
);
|
|
||||||
$the_content = preg_replace_callback(
|
|
||||||
'#<a.*?href=[^>]+>.*?</a>#i',
|
|
||||||
$protect,
|
|
||||||
$the_content
|
|
||||||
);
|
|
||||||
|
|
||||||
$the_content = preg_replace_callback(
|
if ( preg_match( '#^<(/)?([a-z-]+)\b[^>]*>$#i', $chunk, $m ) ) {
|
||||||
'#<img.*?[^>]+>#i',
|
$tag = strtolower( $m[2] );
|
||||||
$protect,
|
if ( '/' === $m[1] ) {
|
||||||
$the_content
|
// Closing tag.
|
||||||
);
|
$i = array_search( $tag, $tag_stack );
|
||||||
|
// We can only remove the tag from the stack if it is in the stack.
|
||||||
|
if ( false !== $i ) {
|
||||||
|
$tag_stack = array_slice( $tag_stack, 0, $i );
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Opening tag, add it to the stack.
|
||||||
|
$tag_stack[] = $tag;
|
||||||
|
}
|
||||||
|
|
||||||
$the_content = \preg_replace_callback( '/@' . ACTIVITYPUB_USERNAME_REGEXP . '/', array( self::class, 'replace_with_links' ), $the_content );
|
// If we're in a protected tag, the tag_stack contains at least one protected tag string.
|
||||||
$the_content = \str_replace( array_reverse( array_keys( $protected_tags ) ), array_reverse( array_values( $protected_tags ) ), $the_content );
|
// The protected tag state can only change when we encounter a start or end tag.
|
||||||
|
$in_protected_tag = array_intersect( $tag_stack, $protected_tags );
|
||||||
|
|
||||||
return $the_content;
|
// Never inspect tags.
|
||||||
|
$content_with_links .= $chunk;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( $in_protected_tag ) {
|
||||||
|
// Don't inspect a chunk inside an inspected tag.
|
||||||
|
$content_with_links .= $chunk;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only reachable when there is no protected tag in the stack.
|
||||||
|
$content_with_links .= \preg_replace_callback( '/@' . ACTIVITYPUB_USERNAME_REGEXP . '/', array( self::class, 'replace_with_links' ), $chunk );
|
||||||
|
}
|
||||||
|
|
||||||
|
return $content_with_links;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -74,7 +87,7 @@ class Mention {
|
||||||
public static function replace_with_links( $result ) {
|
public static function replace_with_links( $result ) {
|
||||||
$metadata = get_remote_metadata_by_actor( $result[0] );
|
$metadata = get_remote_metadata_by_actor( $result[0] );
|
||||||
|
|
||||||
if ( ! is_wp_error( $metadata ) && ! empty( $metadata['url'] ) ) {
|
if ( ! empty( $metadata ) && ! is_wp_error( $metadata ) && ! empty( $metadata['url'] ) ) {
|
||||||
$username = ltrim( $result[0], '@' );
|
$username = ltrim( $result[0], '@' );
|
||||||
if ( ! empty( $metadata['name'] ) ) {
|
if ( ! empty( $metadata['name'] ) ) {
|
||||||
$username = $metadata['name'];
|
$username = $metadata['name'];
|
||||||
|
|
|
@ -105,10 +105,16 @@ class Scheduler {
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public static function update_followers() {
|
public static function update_followers() {
|
||||||
$followers = Followers::get_outdated_followers();
|
$number = 5;
|
||||||
|
|
||||||
|
if ( defined( 'DISABLE_WP_CRON' ) && DISABLE_WP_CRON ) {
|
||||||
|
$number = 50;
|
||||||
|
}
|
||||||
|
|
||||||
|
$followers = Followers::get_outdated_followers( $number );
|
||||||
|
|
||||||
foreach ( $followers as $follower ) {
|
foreach ( $followers as $follower ) {
|
||||||
$meta = get_remote_metadata_by_actor( $follower->get_url(), true );
|
$meta = get_remote_metadata_by_actor( $follower->get_url(), false );
|
||||||
|
|
||||||
if ( empty( $meta ) || ! is_array( $meta ) || is_wp_error( $meta ) ) {
|
if ( empty( $meta ) || ! is_array( $meta ) || is_wp_error( $meta ) ) {
|
||||||
Followers::add_error( $follower->get__id(), $meta );
|
Followers::add_error( $follower->get__id(), $meta );
|
||||||
|
@ -125,10 +131,16 @@ class Scheduler {
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
public static function cleanup_followers() {
|
public static function cleanup_followers() {
|
||||||
$followers = Followers::get_faulty_followers();
|
$number = 5;
|
||||||
|
|
||||||
|
if ( defined( 'DISABLE_WP_CRON' ) && DISABLE_WP_CRON ) {
|
||||||
|
$number = 50;
|
||||||
|
}
|
||||||
|
|
||||||
|
$followers = Followers::get_faulty_followers( $number );
|
||||||
|
|
||||||
foreach ( $followers as $follower ) {
|
foreach ( $followers as $follower ) {
|
||||||
$meta = get_remote_metadata_by_actor( $follower->get_url(), true );
|
$meta = get_remote_metadata_by_actor( $follower->get_url(), false );
|
||||||
|
|
||||||
if ( is_tombstone( $meta ) ) {
|
if ( is_tombstone( $meta ) ) {
|
||||||
$follower->delete();
|
$follower->delete();
|
||||||
|
|
|
@ -46,7 +46,7 @@ class Shortcodes {
|
||||||
|
|
||||||
foreach ( $tags as $tag ) {
|
foreach ( $tags as $tag ) {
|
||||||
$hash_tags[] = \sprintf(
|
$hash_tags[] = \sprintf(
|
||||||
'<a rel="tag" class="u-tag u-category" href="%s">%s</a>',
|
'<a rel="tag" class="hashtag u-tag u-category" href="%s">%s</a>',
|
||||||
\esc_url( \get_tag_link( $tag ) ),
|
\esc_url( \get_tag_link( $tag ) ),
|
||||||
esc_hashtag( $tag->name )
|
esc_hashtag( $tag->name )
|
||||||
);
|
);
|
||||||
|
@ -114,7 +114,7 @@ class Shortcodes {
|
||||||
|
|
||||||
/** This filter is documented in wp-includes/post-template.php */
|
/** This filter is documented in wp-includes/post-template.php */
|
||||||
$excerpt = \apply_filters( 'the_content', $excerpt );
|
$excerpt = \apply_filters( 'the_content', $excerpt );
|
||||||
$excerpt = \str_replace( ']]>', ']]>', $excerpt );
|
$excerpt = \str_replace( ']]>', ']]>', $excerpt );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -359,7 +359,7 @@ class Shortcodes {
|
||||||
|
|
||||||
foreach ( $categories as $category ) {
|
foreach ( $categories as $category ) {
|
||||||
$hash_tags[] = \sprintf(
|
$hash_tags[] = \sprintf(
|
||||||
'<a rel="tag" class="u-tag u-category" href="%s">%s</a>',
|
'<a rel="tag" class="hashtag u-tag u-category" href="%s">%s</a>',
|
||||||
\esc_url( \get_category_link( $category ) ),
|
\esc_url( \get_category_link( $category ) ),
|
||||||
esc_hashtag( $category->name )
|
esc_hashtag( $category->name )
|
||||||
);
|
);
|
||||||
|
|
|
@ -4,7 +4,6 @@ namespace Activitypub;
|
||||||
use WP_Error;
|
use WP_Error;
|
||||||
use DateTime;
|
use DateTime;
|
||||||
use DateTimeZone;
|
use DateTimeZone;
|
||||||
use Activitypub\Model\User;
|
|
||||||
use Activitypub\Collection\Users;
|
use Activitypub\Collection\Users;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -23,22 +22,14 @@ class Signature {
|
||||||
*
|
*
|
||||||
* @return mixed The public key.
|
* @return mixed The public key.
|
||||||
*/
|
*/
|
||||||
public static function get_public_key( $user_id, $force = false ) {
|
public static function get_public_key_for( $user_id, $force = false ) {
|
||||||
if ( $force ) {
|
if ( $force ) {
|
||||||
self::generate_key_pair( $user_id );
|
self::generate_key_pair_for( $user_id );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( User::APPLICATION_USER_ID === $user_id ) {
|
$key_pair = self::get_keypair_for( $user_id );
|
||||||
$key = \get_option( 'activitypub_magic_sig_public_key' );
|
|
||||||
} else {
|
|
||||||
$key = \get_user_meta( $user_id, 'magic_sig_public_key', true );
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( ! $key ) {
|
return $key_pair['public_key'];
|
||||||
return self::get_public_key( $user_id, true );
|
|
||||||
}
|
|
||||||
|
|
||||||
return $key;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -49,22 +40,32 @@ class Signature {
|
||||||
*
|
*
|
||||||
* @return mixed The private key.
|
* @return mixed The private key.
|
||||||
*/
|
*/
|
||||||
public static function get_private_key( $user_id, $force = false ) {
|
public static function get_private_key_for( $user_id, $force = false ) {
|
||||||
if ( $force ) {
|
if ( $force ) {
|
||||||
self::generate_key_pair( $user_id );
|
self::generate_key_pair_for( $user_id );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( User::APPLICATION_USER_ID === $user_id ) {
|
$key_pair = self::get_keypair_for( $user_id );
|
||||||
$key = \get_option( 'activitypub_magic_sig_private_key' );
|
|
||||||
} else {
|
return $key_pair['private_key'];
|
||||||
$key = \get_user_meta( $user_id, 'magic_sig_private_key', true );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! $key ) {
|
/**
|
||||||
return self::get_private_key( $user_id, true );
|
* Return the key pair for a given user.
|
||||||
|
*
|
||||||
|
* @param int $user_id The WordPress User ID.
|
||||||
|
*
|
||||||
|
* @return array The key pair.
|
||||||
|
*/
|
||||||
|
public static function get_keypair_for( $user_id ) {
|
||||||
|
$option_key = self::get_signature_options_key_for( $user_id );
|
||||||
|
$key_pair = \get_option( $option_key );
|
||||||
|
|
||||||
|
if ( ! $key_pair ) {
|
||||||
|
$key_pair = self::generate_key_pair_for( $user_id );
|
||||||
}
|
}
|
||||||
|
|
||||||
return $key;
|
return $key_pair;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -72,9 +73,18 @@ class Signature {
|
||||||
*
|
*
|
||||||
* @param int $user_id The WordPress User ID.
|
* @param int $user_id The WordPress User ID.
|
||||||
*
|
*
|
||||||
* @return void
|
* @return array The key pair.
|
||||||
*/
|
*/
|
||||||
public static function generate_key_pair() {
|
protected static function generate_key_pair_for( $user_id ) {
|
||||||
|
$option_key = self::get_signature_options_key_for( $user_id );
|
||||||
|
$key_pair = self::check_legacy_key_pair_for( $user_id );
|
||||||
|
|
||||||
|
if ( $key_pair ) {
|
||||||
|
\add_option( $option_key, $key_pair );
|
||||||
|
|
||||||
|
return $key_pair;
|
||||||
|
}
|
||||||
|
|
||||||
$config = array(
|
$config = array(
|
||||||
'digest_alg' => 'sha512',
|
'digest_alg' => 'sha512',
|
||||||
'private_key_bits' => 2048,
|
'private_key_bits' => 2048,
|
||||||
|
@ -88,10 +98,78 @@ class Signature {
|
||||||
|
|
||||||
$detail = \openssl_pkey_get_details( $key );
|
$detail = \openssl_pkey_get_details( $key );
|
||||||
|
|
||||||
|
// check if keys are valid
|
||||||
|
if (
|
||||||
|
empty( $priv_key ) || ! is_string( $priv_key ) ||
|
||||||
|
! isset( $detail['key'] ) || ! is_string( $detail['key'] )
|
||||||
|
) {
|
||||||
return array(
|
return array(
|
||||||
|
'private_key' => null,
|
||||||
|
'public_key' => null,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$key_pair = array(
|
||||||
'private_key' => $priv_key,
|
'private_key' => $priv_key,
|
||||||
'public_key' => $detail['key'],
|
'public_key' => $detail['key'],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// persist keys
|
||||||
|
\add_option( $option_key, $key_pair );
|
||||||
|
|
||||||
|
return $key_pair;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the option key for a given user.
|
||||||
|
*
|
||||||
|
* @param int $user_id The WordPress User ID.
|
||||||
|
*
|
||||||
|
* @return string The option key.
|
||||||
|
*/
|
||||||
|
protected static function get_signature_options_key_for( $user_id ) {
|
||||||
|
$id = $user_id;
|
||||||
|
|
||||||
|
if ( $user_id > 0 ) {
|
||||||
|
$user = \get_userdata( $user_id );
|
||||||
|
// sanatize username because it could include spaces and special chars
|
||||||
|
$id = sanitize_title( $user->user_login );
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'activitypub_keypair_for_' . $id;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if there is a legacy key pair
|
||||||
|
*
|
||||||
|
* @param int $user_id The WordPress User ID.
|
||||||
|
*
|
||||||
|
* @return array|bool The key pair or false.
|
||||||
|
*/
|
||||||
|
protected static function check_legacy_key_pair_for( $user_id ) {
|
||||||
|
switch ( $user_id ) {
|
||||||
|
case 0:
|
||||||
|
$public_key = \get_option( 'activitypub_blog_user_public_key' );
|
||||||
|
$private_key = \get_option( 'activitypub_blog_user_private_key' );
|
||||||
|
break;
|
||||||
|
case -1:
|
||||||
|
$public_key = \get_option( 'activitypub_application_user_public_key' );
|
||||||
|
$private_key = \get_option( 'activitypub_application_user_private_key' );
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$public_key = \get_user_meta( $user_id, 'magic_sig_public_key', true );
|
||||||
|
$private_key = \get_user_meta( $user_id, 'magic_sig_private_key', true );
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! empty( $public_key ) && is_string( $public_key ) && ! empty( $private_key ) && is_string( $private_key ) ) {
|
||||||
|
return array(
|
||||||
|
'private_key' => $private_key,
|
||||||
|
'public_key' => $public_key,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -107,7 +185,7 @@ class Signature {
|
||||||
*/
|
*/
|
||||||
public static function generate_signature( $user_id, $http_method, $url, $date, $digest = null ) {
|
public static function generate_signature( $user_id, $http_method, $url, $date, $digest = null ) {
|
||||||
$user = Users::get_by_id( $user_id );
|
$user = Users::get_by_id( $user_id );
|
||||||
$key = $user->get__private_key();
|
$key = self::get_private_key_for( $user->get__id() );
|
||||||
|
|
||||||
$url_parts = \wp_parse_url( $url );
|
$url_parts = \wp_parse_url( $url );
|
||||||
|
|
||||||
|
@ -136,7 +214,6 @@ class Signature {
|
||||||
\openssl_sign( $signed_string, $signature, $key, \OPENSSL_ALGO_SHA256 );
|
\openssl_sign( $signed_string, $signature, $key, \OPENSSL_ALGO_SHA256 );
|
||||||
$signature = \base64_encode( $signature ); // phpcs:ignore
|
$signature = \base64_encode( $signature ); // phpcs:ignore
|
||||||
|
|
||||||
$user = Users::get_by_id( $user_id );
|
|
||||||
$key_id = $user->get_url() . '#main-key';
|
$key_id = $user->get_url() . '#main-key';
|
||||||
|
|
||||||
if ( ! empty( $digest ) ) {
|
if ( ! empty( $digest ) ) {
|
||||||
|
@ -161,28 +238,38 @@ class Signature {
|
||||||
} else {
|
} else {
|
||||||
$route = '/' . rest_get_url_prefix() . '/' . ltrim( $request->get_route(), '/' );
|
$route = '/' . rest_get_url_prefix() . '/' . ltrim( $request->get_route(), '/' );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// fix route for subdirectory installs
|
||||||
|
$path = \wp_parse_url( \get_home_url(), PHP_URL_PATH );
|
||||||
|
|
||||||
|
if ( \is_string( $path ) ) {
|
||||||
|
$path = trim( $path, '/' );
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( $path ) {
|
||||||
|
$route = '/' . $path . $route;
|
||||||
|
}
|
||||||
|
|
||||||
$headers = $request->get_headers();
|
$headers = $request->get_headers();
|
||||||
$actor = isset( json_decode( $request->get_body() )->actor ) ? json_decode( $request->get_body() )->actor : '';
|
|
||||||
$headers['(request-target)'][0] = strtolower( $request->get_method() ) . ' ' . $route;
|
$headers['(request-target)'][0] = strtolower( $request->get_method() ) . ' ' . $route;
|
||||||
} else {
|
} else {
|
||||||
$request = self::format_server_request( $request );
|
$request = self::format_server_request( $request );
|
||||||
$headers = $request['headers']; // $_SERVER array
|
$headers = $request['headers']; // $_SERVER array
|
||||||
$actor = null;
|
|
||||||
$headers['(request-target)'][0] = strtolower( $headers['request_method'][0] ) . ' ' . $headers['request_uri'][0];
|
$headers['(request-target)'][0] = strtolower( $headers['request_method'][0] ) . ' ' . $headers['request_uri'][0];
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! isset( $headers['signature'] ) ) {
|
if ( ! isset( $headers['signature'] ) ) {
|
||||||
return new WP_Error( 'activitypub_signature', 'Request not signed', array( 'status' => 403 ) );
|
return new WP_Error( 'activitypub_signature', __( 'Request not signed', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( array_key_exists( 'signature', $headers ) ) {
|
if ( array_key_exists( 'signature', $headers ) ) {
|
||||||
$signature_block = self::parse_signature_header( $headers['signature'] );
|
$signature_block = self::parse_signature_header( $headers['signature'][0] );
|
||||||
} elseif ( array_key_exists( 'authorization', $headers ) ) {
|
} elseif ( array_key_exists( 'authorization', $headers ) ) {
|
||||||
$signature_block = self::parse_signature_header( $headers['authorization'] );
|
$signature_block = self::parse_signature_header( $headers['authorization'][0] );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! isset( $signature_block ) || ! $signature_block ) {
|
if ( ! isset( $signature_block ) || ! $signature_block ) {
|
||||||
return new WP_Error( 'activitypub_signature', 'Incompatible request signature. keyId and signature are required', array( 'status' => 403 ) );
|
return new WP_Error( 'activitypub_signature', __( 'Incompatible request signature. keyId and signature are required', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
$signed_headers = $signature_block['headers'];
|
$signed_headers = $signature_block['headers'];
|
||||||
|
@ -192,12 +279,12 @@ class Signature {
|
||||||
|
|
||||||
$signed_data = self::get_signed_data( $signed_headers, $signature_block, $headers );
|
$signed_data = self::get_signed_data( $signed_headers, $signature_block, $headers );
|
||||||
if ( ! $signed_data ) {
|
if ( ! $signed_data ) {
|
||||||
return new WP_Error( 'activitypub_signature', 'Signed request date outside acceptable time window', array( 'status' => 403 ) );
|
return new WP_Error( 'activitypub_signature', __( 'Signed request date outside acceptable time window', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
$algorithm = self::get_signature_algorithm( $signature_block );
|
$algorithm = self::get_signature_algorithm( $signature_block );
|
||||||
if ( ! $algorithm ) {
|
if ( ! $algorithm ) {
|
||||||
return new WP_Error( 'activitypub_signature', 'Unsupported signature algorithm (only rsa-sha256 and hs2019 are supported)', array( 'status' => 403 ) );
|
return new WP_Error( 'activitypub_signature', __( 'Unsupported signature algorithm (only rsa-sha256 and hs2019 are supported)', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( \in_array( 'digest', $signed_headers, true ) && isset( $body ) ) {
|
if ( \in_array( 'digest', $signed_headers, true ) && isset( $body ) ) {
|
||||||
|
@ -213,15 +300,12 @@ class Signature {
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( \base64_encode( \hash( $hashalg, $body, true ) ) !== $digest[1] ) { // phpcs:ignore
|
if ( \base64_encode( \hash( $hashalg, $body, true ) ) !== $digest[1] ) { // phpcs:ignore
|
||||||
return new WP_Error( 'activitypub_signature', 'Invalid Digest header', array( 'status' => 403 ) );
|
return new WP_Error( 'activitypub_signature', __( 'Invalid Digest header', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( $actor ) {
|
|
||||||
$public_key = self::get_remote_key( $actor );
|
|
||||||
} else {
|
|
||||||
$public_key = self::get_remote_key( $signature_block['keyId'] );
|
$public_key = self::get_remote_key( $signature_block['keyId'] );
|
||||||
}
|
|
||||||
if ( \is_wp_error( $public_key ) ) {
|
if ( \is_wp_error( $public_key ) ) {
|
||||||
return $public_key;
|
return $public_key;
|
||||||
}
|
}
|
||||||
|
@ -229,7 +313,7 @@ class Signature {
|
||||||
$verified = \openssl_verify( $signed_data, $signature_block['signature'], $public_key, $algorithm ) > 0;
|
$verified = \openssl_verify( $signed_data, $signature_block['signature'], $public_key, $algorithm ) > 0;
|
||||||
|
|
||||||
if ( ! $verified ) {
|
if ( ! $verified ) {
|
||||||
return new WP_Error( 'activitypub_signature', 'Invalid signature', array( 'status' => 403 ) );
|
return new WP_Error( 'activitypub_signature', __( 'Invalid signature', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
return $verified;
|
return $verified;
|
||||||
}
|
}
|
||||||
|
@ -239,17 +323,17 @@ class Signature {
|
||||||
*
|
*
|
||||||
* @param string $key_id The URL to the public key.
|
* @param string $key_id The URL to the public key.
|
||||||
*
|
*
|
||||||
* @return string The public key.
|
* @return WP_Error|string The public key.
|
||||||
*/
|
*/
|
||||||
public static function get_remote_key( $key_id ) { // phpcs:ignore
|
public static function get_remote_key( $key_id ) { // phpcs:ignore
|
||||||
$actor = get_remote_metadata_by_actor( strtok( strip_fragment_from_url( $key_id ), '?' ) ); // phpcs:ignore
|
$actor = get_remote_metadata_by_actor( strip_fragment_from_url( $key_id ) ); // phpcs:ignore
|
||||||
if ( \is_wp_error( $actor ) ) {
|
if ( \is_wp_error( $actor ) ) {
|
||||||
return $actor;
|
return $actor;
|
||||||
}
|
}
|
||||||
if ( isset( $actor['publicKey']['publicKeyPem'] ) ) {
|
if ( isset( $actor['publicKey']['publicKeyPem'] ) ) {
|
||||||
return \rtrim( $actor['publicKey']['publicKeyPem'] ); // phpcs:ignore
|
return \rtrim( $actor['publicKey']['publicKeyPem'] ); // phpcs:ignore
|
||||||
}
|
}
|
||||||
return null;
|
return new WP_Error( 'activitypub_no_remote_key_found', __( 'No Public-Key found', 'activitypub' ), array( 'status' => 403 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -274,32 +358,31 @@ class Signature {
|
||||||
/**
|
/**
|
||||||
* Parses the Signature header
|
* Parses the Signature header
|
||||||
*
|
*
|
||||||
* @param array $header The signature header.
|
* @param string $signature The signature header.
|
||||||
*
|
*
|
||||||
* @return array signature parts
|
* @return array signature parts
|
||||||
*/
|
*/
|
||||||
public static function parse_signature_header( $header ) {
|
public static function parse_signature_header( $signature ) {
|
||||||
$parsed_header = array();
|
$parsed_header = array();
|
||||||
$matches = array();
|
$matches = array();
|
||||||
$h_string = \implode( ',', (array) $header[0] );
|
|
||||||
|
|
||||||
if ( \preg_match( '/keyId="(.*?)"/ism', $h_string, $matches ) ) {
|
if ( \preg_match( '/keyId="(.*?)"/ism', $signature, $matches ) ) {
|
||||||
$parsed_header['keyId'] = $matches[1];
|
$parsed_header['keyId'] = trim( $matches[1] );
|
||||||
}
|
}
|
||||||
if ( \preg_match( '/created=([0-9]*)/ism', $h_string, $matches ) ) {
|
if ( \preg_match( '/created=([0-9]*)/ism', $signature, $matches ) ) {
|
||||||
$parsed_header['(created)'] = $matches[1];
|
$parsed_header['(created)'] = trim( $matches[1] );
|
||||||
}
|
}
|
||||||
if ( \preg_match( '/expires=([0-9]*)/ism', $h_string, $matches ) ) {
|
if ( \preg_match( '/expires=([0-9]*)/ism', $signature, $matches ) ) {
|
||||||
$parsed_header['(expires)'] = $matches[1];
|
$parsed_header['(expires)'] = trim( $matches[1] );
|
||||||
}
|
}
|
||||||
if ( \preg_match( '/algorithm="(.*?)"/ism', $h_string, $matches ) ) {
|
if ( \preg_match( '/algorithm="(.*?)"/ism', $signature, $matches ) ) {
|
||||||
$parsed_header['algorithm'] = $matches[1];
|
$parsed_header['algorithm'] = trim( $matches[1] );
|
||||||
}
|
}
|
||||||
if ( \preg_match( '/headers="(.*?)"/ism', $h_string, $matches ) ) {
|
if ( \preg_match( '/headers="(.*?)"/ism', $signature, $matches ) ) {
|
||||||
$parsed_header['headers'] = \explode( ' ', $matches[1] );
|
$parsed_header['headers'] = \explode( ' ', trim( $matches[1] ) );
|
||||||
}
|
}
|
||||||
if ( \preg_match( '/signature="(.*?)"/ism', $h_string, $matches ) ) {
|
if ( \preg_match( '/signature="(.*?)"/ism', $signature, $matches ) ) {
|
||||||
$parsed_header['signature'] = \base64_decode( preg_replace( '/\s+/', '', $matches[1] ) ); // phpcs:ignore
|
$parsed_header['signature'] = \base64_decode( preg_replace( '/\s+/', '', trim( $matches[1] ) ) ); // phpcs:ignore
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ( $parsed_header['signature'] ) && ( $parsed_header['algorithm'] ) && ( ! $parsed_header['headers'] ) ) {
|
if ( ( $parsed_header['signature'] ) && ( $parsed_header['algorithm'] ) && ( ! $parsed_header['headers'] ) ) {
|
||||||
|
@ -312,7 +395,7 @@ class Signature {
|
||||||
/**
|
/**
|
||||||
* Gets the header data from the included pseudo headers
|
* Gets the header data from the included pseudo headers
|
||||||
*
|
*
|
||||||
* @param array $signed_headers
|
* @param array $signed_headers The signed headers.
|
||||||
* @param array $signature_block (pseudo-headers)
|
* @param array $signature_block (pseudo-headers)
|
||||||
* @param array $headers (http headers)
|
* @param array $headers (http headers)
|
||||||
*
|
*
|
||||||
|
|
|
@ -26,7 +26,7 @@ class Webfinger {
|
||||||
}
|
}
|
||||||
|
|
||||||
$user = Users::get_by_id( $user_id );
|
$user = Users::get_by_id( $user_id );
|
||||||
if ( ! $user ) {
|
if ( ! $user || is_wp_error( $user ) ) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,7 +63,7 @@ class Webfinger {
|
||||||
$url,
|
$url,
|
||||||
array(
|
array(
|
||||||
'headers' => array( 'Accept' => 'application/jrd+json' ),
|
'headers' => array( 'Accept' => 'application/jrd+json' ),
|
||||||
'redirection' => 0,
|
'redirection' => 2,
|
||||||
'timeout' => 2,
|
'timeout' => 2,
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -198,6 +198,6 @@ class Webfinger {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return new WP_Error( 'webfinger_remote_follow_endpoint_invalid', null, $data );
|
return new WP_Error( 'webfinger_remote_follow_endpoint_invalid', $data, array( 'status' => 417 ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -160,7 +160,7 @@ class Followers {
|
||||||
* @param int $user_id The ID of the WordPress User
|
* @param int $user_id The ID of the WordPress User
|
||||||
* @param string $actor The Actor URL
|
* @param string $actor The Actor URL
|
||||||
*
|
*
|
||||||
* @return array|WP_Error The Follower (WP_Term array) or an WP_Error
|
* @return array|WP_Error The Follower (WP_Post array) or an WP_Error
|
||||||
*/
|
*/
|
||||||
public static function add_follower( $user_id, $actor ) {
|
public static function add_follower( $user_id, $actor ) {
|
||||||
$meta = get_remote_metadata_by_actor( $actor );
|
$meta = get_remote_metadata_by_actor( $actor );
|
||||||
|
@ -169,29 +169,30 @@ class Followers {
|
||||||
return $meta;
|
return $meta;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ( empty( $meta ) || ! is_array( $meta ) || is_wp_error( $meta ) ) {
|
||||||
|
return new WP_Error( 'activitypub_invalid_follower', __( 'Invalid Follower', 'activitypub' ), array( 'status' => 400 ) );
|
||||||
|
}
|
||||||
|
|
||||||
$error = null;
|
$error = null;
|
||||||
|
|
||||||
$follower = new Follower();
|
$follower = new Follower();
|
||||||
|
|
||||||
if ( empty( $meta ) || ! is_array( $meta ) || is_wp_error( $meta ) ) {
|
|
||||||
$follower->set_id( $actor );
|
|
||||||
$follower->set_url( $actor );
|
|
||||||
$error = $meta;
|
|
||||||
} else {
|
|
||||||
$follower->from_array( $meta );
|
$follower->from_array( $meta );
|
||||||
|
|
||||||
|
$id = $follower->upsert();
|
||||||
|
|
||||||
|
if ( is_wp_error( $id ) ) {
|
||||||
|
return $id;
|
||||||
}
|
}
|
||||||
|
|
||||||
$follower->upsert();
|
$meta = get_post_meta( $id, 'activitypub_user_id' );
|
||||||
|
|
||||||
$meta = get_post_meta( $follower->get__id(), 'activitypub_user_id' );
|
|
||||||
|
|
||||||
if ( $error ) {
|
if ( $error ) {
|
||||||
self::add_error( $follower->get__id(), $error );
|
self::add_error( $id, $error );
|
||||||
}
|
}
|
||||||
|
|
||||||
// phpcs:ignore WordPress.PHP.StrictInArray.MissingTrueStrict
|
// phpcs:ignore WordPress.PHP.StrictInArray.MissingTrueStrict
|
||||||
if ( is_array( $meta ) && ! in_array( $user_id, $meta ) ) {
|
if ( is_array( $meta ) && ! in_array( $user_id, $meta ) ) {
|
||||||
add_post_meta( $follower->get__id(), 'activitypub_user_id', $user_id );
|
add_post_meta( $id, 'activitypub_user_id', $user_id );
|
||||||
wp_cache_delete( sprintf( self::CACHE_KEY_INBOXES, $user_id ), 'activitypub' );
|
wp_cache_delete( sprintf( self::CACHE_KEY_INBOXES, $user_id ), 'activitypub' );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -360,7 +361,17 @@ class Followers {
|
||||||
public static function get_all_followers() {
|
public static function get_all_followers() {
|
||||||
$args = array(
|
$args = array(
|
||||||
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
||||||
'meta_query' => array(),
|
'meta_query' => array(
|
||||||
|
'relation' => 'AND',
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_inbox',
|
||||||
|
'compare' => 'EXISTS',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_actor_json',
|
||||||
|
'compare' => 'EXISTS',
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
return self::get_followers( null, null, null, $args );
|
return self::get_followers( null, null, null, $args );
|
||||||
}
|
}
|
||||||
|
@ -379,10 +390,19 @@ class Followers {
|
||||||
'fields' => 'ids',
|
'fields' => 'ids',
|
||||||
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
||||||
'meta_query' => array(
|
'meta_query' => array(
|
||||||
|
'relation' => 'AND',
|
||||||
array(
|
array(
|
||||||
'key' => 'activitypub_user_id',
|
'key' => 'activitypub_user_id',
|
||||||
'value' => $user_id,
|
'value' => $user_id,
|
||||||
),
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_inbox',
|
||||||
|
'compare' => 'EXISTS',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_actor_json',
|
||||||
|
'compare' => 'EXISTS',
|
||||||
|
),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -412,6 +432,7 @@ class Followers {
|
||||||
'fields' => 'ids',
|
'fields' => 'ids',
|
||||||
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
||||||
'meta_query' => array(
|
'meta_query' => array(
|
||||||
|
'relation' => 'AND',
|
||||||
array(
|
array(
|
||||||
'key' => 'activitypub_inbox',
|
'key' => 'activitypub_inbox',
|
||||||
'compare' => 'EXISTS',
|
'compare' => 'EXISTS',
|
||||||
|
@ -420,6 +441,11 @@ class Followers {
|
||||||
'key' => 'activitypub_user_id',
|
'key' => 'activitypub_user_id',
|
||||||
'value' => $user_id,
|
'value' => $user_id,
|
||||||
),
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_inbox',
|
||||||
|
'value' => '',
|
||||||
|
'compare' => '!=',
|
||||||
|
),
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -462,7 +488,7 @@ class Followers {
|
||||||
'post_type' => self::POST_TYPE,
|
'post_type' => self::POST_TYPE,
|
||||||
'posts_per_page' => $number,
|
'posts_per_page' => $number,
|
||||||
'orderby' => 'modified',
|
'orderby' => 'modified',
|
||||||
'order' => 'DESC',
|
'order' => 'ASC',
|
||||||
'post_status' => 'any', // 'any' includes 'trash
|
'post_status' => 'any', // 'any' includes 'trash
|
||||||
'date_query' => array(
|
'date_query' => array(
|
||||||
array(
|
array(
|
||||||
|
@ -490,16 +516,35 @@ class Followers {
|
||||||
*
|
*
|
||||||
* @return mixed The Term list of Followers, the format depends on $output.
|
* @return mixed The Term list of Followers, the format depends on $output.
|
||||||
*/
|
*/
|
||||||
public static function get_faulty_followers( $number = 10 ) {
|
public static function get_faulty_followers( $number = 20 ) {
|
||||||
$args = array(
|
$args = array(
|
||||||
'post_type' => self::POST_TYPE,
|
'post_type' => self::POST_TYPE,
|
||||||
'posts_per_page' => $number,
|
'posts_per_page' => $number,
|
||||||
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
// phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query
|
||||||
'meta_query' => array(
|
'meta_query' => array(
|
||||||
|
'relation' => 'OR',
|
||||||
array(
|
array(
|
||||||
'key' => 'activitypub_errors',
|
'key' => 'activitypub_errors',
|
||||||
'compare' => 'EXISTS',
|
'compare' => 'EXISTS',
|
||||||
),
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_inbox',
|
||||||
|
'compare' => 'NOT EXISTS',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_actor_json',
|
||||||
|
'compare' => 'NOT EXISTS',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_inbox',
|
||||||
|
'value' => '',
|
||||||
|
'compare' => '=',
|
||||||
|
),
|
||||||
|
array(
|
||||||
|
'key' => 'activitypub_actor_json',
|
||||||
|
'value' => '',
|
||||||
|
'compare' => '=',
|
||||||
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -54,7 +54,7 @@ function get_remote_metadata_by_actor( $actor, $cached = true ) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! $actor ) {
|
if ( ! $actor ) {
|
||||||
return null;
|
return new WP_Error( 'activitypub_no_valid_actor_identifier', \__( 'The "actor" identifier is not valid', 'activitypub' ), array( 'status' => 404, 'actor' => $actor ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( is_wp_error( $actor ) ) {
|
if ( is_wp_error( $actor ) ) {
|
||||||
|
@ -73,7 +73,7 @@ function get_remote_metadata_by_actor( $actor, $cached = true ) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! \wp_http_validate_url( $actor ) ) {
|
if ( ! \wp_http_validate_url( $actor ) ) {
|
||||||
$metadata = new \WP_Error( 'activitypub_no_valid_actor_url', \__( 'The "actor" is no valid URL', 'activitypub' ), $actor );
|
$metadata = new WP_Error( 'activitypub_no_valid_actor_url', \__( 'The "actor" is no valid URL', 'activitypub' ), array( 'status' => 400, 'actor' => $actor ) );
|
||||||
\set_transient( $transient_key, $metadata, HOUR_IN_SECONDS ); // Cache the error for a shorter period.
|
\set_transient( $transient_key, $metadata, HOUR_IN_SECONDS ); // Cache the error for a shorter period.
|
||||||
return $metadata;
|
return $metadata;
|
||||||
}
|
}
|
||||||
|
@ -95,7 +95,7 @@ function get_remote_metadata_by_actor( $actor, $cached = true ) {
|
||||||
\set_transient( $transient_key, $metadata, WEEK_IN_SECONDS );
|
\set_transient( $transient_key, $metadata, WEEK_IN_SECONDS );
|
||||||
|
|
||||||
if ( ! $metadata ) {
|
if ( ! $metadata ) {
|
||||||
$metadata = new \WP_Error( 'activitypub_invalid_json', \__( 'No valid JSON data', 'activitypub' ), $actor );
|
$metadata = new WP_Error( 'activitypub_invalid_json', \__( 'No valid JSON data', 'activitypub' ), array( 'status' => 400, 'actor' => $actor ) );
|
||||||
\set_transient( $transient_key, $metadata, HOUR_IN_SECONDS ); // Cache the error for a shorter period.
|
\set_transient( $transient_key, $metadata, HOUR_IN_SECONDS ); // Cache the error for a shorter period.
|
||||||
return $metadata;
|
return $metadata;
|
||||||
}
|
}
|
||||||
|
@ -680,7 +680,7 @@ function is_user_type_disabled( $type ) {
|
||||||
$return = false;
|
$return = false;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
$return = new WP_Error( 'activitypub_wrong_user_type', __( 'Wrong user type', 'activitypub' ) );
|
$return = new WP_Error( 'activitypub_wrong_user_type', __( 'Wrong user type', 'activitypub' ), array( 'status' => 400 ) );
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -693,20 +693,14 @@ function is_user_type_disabled( $type ) {
|
||||||
* @return boolean True if the blog is in single-user mode, false otherwise.
|
* @return boolean True if the blog is in single-user mode, false otherwise.
|
||||||
*/
|
*/
|
||||||
function is_single_user() {
|
function is_single_user() {
|
||||||
$return = false;
|
if (
|
||||||
|
|
||||||
if ( \defined( 'ACTIVITYPUB_SINGLE_USER_MODE' ) ) {
|
|
||||||
if ( ACTIVITYPUB_SINGLE_USER_MODE ) {
|
|
||||||
$return = true;
|
|
||||||
}
|
|
||||||
} elseif (
|
|
||||||
false === is_user_type_disabled( 'blog' ) &&
|
false === is_user_type_disabled( 'blog' ) &&
|
||||||
true === is_user_type_disabled( 'user' )
|
true === is_user_type_disabled( 'user' )
|
||||||
) {
|
) {
|
||||||
$return = true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $return;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( ! function_exists( 'get_self_link' ) ) {
|
if ( ! function_exists( 'get_self_link' ) ) {
|
||||||
|
@ -721,3 +715,33 @@ if ( ! function_exists( 'get_self_link' ) ) {
|
||||||
return esc_url( apply_filters( 'self_link', set_url_scheme( 'http://' . $host['host'] . $path ) ) );
|
return esc_url( apply_filters( 'self_link', set_url_scheme( 'http://' . $host['host'] . $path ) ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a site supports the block editor.
|
||||||
|
*
|
||||||
|
* @return boolean True if the site supports the block editor, false otherwise.
|
||||||
|
*/
|
||||||
|
function site_supports_blocks() {
|
||||||
|
if ( ! \function_exists( 'register_block_type_from_metadata' ) ) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allow plugins to disable block editor support,
|
||||||
|
* thus disabling blocks registered by the ActivityPub plugin.
|
||||||
|
*
|
||||||
|
* @param boolean $supports_blocks True if the site supports the block editor, false otherwise.
|
||||||
|
*/
|
||||||
|
return apply_filters( 'activitypub_site_supports_blocks', true );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if data is valid JSON.
|
||||||
|
*
|
||||||
|
* @param string $data The data to check.
|
||||||
|
*
|
||||||
|
* @return boolean True if the data is JSON, false otherwise.
|
||||||
|
*/
|
||||||
|
function is_json( $data ) {
|
||||||
|
return \is_array( \json_decode( $data, true ) ) ? true : false;
|
||||||
|
}
|
||||||
|
|
|
@ -22,6 +22,13 @@ class Application_User extends Blog_User {
|
||||||
*/
|
*/
|
||||||
protected $type = 'Application';
|
protected $type = 'Application';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the User is discoverable.
|
||||||
|
*
|
||||||
|
* @var boolean
|
||||||
|
*/
|
||||||
|
protected $discoverable = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the User-Url.
|
* Get the User-Url.
|
||||||
*
|
*
|
||||||
|
@ -35,58 +42,10 @@ class Application_User extends Blog_User {
|
||||||
return 'application';
|
return 'application';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get_username() {
|
public function get_preferred_username() {
|
||||||
return $this::get_name();
|
return $this::get_name();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get__public_key() {
|
|
||||||
$key = \get_option( 'activitypub_application_user_public_key' );
|
|
||||||
|
|
||||||
if ( $key ) {
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->generate_key_pair();
|
|
||||||
|
|
||||||
$key = \get_option( 'activitypub_application_user_public_key' );
|
|
||||||
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param int $user_id
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function get__private_key() {
|
|
||||||
$key = \get_option( 'activitypub_application_user_private_key' );
|
|
||||||
|
|
||||||
if ( $key ) {
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->generate_key_pair();
|
|
||||||
|
|
||||||
return \get_option( 'activitypub_application_user_private_key' );
|
|
||||||
}
|
|
||||||
|
|
||||||
private function generate_key_pair() {
|
|
||||||
$key_pair = Signature::generate_key_pair();
|
|
||||||
|
|
||||||
if ( ! is_wp_error( $key_pair ) ) {
|
|
||||||
\update_option( 'activitypub_application_user_public_key', $key_pair['public_key'] );
|
|
||||||
\update_option( 'activitypub_application_user_private_key', $key_pair['private_key'] );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function get_inbox() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function get_outbox() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function get_followers() {
|
public function get_followers() {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -96,14 +55,18 @@ class Application_User extends Blog_User {
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get_attachment() {
|
public function get_attachment() {
|
||||||
return array();
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get_featured_tags() {
|
public function get_featured_tags() {
|
||||||
return array();
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get_featured() {
|
public function get_featured() {
|
||||||
return array();
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get_moderators() {
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ use Activitypub\Collection\Users;
|
||||||
|
|
||||||
use function Activitypub\is_single_user;
|
use function Activitypub\is_single_user;
|
||||||
use function Activitypub\is_user_disabled;
|
use function Activitypub\is_user_disabled;
|
||||||
|
use function Activitypub\get_rest_url_by_path;
|
||||||
|
|
||||||
class Blog_User extends User {
|
class Blog_User extends User {
|
||||||
/**
|
/**
|
||||||
|
@ -21,7 +22,7 @@ class Blog_User extends User {
|
||||||
*
|
*
|
||||||
* @var string
|
* @var string
|
||||||
*/
|
*/
|
||||||
protected $type = 'Group';
|
protected $type = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Is Account discoverable?
|
* Is Account discoverable?
|
||||||
|
@ -45,6 +46,21 @@ class Blog_User extends User {
|
||||||
return $object;
|
return $object;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the type of the object.
|
||||||
|
*
|
||||||
|
* If the Blog is in "single user" mode, return "Person" insted of "Group".
|
||||||
|
*
|
||||||
|
* @return string The type of the object.
|
||||||
|
*/
|
||||||
|
public function get_type() {
|
||||||
|
if ( is_single_user() ) {
|
||||||
|
return 'Person';
|
||||||
|
} else {
|
||||||
|
return 'Group';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the User-Name.
|
* Get the User-Name.
|
||||||
*
|
*
|
||||||
|
@ -122,23 +138,34 @@ class Blog_User extends User {
|
||||||
/**
|
/**
|
||||||
* Get the User-Icon.
|
* Get the User-Icon.
|
||||||
*
|
*
|
||||||
* @return array|null The User-Icon.
|
* @return array The User-Icon.
|
||||||
*/
|
*/
|
||||||
public function get_icon() {
|
public function get_icon() {
|
||||||
// try site icon first
|
// try site icon first
|
||||||
$icon_id = get_option( 'site_icon' );
|
$icon_id = get_option( 'site_icon' );
|
||||||
|
|
||||||
// try custom logo second
|
// try custom logo second
|
||||||
if ( ! $icon_id ) {
|
if ( ! $icon_id ) {
|
||||||
$icon_id = get_theme_mod( 'custom_logo' );
|
$icon_id = get_theme_mod( 'custom_logo' );
|
||||||
}
|
}
|
||||||
if ( ! $icon_id ) {
|
|
||||||
return null;
|
$icon_url = false;
|
||||||
|
|
||||||
|
if ( $icon_id ) {
|
||||||
|
$icon = wp_get_attachment_image_src( $icon_id, 'full' );
|
||||||
|
if ( $icon ) {
|
||||||
|
$icon_url = $icon[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( ! $icon_url ) {
|
||||||
|
// fallback to default icon
|
||||||
|
$icon_url = plugins_url( '/assets/img/wp-logo.png', ACTIVITYPUB_PLUGIN_FILE );
|
||||||
}
|
}
|
||||||
|
|
||||||
$image = wp_get_attachment_image_src( $icon_id, 'full' );
|
|
||||||
return array(
|
return array(
|
||||||
'type' => 'Image',
|
'type' => 'Image',
|
||||||
'url' => esc_url( $image[0] ),
|
'url' => esc_url( $icon_url ),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -176,48 +203,6 @@ class Blog_User extends User {
|
||||||
return \gmdate( 'Y-m-d\TH:i:s\Z', $time );
|
return \gmdate( 'Y-m-d\TH:i:s\Z', $time );
|
||||||
}
|
}
|
||||||
|
|
||||||
public function get__public_key() {
|
|
||||||
$key = \get_option( 'activitypub_blog_user_public_key' );
|
|
||||||
|
|
||||||
if ( $key ) {
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->generate_key_pair();
|
|
||||||
|
|
||||||
$key = \get_option( 'activitypub_blog_user_public_key' );
|
|
||||||
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the User-Private-Key.
|
|
||||||
*
|
|
||||||
* @param int $user_id
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function get__private_key() {
|
|
||||||
$key = \get_option( 'activitypub_blog_user_private_key' );
|
|
||||||
|
|
||||||
if ( $key ) {
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->generate_key_pair();
|
|
||||||
|
|
||||||
return \get_option( 'activitypub_blog_user_private_key' );
|
|
||||||
}
|
|
||||||
|
|
||||||
private function generate_key_pair() {
|
|
||||||
$key_pair = Signature::generate_key_pair();
|
|
||||||
|
|
||||||
if ( ! is_wp_error( $key_pair ) ) {
|
|
||||||
\update_option( 'activitypub_blog_user_public_key', $key_pair['public_key'] );
|
|
||||||
\update_option( 'activitypub_blog_user_private_key', $key_pair['private_key'] );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public function get_attachment() {
|
public function get_attachment() {
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
|
@ -226,18 +211,27 @@ class Blog_User extends User {
|
||||||
return \home_url();
|
return \home_url();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public function get_moderators() {
|
||||||
* Get the type of the object.
|
if ( is_single_user() || 'Group' !== $this->get_type() ) {
|
||||||
*
|
return null;
|
||||||
* If the Blog is in "single user" mode, return "Person" insted of "Group".
|
}
|
||||||
*
|
|
||||||
* @return string The type of the object.
|
return get_rest_url_by_path( 'collections/moderators' );
|
||||||
*/
|
}
|
||||||
public function get_type() {
|
|
||||||
if ( is_single_user() ) {
|
public function get_attributed_to() {
|
||||||
return 'Person';
|
if ( is_single_user() || 'Group' !== $this->get_type() ) {
|
||||||
} else {
|
return null;
|
||||||
return $this->type;
|
}
|
||||||
}
|
|
||||||
|
return get_rest_url_by_path( 'collections/moderators' );
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get_posting_restricted_to_mods() {
|
||||||
|
if ( 'Group' === $this->get_type() ) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<?php
|
<?php
|
||||||
namespace Activitypub\Model;
|
namespace Activitypub\Model;
|
||||||
|
|
||||||
|
use WP_Error;
|
||||||
use WP_Query;
|
use WP_Query;
|
||||||
use Activitypub\Activity\Actor;
|
use Activitypub\Activity\Actor;
|
||||||
use Activitypub\Collection\Followers;
|
use Activitypub\Collection\Followers;
|
||||||
|
@ -110,12 +111,40 @@ class Follower extends Actor {
|
||||||
$this->save();
|
$this->save();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validate the current Follower-Object.
|
||||||
|
*
|
||||||
|
* @return boolean True if the verification was successful.
|
||||||
|
*/
|
||||||
|
public function is_valid() {
|
||||||
|
// the minimum required attributes
|
||||||
|
$required_attributes = array(
|
||||||
|
'id',
|
||||||
|
'preferredUsername',
|
||||||
|
'inbox',
|
||||||
|
'publicKey',
|
||||||
|
'publicKeyPem',
|
||||||
|
);
|
||||||
|
|
||||||
|
foreach ( $required_attributes as $attribute ) {
|
||||||
|
if ( ! $this->get( $attribute ) ) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save the current Follower-Object.
|
* Save the current Follower-Object.
|
||||||
*
|
*
|
||||||
* @return void
|
* @return int|WP_Error The Post-ID or an WP_Error.
|
||||||
*/
|
*/
|
||||||
public function save() {
|
public function save() {
|
||||||
|
if ( ! $this->is_valid() ) {
|
||||||
|
return new WP_Error( 'activitypub_invalid_follower', __( 'Invalid Follower', 'activitypub' ), array( 'status' => 400 ) );
|
||||||
|
}
|
||||||
|
|
||||||
if ( ! $this->get__id() ) {
|
if ( ! $this->get__id() ) {
|
||||||
global $wpdb;
|
global $wpdb;
|
||||||
|
|
||||||
|
@ -147,15 +176,17 @@ class Follower extends Actor {
|
||||||
|
|
||||||
$post_id = wp_insert_post( $args );
|
$post_id = wp_insert_post( $args );
|
||||||
$this->_id = $post_id;
|
$this->_id = $post_id;
|
||||||
|
|
||||||
|
return $post_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Upsert the current Follower-Object.
|
* Upsert the current Follower-Object.
|
||||||
*
|
*
|
||||||
* @return void
|
* @return int|WP_Error The Post-ID or an WP_Error.
|
||||||
*/
|
*/
|
||||||
public function upsert() {
|
public function upsert() {
|
||||||
$this->save();
|
return $this->save();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -36,6 +36,15 @@ class User extends Actor {
|
||||||
*/
|
*/
|
||||||
protected $featured;
|
protected $featured;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Moderators endpoint.
|
||||||
|
*
|
||||||
|
* @see https://join-lemmy.org/docs/contributors/05-federation.html
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $moderators;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The User-Type
|
* The User-Type
|
||||||
*
|
*
|
||||||
|
@ -43,6 +52,38 @@ class User extends Actor {
|
||||||
*/
|
*/
|
||||||
protected $type = 'Person';
|
protected $type = 'Person';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the User is discoverable.
|
||||||
|
*
|
||||||
|
* @see https://docs.joinmastodon.org/spec/activitypub/#discoverable
|
||||||
|
*
|
||||||
|
* @var boolean
|
||||||
|
*/
|
||||||
|
protected $discoverable = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If the User is indexable.
|
||||||
|
*
|
||||||
|
* @var boolean
|
||||||
|
*/
|
||||||
|
protected $indexable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The WebFinger Resource.
|
||||||
|
*
|
||||||
|
* @var string<url>
|
||||||
|
*/
|
||||||
|
protected $resource;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restrict posting to mods
|
||||||
|
*
|
||||||
|
* @see https://join-lemmy.org/docs/contributors/05-federation.html
|
||||||
|
*
|
||||||
|
* @var boolean
|
||||||
|
*/
|
||||||
|
protected $posting_restricted_to_mods = null;
|
||||||
|
|
||||||
public static function from_wp_user( $user_id ) {
|
public static function from_wp_user( $user_id ) {
|
||||||
if ( is_user_disabled( $user_id ) ) {
|
if ( is_user_disabled( $user_id ) ) {
|
||||||
return new WP_Error(
|
return new WP_Error(
|
||||||
|
@ -145,53 +186,10 @@ class User extends Actor {
|
||||||
return array(
|
return array(
|
||||||
'id' => $this->get_id() . '#main-key',
|
'id' => $this->get_id() . '#main-key',
|
||||||
'owner' => $this->get_id(),
|
'owner' => $this->get_id(),
|
||||||
'publicKeyPem' => $this->get__public_key(),
|
'publicKeyPem' => Signature::get_public_key_for( $this->get__id() ),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @param int $this->get__id()
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function get__public_key() {
|
|
||||||
$key = \get_user_meta( $this->get__id(), 'magic_sig_public_key', true );
|
|
||||||
|
|
||||||
if ( $key ) {
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->generate_key_pair();
|
|
||||||
|
|
||||||
return \get_user_meta( $this->get__id(), 'magic_sig_public_key', true );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param int $this->get__id()
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function get__private_key() {
|
|
||||||
$key = \get_user_meta( $this->get__id(), 'magic_sig_private_key', true );
|
|
||||||
|
|
||||||
if ( $key ) {
|
|
||||||
return $key;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->generate_key_pair();
|
|
||||||
|
|
||||||
return \get_user_meta( $this->get__id(), 'magic_sig_private_key', true );
|
|
||||||
}
|
|
||||||
|
|
||||||
private function generate_key_pair() {
|
|
||||||
$key_pair = Signature::generate_key_pair();
|
|
||||||
|
|
||||||
if ( ! is_wp_error( $key_pair ) ) {
|
|
||||||
\update_user_meta( $this->get__id(), 'magic_sig_public_key', $key_pair['public_key'], true );
|
|
||||||
\update_user_meta( $this->get__id(), 'magic_sig_private_key', $key_pair['private_key'], true );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the Inbox-API-Endpoint.
|
* Returns the Inbox-API-Endpoint.
|
||||||
*
|
*
|
||||||
|
@ -301,4 +299,20 @@ class User extends Actor {
|
||||||
public function get_canonical_url() {
|
public function get_canonical_url() {
|
||||||
return $this->get_url();
|
return $this->get_url();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function get_streams() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get_tag() {
|
||||||
|
return array();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get_indexable() {
|
||||||
|
if ( \get_option( 'blog_public', 1 ) ) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
<?php
|
<?php
|
||||||
namespace Activitypub\Rest;
|
namespace Activitypub\Rest;
|
||||||
|
|
||||||
|
use WP_Error;
|
||||||
use WP_REST_Server;
|
use WP_REST_Server;
|
||||||
use WP_REST_Response;
|
use WP_REST_Response;
|
||||||
use Activitypub\Transformer\Post;
|
use Activitypub\Transformer\Post;
|
||||||
|
@ -56,6 +57,18 @@ class Collection {
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
\register_rest_route(
|
||||||
|
ACTIVITYPUB_REST_NAMESPACE,
|
||||||
|
'/collections/moderators',
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'methods' => WP_REST_Server::READABLE,
|
||||||
|
'callback' => array( self::class, 'moderators_get' ),
|
||||||
|
'permission_callback' => '__return_true',
|
||||||
|
),
|
||||||
|
)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -158,6 +171,30 @@ class Collection {
|
||||||
return new WP_REST_Response( $response, 200 );
|
return new WP_REST_Response( $response, 200 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Moderators endpoint
|
||||||
|
*
|
||||||
|
* @param WP_REST_Request $request The request object.
|
||||||
|
*
|
||||||
|
* @return WP_REST_Response The response object.
|
||||||
|
*/
|
||||||
|
public static function moderators_get( $request ) {
|
||||||
|
$response = array(
|
||||||
|
'@context' => Activity::CONTEXT,
|
||||||
|
'id' => get_rest_url_by_path( 'collections/moderators' ),
|
||||||
|
'type' => 'OrderedCollection',
|
||||||
|
'orderedItems' => array(),
|
||||||
|
);
|
||||||
|
|
||||||
|
$users = User_Collection::get_collection();
|
||||||
|
|
||||||
|
foreach ( $users as $user ) {
|
||||||
|
$response['orderedItems'][] = $user->get_url();
|
||||||
|
}
|
||||||
|
|
||||||
|
return new WP_REST_Response( $response, 200 );
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The supported parameters
|
* The supported parameters
|
||||||
*
|
*
|
||||||
|
|
|
@ -131,7 +131,7 @@ class Inbox {
|
||||||
return $user;
|
return $user;
|
||||||
}
|
}
|
||||||
|
|
||||||
$data = $request->get_params();
|
$data = $request->get_json_params();
|
||||||
$type = $request->get_param( 'type' );
|
$type = $request->get_param( 'type' );
|
||||||
$type = \strtolower( $type );
|
$type = \strtolower( $type );
|
||||||
|
|
||||||
|
@ -149,7 +149,7 @@ class Inbox {
|
||||||
* @return WP_REST_Response
|
* @return WP_REST_Response
|
||||||
*/
|
*/
|
||||||
public static function shared_inbox_post( $request ) {
|
public static function shared_inbox_post( $request ) {
|
||||||
$data = $request->get_params();
|
$data = $request->get_json_params();
|
||||||
$type = $request->get_param( 'type' );
|
$type = $request->get_param( 'type' );
|
||||||
$users = self::extract_recipients( $data );
|
$users = self::extract_recipients( $data );
|
||||||
|
|
||||||
|
@ -171,6 +171,12 @@ class Inbox {
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ( $users as $user ) {
|
foreach ( $users as $user ) {
|
||||||
|
$user = User_Collection::get_by_various( $user );
|
||||||
|
|
||||||
|
if ( is_wp_error( $user ) ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
$type = \strtolower( $type );
|
$type = \strtolower( $type );
|
||||||
|
|
||||||
\do_action( 'activitypub_inbox', $data, $user->ID, $type );
|
\do_action( 'activitypub_inbox', $data, $user->ID, $type );
|
||||||
|
@ -325,61 +331,6 @@ class Inbox {
|
||||||
return $params;
|
return $params;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles "Reaction" requests
|
|
||||||
*
|
|
||||||
* @param array $object The activity-object
|
|
||||||
* @param int $user_id The id of the local blog-user
|
|
||||||
*/
|
|
||||||
public static function handle_reaction( $object, $user_id ) {
|
|
||||||
$meta = get_remote_metadata_by_actor( $object['actor'] );
|
|
||||||
|
|
||||||
$comment_post_id = \url_to_postid( $object['object'] );
|
|
||||||
|
|
||||||
// save only replys and reactions
|
|
||||||
if ( ! $comment_post_id ) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$commentdata = array(
|
|
||||||
'comment_post_ID' => $comment_post_id,
|
|
||||||
'comment_author' => \esc_attr( $meta['name'] ),
|
|
||||||
'comment_author_email' => '',
|
|
||||||
'comment_author_url' => \esc_url_raw( $object['actor'] ),
|
|
||||||
'comment_content' => \esc_url_raw( $object['actor'] ),
|
|
||||||
'comment_type' => \esc_attr( \strtolower( $object['type'] ) ),
|
|
||||||
'comment_parent' => 0,
|
|
||||||
'comment_meta' => array(
|
|
||||||
'source_url' => \esc_url_raw( $object['id'] ),
|
|
||||||
'avatar_url' => \esc_url_raw( $meta['icon']['url'] ),
|
|
||||||
'protocol' => 'activitypub',
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
// disable flood control
|
|
||||||
\remove_action( 'check_comment_flood', 'check_comment_flood_db', 10 );
|
|
||||||
|
|
||||||
// do not require email for AP entries
|
|
||||||
\add_filter( 'pre_option_require_name_email', '__return_false' );
|
|
||||||
|
|
||||||
// No nonce possible for this submission route
|
|
||||||
\add_filter(
|
|
||||||
'akismet_comment_nonce',
|
|
||||||
function() {
|
|
||||||
return 'inactive';
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
$state = \wp_new_comment( $commentdata, true );
|
|
||||||
|
|
||||||
\remove_filter( 'pre_option_require_name_email', '__return_false' );
|
|
||||||
|
|
||||||
// re-add flood control
|
|
||||||
\add_action( 'check_comment_flood', 'check_comment_flood_db', 10, 4 );
|
|
||||||
|
|
||||||
do_action( 'activitypub_handled_reaction', $object, $user_id, $state, $commentdata );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts a new ActivityPub object to comment data suitable for creating a comment
|
* Converts a new ActivityPub object to comment data suitable for creating a comment
|
||||||
*
|
*
|
||||||
|
@ -387,9 +338,10 @@ class Inbox {
|
||||||
*
|
*
|
||||||
* @return array Comment data suitable for creating a comment.
|
* @return array Comment data suitable for creating a comment.
|
||||||
*/
|
*/
|
||||||
public static function convert_object_to_comment_data( $object ) {
|
public static function convert_object_to_comment_data( $object, $user_id ) {
|
||||||
if ( ! isset( $object['object'] ) ) {
|
|
||||||
return false;
|
if ( ! isset( $object['object']['inReplyTo'] ) ) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// check if Activity is public or not
|
// check if Activity is public or not
|
||||||
|
|
|
@ -46,6 +46,11 @@ class Server {
|
||||||
*/
|
*/
|
||||||
public static function application_actor() {
|
public static function application_actor() {
|
||||||
$user = new Application_User();
|
$user = new Application_User();
|
||||||
|
|
||||||
|
$user->set_context(
|
||||||
|
\Activitypub\Activity\Activity::CONTEXT
|
||||||
|
);
|
||||||
|
|
||||||
$json = $user->to_array();
|
$json = $user->to_array();
|
||||||
|
|
||||||
$response = new WP_REST_Response( $json, 200 );
|
$response = new WP_REST_Response( $json, 200 );
|
||||||
|
@ -80,12 +85,12 @@ class Server {
|
||||||
}
|
}
|
||||||
|
|
||||||
// POST-Requets are always signed
|
// POST-Requets are always signed
|
||||||
if ( 'POST' === $request->get_method() ) {
|
if ( 'post' === \strtolower( $request->get_method() ) ) {
|
||||||
$verified_request = Signature::verify_http_signature( $request );
|
$verified_request = Signature::verify_http_signature( $request );
|
||||||
if ( \is_wp_error( $verified_request ) ) {
|
if ( \is_wp_error( $verified_request ) ) {
|
||||||
return $verified_request;
|
return $verified_request;
|
||||||
}
|
}
|
||||||
} elseif ( 'GET' === $request->get_method() ) { // GET-Requests are only signed in secure mode
|
} elseif ( 'get' === \strtolower( $request->get_method() ) ) { // GET-Requests are only signed in secure mode
|
||||||
if ( ACTIVITYPUB_AUTHORIZED_FETCH ) {
|
if ( ACTIVITYPUB_AUTHORIZED_FETCH ) {
|
||||||
$verified_request = Signature::verify_http_signature( $request );
|
$verified_request = Signature::verify_http_signature( $request );
|
||||||
if ( \is_wp_error( $verified_request ) ) {
|
if ( \is_wp_error( $verified_request ) ) {
|
||||||
|
|
|
@ -9,6 +9,7 @@ use Activitypub\Activity\Base_Object;
|
||||||
use function Activitypub\esc_hashtag;
|
use function Activitypub\esc_hashtag;
|
||||||
use function Activitypub\is_single_user;
|
use function Activitypub\is_single_user;
|
||||||
use function Activitypub\get_rest_url_by_path;
|
use function Activitypub\get_rest_url_by_path;
|
||||||
|
use function Activitypub\site_supports_blocks;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* WordPress Post Transformer
|
* WordPress Post Transformer
|
||||||
|
@ -114,8 +115,8 @@ class Post {
|
||||||
$wp_post = $this->wp_post;
|
$wp_post = $this->wp_post;
|
||||||
$object = new Base_Object();
|
$object = new Base_Object();
|
||||||
|
|
||||||
$object->set_id( \esc_url( \get_permalink( $wp_post->ID ) ) );
|
$object->set_id( $this->get_id() );
|
||||||
$object->set_url( \esc_url( \get_permalink( $wp_post->ID ) ) );
|
$object->set_url( $this->get_url() );
|
||||||
$object->set_type( $this->get_object_type() );
|
$object->set_type( $this->get_object_type() );
|
||||||
|
|
||||||
$published = \strtotime( $wp_post->post_date_gmt );
|
$published = \strtotime( $wp_post->post_date_gmt );
|
||||||
|
@ -150,6 +151,32 @@ class Post {
|
||||||
return $object;
|
return $object;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the ID of the Post.
|
||||||
|
*
|
||||||
|
* @return string The Posts ID.
|
||||||
|
*/
|
||||||
|
public function get_id() {
|
||||||
|
return $this->get_url();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the URL of the Post.
|
||||||
|
*
|
||||||
|
* @return string The Posts URL.
|
||||||
|
*/
|
||||||
|
public function get_url() {
|
||||||
|
$post = $this->wp_post;
|
||||||
|
|
||||||
|
if ( 'trash' === get_post_status( $post ) ) {
|
||||||
|
$permalink = \get_post_meta( $post->ID, 'activitypub_canonical_url', true );
|
||||||
|
} else {
|
||||||
|
$permalink = \get_permalink( $post );
|
||||||
|
}
|
||||||
|
|
||||||
|
return \esc_url( $permalink );
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the User-URL of the Author of the Post.
|
* Returns the User-URL of the Author of the Post.
|
||||||
*
|
*
|
||||||
|
@ -166,6 +193,66 @@ class Post {
|
||||||
return Users::get_by_id( $this->wp_post->post_author )->get_url();
|
return Users::get_by_id( $this->wp_post->post_author )->get_url();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the Image Attachments for this Post, parsed from blocks.
|
||||||
|
* @param int $max_images The maximum number of images to return.
|
||||||
|
* @param array $image_ids The image IDs to append new IDs to.
|
||||||
|
*
|
||||||
|
* @return array The image IDs.
|
||||||
|
*/
|
||||||
|
protected function get_block_image_ids( $max_images, $image_ids = [] ) {
|
||||||
|
$blocks = \parse_blocks( $this->wp_post->post_content );
|
||||||
|
return self::get_image_ids_from_blocks( $blocks, $image_ids, $max_images );
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Recursively get image IDs from blocks.
|
||||||
|
* @param array $blocks The blocks to search for image IDs
|
||||||
|
* @param array $image_ids The image IDs to append new IDs to
|
||||||
|
* @param int $max_images The maximum number of images to return.
|
||||||
|
*
|
||||||
|
* @return array The image IDs.
|
||||||
|
*/
|
||||||
|
protected static function get_image_ids_from_blocks( $blocks, $image_ids, $max_images ) {
|
||||||
|
foreach ( $blocks as $block ) {
|
||||||
|
// recurse into inner blocks
|
||||||
|
if ( ! empty( $block['innerBlocks'] ) ) {
|
||||||
|
$image_ids = self::get_image_ids_from_blocks( $block['innerBlocks'], $image_ids, $max_images );
|
||||||
|
}
|
||||||
|
|
||||||
|
switch ( $block['blockName'] ) {
|
||||||
|
case 'core/image':
|
||||||
|
case 'core/cover':
|
||||||
|
if ( ! empty( $block['attrs']['id'] ) ) {
|
||||||
|
$image_ids[] = $block['attrs']['id'];
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'jetpack/slideshow':
|
||||||
|
case 'jetpack/tiled-gallery':
|
||||||
|
if ( ! empty( $block['attrs']['ids'] ) ) {
|
||||||
|
$image_ids = array_merge( $image_ids, $block['attrs']['ids'] );
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'jetpack/image-compare':
|
||||||
|
if ( ! empty( $block['attrs']['beforeImageId'] ) ) {
|
||||||
|
$image_ids[] = $block['attrs']['beforeImageId'];
|
||||||
|
}
|
||||||
|
if ( ! empty( $block['attrs']['afterImageId'] ) ) {
|
||||||
|
$image_ids[] = $block['attrs']['afterImageId'];
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// we could be at or over max, stop unneeded work
|
||||||
|
if ( count( $image_ids ) >= $max_images ) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// still need to slice it because one gallery could knock us over the limit
|
||||||
|
return \array_slice( $image_ids, 0, $max_images );
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generates all Image Attachments for a Post.
|
* Generates all Image Attachments for a Post.
|
||||||
*
|
*
|
||||||
|
@ -192,7 +279,11 @@ class Post {
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( $max_images > 0 ) {
|
if ( $max_images > 0 ) {
|
||||||
// then list any image attachments
|
// first try to get images that are actually in the post content
|
||||||
|
if ( site_supports_blocks() && \has_blocks( $this->wp_post->post_content ) ) {
|
||||||
|
$image_ids = $this->get_block_image_ids( $max_images, $image_ids );
|
||||||
|
} else {
|
||||||
|
// fallback to images attached to the post
|
||||||
$query = new \WP_Query(
|
$query = new \WP_Query(
|
||||||
array(
|
array(
|
||||||
'post_parent' => $id,
|
'post_parent' => $id,
|
||||||
|
@ -210,6 +301,7 @@ class Post {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$image_ids = \array_unique( $image_ids );
|
$image_ids = \array_unique( $image_ids );
|
||||||
|
|
||||||
|
@ -275,7 +367,7 @@ class Post {
|
||||||
* @param int $id The attachment ID.
|
* @param int $id The attachment ID.
|
||||||
* @param string $image_size The image size to retrieve. Set to 'full' by default.
|
* @param string $image_size The image size to retrieve. Set to 'full' by default.
|
||||||
*/
|
*/
|
||||||
do_action( 'activitypub_get_image_pre', $id, $image_size );
|
do_action( 'activitypub_get_image_post', $id, $image_size );
|
||||||
|
|
||||||
return $thumbnail;
|
return $thumbnail;
|
||||||
}
|
}
|
||||||
|
@ -357,7 +449,7 @@ class Post {
|
||||||
|
|
||||||
$mentions = $this->get_mentions();
|
$mentions = $this->get_mentions();
|
||||||
if ( $mentions ) {
|
if ( $mentions ) {
|
||||||
foreach ( $mentions as $mention => $url ) {
|
foreach ( $mentions as $url ) {
|
||||||
$cc[] = $url;
|
$cc[] = $url;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -447,7 +539,7 @@ class Post {
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( 'content' === \get_option( 'activitypub_post_content_type', 'content' ) ) {
|
if ( 'content' === \get_option( 'activitypub_post_content_type', 'content' ) ) {
|
||||||
return "[ap_content]\n\n[ap_hashtags]\n\n[ap_permalink type=\"html\"]";
|
return "[ap_content]\n\n[ap_permalink type=\"html\"]\n\n[ap_hashtags]";
|
||||||
}
|
}
|
||||||
|
|
||||||
return \get_option( 'activitypub_custom_post_content', ACTIVITYPUB_CUSTOM_POST_CONTENT );
|
return \get_option( 'activitypub_custom_post_content', ACTIVITYPUB_CUSTOM_POST_CONTENT );
|
||||||
|
|
|
@ -11,7 +11,8 @@
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "wp-scripts start",
|
"dev": "wp-scripts start",
|
||||||
"build": "wp-scripts build"
|
"build": "wp-scripts build",
|
||||||
|
"readme": "grunt wp_readme_to_markdown"
|
||||||
},
|
},
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"bugs": {
|
"bugs": {
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
<rule ref="WordPress-Core">
|
<rule ref="WordPress-Core">
|
||||||
<exclude name="Generic.Formatting.MultipleStatementAlignment.NotSameWarning" />
|
<exclude name="Generic.Formatting.MultipleStatementAlignment.NotSameWarning" />
|
||||||
<exclude name="WordPress.Arrays.MultipleStatementAlignment.DoubleArrowNotAligned" />
|
<exclude name="WordPress.Arrays.MultipleStatementAlignment.DoubleArrowNotAligned" />
|
||||||
|
<exclude name="Squiz.Functions.MultiLineFunctionDeclaration.SpaceAfterFunction" />
|
||||||
</rule>
|
</rule>
|
||||||
<rule ref="WordPress.Files.FileName">
|
<rule ref="WordPress.Files.FileName">
|
||||||
<properties>
|
<properties>
|
||||||
|
|
86
readme.txt
86
readme.txt
|
@ -1,9 +1,9 @@
|
||||||
=== ActivityPub ===
|
=== ActivityPub ===
|
||||||
Contributors: automattic, pfefferle, mediaformat, mattwiebe, akirk, jeherve, nuriapena
|
Contributors: automattic, pfefferle, mediaformat, mattwiebe, akirk, jeherve, nuriapena, cavalierlife
|
||||||
Tags: OStatus, fediverse, activitypub, activitystream
|
Tags: OStatus, fediverse, activitypub, activitystream
|
||||||
Requires at least: 4.7
|
Requires at least: 4.7
|
||||||
Tested up to: 6.3
|
Tested up to: 6.3
|
||||||
Stable tag: 1.0.0
|
Stable tag: 1.0.1
|
||||||
Requires PHP: 5.6
|
Requires PHP: 5.6
|
||||||
License: MIT
|
License: MIT
|
||||||
License URI: http://opensource.org/licenses/MIT
|
License URI: http://opensource.org/licenses/MIT
|
||||||
|
@ -12,39 +12,39 @@ The ActivityPub protocol is a decentralized social networking protocol based upo
|
||||||
|
|
||||||
== Description ==
|
== Description ==
|
||||||
|
|
||||||
This is BETA software, see the FAQ to see the current feature set or rather what is still planned.
|
Enter the fediverse with **ActivityPub**, broadcasting your blog to a wider audience! Attract followers, deliver updates, and receive comments from a diverse user base of **ActivityPub**\-compliant platforms.
|
||||||
|
|
||||||
The plugin implements the ActivityPub protocol for your blog, which means that your readers will be able to follow your blog posts on Mastodon and other federated platforms that support ActivityPub. In addition, replies to your posts on Mastodon and related platforms will automatically become comments on your blog post.
|
With the ActivityPub plugin installed, your WordPress blog itself function as a federated profile, along with profiles for each author. For instance, if your website is `example.com`, then the blog-wide profile can be found at `@example.com@example.com`, and authors like Jane and Bob would have their individual profiles at `@jane@example.com` and `@bobz@example.com`, respectively.
|
||||||
|
|
||||||
|
An example: I give you my Mastodon profile name: `@pfefferle@mastodon.social`. You search, see my profile, and hit follow. Now, any post I make appears in your Home feed. Similarly, with the ActivityPub plugin, you can find and follow Jane's profile at `@jane@example.com`.
|
||||||
|
|
||||||
|
Once you follow Jane's `@jane@example.com` profile, any blog post she crafts on `example.com` will land in your Home feed. Simultaneously, by following the blog-wide profile `@example.com@example.com`, you'll receive updates from all authors.
|
||||||
|
|
||||||
|
**Note**: if no one follows your author or blog instance, your posts remain unseen. The simplest method to verify the plugin's operation is by following your profile. If you possess a Mastodon profile, initiate by following your new one.
|
||||||
|
|
||||||
The plugin works with the following tested federated platforms, but there may be more that it works with as well:
|
The plugin works with the following tested federated platforms, but there may be more that it works with as well:
|
||||||
|
|
||||||
* [Mastodon](https://joinmastodon.org/)
|
* [Mastodon](https://joinmastodon.org/)
|
||||||
* [Pleroma](https://pleroma.social/)
|
* [Pleroma](https://pleroma.social/)/[Akkoma](https://akkoma.social/)
|
||||||
* [friendica](https://friendi.ca/)
|
* [friendica](https://friendi.ca/)
|
||||||
* [Hubzilla](https://hubzilla.org/)
|
* [Hubzilla](https://hubzilla.org/)
|
||||||
* [Pixelfed](https://pixelfed.org/)
|
* [Pixelfed](https://pixelfed.org/)
|
||||||
* [Socialhome](https://socialhome.network/)
|
* [Socialhome](https://socialhome.network/)
|
||||||
* [Misskey](https://join.misskey.page/)
|
* [Misskey](https://join.misskey.page/)
|
||||||
* [Calckey](https://calckey.org/)
|
* [Firefish](https://joinfirefish.org/) (rebrand of Calckey)
|
||||||
|
|
||||||
Here’s what that means and what you can expect.
|
|
||||||
|
|
||||||
Once the ActivityPub plugin is installed, each author’s page on your WordPress blog will become its own federated instance. In other words, if you have two authors, Jane and Bob, on your website, `example.com`, then your authors would have their own author pages at `example.com/author/jane` and `example.com/author/bob`. Each of those author pages would now be available to Mastodon users (and all other federated platform users) as a profile that can be followed. Let’s break that down further. Let’s say you have a friend on Mastodon who tells you to follow them and they give you their profile name `@janelivesheresomeofthetime@mastodon.social`. You search for her name, see her profile, and click the follow button, right? From then on, everything Jane posts on her profile shows up in your Home feed. Okay, similarly, now that Jane has installed the ActivityPub plugin on her `example.com` site, her friends can also follow her on Mastodon by searching for `@jane@example.com` and clicking the Follow button on that profile.
|
|
||||||
|
|
||||||
From now on, every blog post Jane publishes on example.com will show up on your Home feed because you follow her `@jane@example.com` profile.
|
|
||||||
Of course, if no one follows your author instance, then no one will ever see the posts - including you! So the easiest way to even know if the plugin is working is to follow your new profile yourself. If you already have a Mastodon profile, just follow your new one from there.
|
|
||||||
|
|
||||||
Some things to note:
|
Some things to note:
|
||||||
|
|
||||||
1. Many single-author blogs have chosen to turn off or redirect their author profile pages, usually via an SEO plugin like Yoast or Rank Math. This is usually done to avoid duplicate content with your blog’s home page. If your author page has been deactivated in this way, then ActivityPub won’t work for you. Instead, you can turn your author profile page back on, and then use the option in your SEO plugin to noindex the author page. This will enable the page to be live and ActivityPub will now work, but the live page won’t cause any duplicate content issues with search engines.
|
1. The blog-wide profile is only compatible with sites with rewrite rules enabled. If your site does not have rewrite rules enabled, the author-specific profiles may still work.
|
||||||
1. Once ActivityPub is installed, only new posts going forward will be available in the fediverse. Likewise, even if you’ve been using ActivityPub for a while, anyone who follows your site, will only see new posts you publish from that moment on. They will never see previously-published posts in their Home feed. This process is very similar to subscribing to a newsletter. If you subscribe to a newsletter, you will only receive future emails, but not the old archived ones. With ActivityPub, if someone follows your site, they will only receive new blog posts you publish from then on.
|
1. Many single-author blogs have chosen to turn off or redirect their author profile pages, usually via an SEO plugin like Yoast or Rank Math. This is usually done to avoid duplicate content with your blog’s home page. If your author page has been deactivated in this way, then ActivityPub author profiles won’t work for you. Instead, you can turn your author profile page back on, and then use the option in your SEO plugin to noindex the author page. This will duplicate content issues with search engines and will enable ActivityPub author profiles to work.
|
||||||
|
1. Once ActivityPub is installed, *only new posts going forward* will be available in the fediverse. Likewise, even if you’ve been using ActivityPub for a while, anyone who follows your site, will only see new posts you publish from that moment on. They will never see previously-published posts in their Home feed. This process is very similar to subscribing to a newsletter. If you subscribe to a newsletter, you will only receive future emails, but not the old archived ones. With ActivityPub, if someone follows your site, they will only receive new blog posts you publish from then on.
|
||||||
|
|
||||||
So what’s the process?
|
So what’s the process?
|
||||||
|
|
||||||
1. Install the ActivityPub plugin.
|
1. Install the ActivityPub plugin.
|
||||||
1. Go to the plugin’s settings page and adjust the settings to your liking. Click the Save button when ready.
|
1. Go to the plugin’s settings page and adjust the settings to your liking. Click the Save button when ready.
|
||||||
1. Make sure your blog’s author profile page is active.
|
1. Make sure your blog’s author profile page is active if you are using author profiles.
|
||||||
1. Go to Mastodon or any other federated platform, search for your author’s new federated profile, and follow it. Your new profile will be in the form of @yourauthorname@yourwebsite.com, so that is what you’ll search for.
|
1. Go to Mastodon or any other federated platform, and search for your profile, and follow it. Your new profile will be in the form of either `@your_username@example.com` or `@example.com@example.com`, so that is what you’ll search for.
|
||||||
1. On your blog, publish a new post.
|
1. On your blog, publish a new post.
|
||||||
1. From Mastodon, check to see if the new post appears in your Home feed.
|
1. From Mastodon, check to see if the new post appears in your Home feed.
|
||||||
|
|
||||||
|
@ -54,22 +54,14 @@ Please note that it may take up to 15 minutes or so for the new post to show up
|
||||||
|
|
||||||
= tl;dr =
|
= tl;dr =
|
||||||
|
|
||||||
This plugin connects your WordPress blog to popular social platforms like Mastodon, making your posts more accessible to a wider audience. Once installed, your blog's author pages can be followed by users on these platforms, allowing them to receive your new posts in their feeds.
|
This plugin connects your WordPress blog to popular social platforms like Mastodon, making your posts more accessible to a wider audience. Once installed, your blog can be followed by users on these platforms, allowing them to receive your new posts in their feeds.
|
||||||
|
|
||||||
Here's how it works:
|
|
||||||
|
|
||||||
1. Install the plugin and adjust settings as needed.
|
|
||||||
1. Ensure your blog's author profile page is active.
|
|
||||||
1. On Mastodon or other supported platforms, search for and follow your author's new profile (e.g., `@yourauthorname@yourwebsite.com`).
|
|
||||||
1. Publish a new post on your blog and check if it appears in your Mastodon feed.
|
|
||||||
|
|
||||||
Please note that it may take up to 15 minutes for a new post to appear in your feed, as messages are sent on a delay to avoid overwhelming your followers. Be patient and give it some time.
|
|
||||||
|
|
||||||
= What is the status of this plugin? =
|
= What is the status of this plugin? =
|
||||||
|
|
||||||
Implemented:
|
Implemented:
|
||||||
|
|
||||||
* profile pages (JSON representation)
|
* blog profile pages (JSON representation)
|
||||||
|
* author profile pages (JSON representation)
|
||||||
* custom links
|
* custom links
|
||||||
* functional inbox/outbox
|
* functional inbox/outbox
|
||||||
* follow (accept follows)
|
* follow (accept follows)
|
||||||
|
@ -79,8 +71,8 @@ Implemented:
|
||||||
|
|
||||||
To implement:
|
To implement:
|
||||||
|
|
||||||
* better configuration possibilities
|
|
||||||
* threaded comments support
|
* threaded comments support
|
||||||
|
* replace shortcodes with blocks for layout
|
||||||
|
|
||||||
= What is "ActivityPub for WordPress" =
|
= What is "ActivityPub for WordPress" =
|
||||||
|
|
||||||
|
@ -94,7 +86,7 @@ In order for webfinger to work, it must be mapped to the root directory of the U
|
||||||
|
|
||||||
Add the following to the .htaccess file in the root directory:
|
Add the following to the .htaccess file in the root directory:
|
||||||
|
|
||||||
RedirectMatch "^\/\.well-known/(webfinger|nodeinfo|x-nodeinfo2)(.*)$" "\/blog\/\.well-known$1$2"
|
RedirectMatch "^\/\.well-known/(webfinger|nodeinfo|x-nodeinfo2)(.*)$" /blog/.well-known/$1$2
|
||||||
|
|
||||||
Where 'blog' is the path to the subdirectory at which your blog resides.
|
Where 'blog' is the path to the subdirectory at which your blog resides.
|
||||||
|
|
||||||
|
@ -113,20 +105,36 @@ Where 'blog' is the path to the subdirectory at which your blog resides.
|
||||||
|
|
||||||
Project maintained on GitHub at [automattic/wordpress-activitypub](https://github.com/automattic/wordpress-activitypub).
|
Project maintained on GitHub at [automattic/wordpress-activitypub](https://github.com/automattic/wordpress-activitypub).
|
||||||
|
|
||||||
|
= 1.0.1 =
|
||||||
|
|
||||||
|
* Update: improve image attachment detection using the block editor
|
||||||
|
* Update: better error code handling for API responses
|
||||||
|
* Update: use a tag stack instead of regex for protecting tags for Hashtags and @-Mentions
|
||||||
|
* Compatibility: better signature support for subpath-installations
|
||||||
|
* Compatibility: allow deactivating blocks registered by the plugin
|
||||||
|
* Compatibility: avoid Fatal Errors when using ClassicPress
|
||||||
|
* Compatibility: improve the Group-Actor to play nicely with existing implementations
|
||||||
|
* Fixed: truncate long blog titles and handles for the "Follow me" block
|
||||||
|
* Fixed: ensure that only a valid user can be selected for the "Follow me" block
|
||||||
|
* Fixed: fix a typo in a hook name
|
||||||
|
* Fixed: a problem with signatures when running WordPress in a sub-path
|
||||||
|
|
||||||
= 1.0.0 =
|
= 1.0.0 =
|
||||||
|
|
||||||
* Add: blog-wide Account (catchall, like `mydomain.com@mydomain.com`)
|
* Add: blog-wide Account (catchall, like `example.com@example.com`)
|
||||||
* Add: Signature Verification: https://docs.joinmastodon.org/spec/security/ .
|
* Add: a Follow Me block (help visitors to follow your Profile)
|
||||||
* Add: a Followers Block.
|
* Add: Signature Verification: https://docs.joinmastodon.org/spec/security/
|
||||||
|
* Add: a Followers Block (show off your Followers)
|
||||||
* Add: Simple caching
|
* Add: Simple caching
|
||||||
* Add: Collection endpoints for Featured Tags and Featured Posts
|
* Add: Collection endpoints for Featured Tags and Featured Posts
|
||||||
* Update: Complete rewrite of the Follower-System based on Custom Post Types.
|
* Add: Better handling of Hashtags in mobile apps
|
||||||
|
* Update: Complete rewrite of the Follower-System based on Custom Post Types
|
||||||
* Update: Improved linter (PHPCS)
|
* Update: Improved linter (PHPCS)
|
||||||
* Compatibility: Add a new conditional, `\Activitypub\is_activitypub_request()`, to allow third-party plugins to detect ActivityPub requests.
|
* Compatibility: Add a new conditional, `\Activitypub\is_activitypub_request()`, to allow third-party plugins to detect ActivityPub requests
|
||||||
* Compatibility: Add hooks to allow modifying images returned in ActivityPub requests.
|
* Compatibility: Add hooks to allow modifying images returned in ActivityPub requests
|
||||||
* Compatibility: Indicate that the plugin is compatible and has been tested with the latest version of WordPress, 6.3.
|
* Compatibility: Indicate that the plugin is compatible and has been tested with the latest version of WordPress, 6.3
|
||||||
* Compatibility: Avoid PHP notice on sites using PHP 8.2.
|
* Compatibility: Avoid PHP notice on sites using PHP 8.2
|
||||||
* Fixed: Load the plugin later in the WordPress code lifecycle to avoid errors in some requests.
|
* Fixed: Load the plugin later in the WordPress code lifecycle to avoid errors in some requests
|
||||||
* Fixed: Updating posts
|
* Fixed: Updating posts
|
||||||
* Fixed: Hashtag now support CamelCase and UTF-8
|
* Fixed: Hashtag now support CamelCase and UTF-8
|
||||||
|
|
||||||
|
|
44
src/follow-me/block.json
Normal file
44
src/follow-me/block.json
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
{
|
||||||
|
"$schema": "https://schemas.wp.org/trunk/block.json",
|
||||||
|
"name": "activitypub/follow-me",
|
||||||
|
"apiVersion": 3,
|
||||||
|
"version": "1.0.0",
|
||||||
|
"title": "Follow me on the Fediverse",
|
||||||
|
"category": "widgets",
|
||||||
|
"description": "Display your Fediverse profile so that visitors can follow you.",
|
||||||
|
"textdomain": "activitypub",
|
||||||
|
"icon": "groups",
|
||||||
|
"supports": {
|
||||||
|
"html": false,
|
||||||
|
"color": {
|
||||||
|
"gradients": true,
|
||||||
|
"link": true,
|
||||||
|
"__experimentalDefaultControls": {
|
||||||
|
"background": true,
|
||||||
|
"text": true,
|
||||||
|
"link": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"__experimentalBorder": {
|
||||||
|
"radius": true,
|
||||||
|
"width": true,
|
||||||
|
"color": true,
|
||||||
|
"style": true
|
||||||
|
},
|
||||||
|
"typography": {
|
||||||
|
"fontSize": true,
|
||||||
|
"__experimentalDefaultControls": {
|
||||||
|
"fontSize": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"attributes": {
|
||||||
|
"selectedUser": {
|
||||||
|
"type": "string",
|
||||||
|
"default": "site"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"editorScript": "file:./index.js",
|
||||||
|
"viewScript": "file:./view.js",
|
||||||
|
"style": ["file:./style-index.css", "wp-components"]
|
||||||
|
}
|
75
src/follow-me/button-style.js
Normal file
75
src/follow-me/button-style.js
Normal file
|
@ -0,0 +1,75 @@
|
||||||
|
function presetVarColorCss( color ) {
|
||||||
|
return `var(--wp--preset--color--${ color })`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getBackgroundColor( color ) {
|
||||||
|
// if color is a string, it's a var like this.
|
||||||
|
if ( typeof color === 'string' ) {
|
||||||
|
return presetVarColorCss( color );
|
||||||
|
}
|
||||||
|
|
||||||
|
return color?.color?.background || null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLinkColor( text ) {
|
||||||
|
if ( typeof text !== 'string' ) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
// if it starts with a hash, leave it be
|
||||||
|
if ( text.match( /^#/ ) ) {
|
||||||
|
// we don't handle the alpha channel if present.
|
||||||
|
return text.substring( 0, 7 );
|
||||||
|
}
|
||||||
|
// var:preset|color|luminous-vivid-amber
|
||||||
|
// var(--wp--preset--color--luminous-vivid-amber)
|
||||||
|
// we will receive the top format, we need to output the bottom format
|
||||||
|
const [ , , color ] = text.split( '|' );
|
||||||
|
return presetVarColorCss( color );
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateSelector( selector, prop, value = null, pseudo = '' ) {
|
||||||
|
if ( ! value ) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
return `${ selector }${ pseudo } { ${ prop }: ${ value }; }\n`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getStyles( selector, button, text, hover ) {
|
||||||
|
return generateSelector( selector, 'background-color', button )
|
||||||
|
+ generateSelector( selector, 'color', text )
|
||||||
|
+ generateSelector( selector, 'background-color', hover, ':hover' )
|
||||||
|
+ generateSelector( selector, 'background-color', hover, ':focus' );
|
||||||
|
}
|
||||||
|
|
||||||
|
function getBlockStyles( base, style, backgroundColor ) {
|
||||||
|
const selector = `${ base } .components-button`;
|
||||||
|
// we grab the background color if set as a good color for our button text
|
||||||
|
const buttonTextColor = getBackgroundColor( backgroundColor )
|
||||||
|
// bg might be in this form.
|
||||||
|
|| style?.color?.background;
|
||||||
|
// we misuse the link color for the button background
|
||||||
|
const buttonColor = getLinkColor( style?.elements?.link?.color?.text );
|
||||||
|
// hover!
|
||||||
|
const buttonHoverColor = getLinkColor( style?.elements?.link?.[':hover']?.color?.text );
|
||||||
|
|
||||||
|
return getStyles( selector, buttonColor, buttonTextColor, buttonHoverColor );
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPopupStyles( style ) {
|
||||||
|
// we don't acept backgroundColor because the popup is always white (right?)
|
||||||
|
const buttonColor = getLinkColor( style?.elements?.link?.color?.text )
|
||||||
|
|| '#111';
|
||||||
|
const buttonTextColor = '#fff';
|
||||||
|
const buttonHoverColor = getLinkColor( style?.elements?.link?.[':hover']?.color?.text )
|
||||||
|
|| '#333';
|
||||||
|
const selector = '.apfmd__button-group .components-button';
|
||||||
|
|
||||||
|
return getStyles( selector, buttonColor, buttonTextColor, buttonHoverColor );
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ButtonStyle( { selector, style, backgroundColor } ) {
|
||||||
|
const css = getBlockStyles( selector, style, backgroundColor );
|
||||||
|
return (
|
||||||
|
<style>{ css }</style>
|
||||||
|
);
|
||||||
|
}
|
39
src/follow-me/edit.js
Normal file
39
src/follow-me/edit.js
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
import { InspectorControls, useBlockProps } from '@wordpress/block-editor';
|
||||||
|
import { __ } from '@wordpress/i18n';
|
||||||
|
import { SelectControl, PanelBody } from '@wordpress/components';
|
||||||
|
import { useUserOptions } from '../shared/use-user-options';
|
||||||
|
import FollowMe from './follow-me';
|
||||||
|
import { useEffect } from '@wordpress/element';
|
||||||
|
|
||||||
|
export default function Edit( { attributes, setAttributes } ) {
|
||||||
|
const blockProps = useBlockProps();
|
||||||
|
const usersOptions = useUserOptions();
|
||||||
|
const { selectedUser } = attributes;
|
||||||
|
|
||||||
|
useEffect( () => {
|
||||||
|
// if there are no users yet, do nothing
|
||||||
|
if ( ! usersOptions.length ) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// ensure that the selected user is in the list of options, if not, select the first available user
|
||||||
|
if ( ! usersOptions.find( ( { value } ) => value === selectedUser ) ) {
|
||||||
|
setAttributes( { selectedUser: usersOptions[ 0 ].value } );
|
||||||
|
}
|
||||||
|
}, [ selectedUser, usersOptions ] );
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div { ...blockProps }>
|
||||||
|
<InspectorControls key="setting">
|
||||||
|
<PanelBody title={ __( 'Followers Options', 'activitypub' ) }>
|
||||||
|
<SelectControl
|
||||||
|
label= { __( 'Select User', 'activitypub' ) }
|
||||||
|
value={ attributes.selectedUser }
|
||||||
|
options={ usersOptions }
|
||||||
|
onChange={ ( value ) => setAttributes( { selectedUser: value } ) }
|
||||||
|
/>
|
||||||
|
</PanelBody>
|
||||||
|
</InspectorControls>
|
||||||
|
<FollowMe { ...attributes } id={ blockProps.id } />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
178
src/follow-me/follow-me.js
Normal file
178
src/follow-me/follow-me.js
Normal file
|
@ -0,0 +1,178 @@
|
||||||
|
|
||||||
|
import apiFetch from '@wordpress/api-fetch';
|
||||||
|
import { useCallback, useEffect, useState, createInterpolateElement } from '@wordpress/element';
|
||||||
|
import { Button, Modal } from '@wordpress/components';
|
||||||
|
import { __, sprintf } from '@wordpress/i18n';
|
||||||
|
import { copy, check, Icon } from '@wordpress/icons';
|
||||||
|
import { useCopyToClipboard } from '@wordpress/compose';
|
||||||
|
import { ButtonStyle, getPopupStyles } from './button-style';
|
||||||
|
import './style.scss';
|
||||||
|
const { namespace } = window._activityPubOptions;
|
||||||
|
|
||||||
|
const DEFAULT_PROFILE_DATA = {
|
||||||
|
avatar: '',
|
||||||
|
resource: '@well@hello.dolly',
|
||||||
|
name: __( 'Hello Dolly Fan Account', 'activitypub' ),
|
||||||
|
url: '#',
|
||||||
|
};
|
||||||
|
|
||||||
|
function getNormalizedProfile( profile ) {
|
||||||
|
if ( ! profile ) {
|
||||||
|
return DEFAULT_PROFILE_DATA;
|
||||||
|
}
|
||||||
|
const data = { ...DEFAULT_PROFILE_DATA, ...profile };
|
||||||
|
data.avatar = data?.icon?.url;
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
function fetchProfile( userId ) {
|
||||||
|
const fetchOptions = {
|
||||||
|
headers: { Accept: 'application/activity+json' },
|
||||||
|
path: `/${ namespace }/users/${ userId }`,
|
||||||
|
};
|
||||||
|
return apiFetch( fetchOptions );
|
||||||
|
}
|
||||||
|
|
||||||
|
function Profile( { profile, popupStyles, userId } ) {
|
||||||
|
const { avatar, name, resource } = profile;
|
||||||
|
return (
|
||||||
|
<div className="activitypub-profile">
|
||||||
|
<img className="activitypub-profile__avatar" src={ avatar } />
|
||||||
|
<div className="activitypub-profile__content">
|
||||||
|
<div className="activitypub-profile__name">{ name }</div>
|
||||||
|
<div className="activitypub-profile__handle">{ resource }</div>
|
||||||
|
</div>
|
||||||
|
<Follow profile={ profile } popupStyles={ popupStyles } userId={ userId } />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function Follow( { profile, popupStyles, userId } ) {
|
||||||
|
const [ isOpen, setIsOpen ] = useState( false );
|
||||||
|
const title = sprintf( __( 'Follow %s', 'activitypub' ), profile?.name );
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Button className="activitypub-profile__follow" onClick={ () => setIsOpen( true ) } >
|
||||||
|
{ __( 'Follow', 'activitypub' ) }
|
||||||
|
</Button>
|
||||||
|
{ isOpen && (
|
||||||
|
<Modal
|
||||||
|
className="activitypub-profile__confirm"
|
||||||
|
onRequestClose={ () => setIsOpen( false ) }
|
||||||
|
title={ title }
|
||||||
|
>
|
||||||
|
<Dialog profile={ profile } userId={ userId } />
|
||||||
|
<style>{ popupStyles }</style>
|
||||||
|
</Modal>
|
||||||
|
) }
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function isUrl( string ) {
|
||||||
|
try {
|
||||||
|
new URL( string );
|
||||||
|
return true;
|
||||||
|
} catch ( _ ) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function isHandle( string ) {
|
||||||
|
// remove leading @, there should still be an @ in there
|
||||||
|
const parts = string.replace( /^@/, '' ).split( '@' );
|
||||||
|
return parts.length === 2 && isUrl( `https://${ parts[ 1 ] }` );
|
||||||
|
}
|
||||||
|
|
||||||
|
function Dialog( { profile, userId } ) {
|
||||||
|
const { resource } = profile;
|
||||||
|
const followText = __( 'Follow', 'activitypub' );
|
||||||
|
const loadingText = __( 'Loading...', 'activitypub' );
|
||||||
|
const openingText = __( 'Opening...', 'activitypub' );
|
||||||
|
const errorText = __( 'Error', 'activitypub' );
|
||||||
|
const invalidText = __( 'Invalid', 'activitypub' );
|
||||||
|
const [ buttonText, setButtonText ] = useState( followText );
|
||||||
|
const [ buttonIcon, setButtonIcon ] = useState( copy );
|
||||||
|
const ref = useCopyToClipboard( resource, () => {
|
||||||
|
setButtonIcon( check );
|
||||||
|
setTimeout( () => setButtonIcon( copy ), 1000 );
|
||||||
|
} );
|
||||||
|
const [ remoteProfile, setRemoteProfile ] = useState( '' );
|
||||||
|
const retrieveAndFollow = useCallback( () => {
|
||||||
|
let timeout;
|
||||||
|
if ( ! ( isUrl( remoteProfile ) || isHandle( remoteProfile ) ) ) {
|
||||||
|
setButtonText( invalidText );
|
||||||
|
timeout = setTimeout( () => setButtonText( followText ), 2000 );
|
||||||
|
return () => clearTimeout( timeout );
|
||||||
|
}
|
||||||
|
const path = `/${ namespace }/users/${userId}/remote-follow?resource=${ remoteProfile }`;
|
||||||
|
setButtonText( loadingText );
|
||||||
|
apiFetch( { path } ).then( ( { url } ) => {
|
||||||
|
setButtonText( openingText );
|
||||||
|
setTimeout( () => {
|
||||||
|
window.open( url, '_blank' );
|
||||||
|
setButtonText( followText );
|
||||||
|
}, 200 );
|
||||||
|
} ).catch( () => {
|
||||||
|
setButtonText( errorText );
|
||||||
|
setTimeout( () => setButtonText( followText ), 2000 );
|
||||||
|
} );
|
||||||
|
}, [ remoteProfile ] );
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="activitypub-follow-me__dialog">
|
||||||
|
<div className="apmfd__section">
|
||||||
|
<h4>{ __( 'My Profile', 'activitypub' ) }</h4>
|
||||||
|
<div className="apfmd-description">
|
||||||
|
{ __( 'Copy and paste my profile into the search field of your favorite fediverse app or server.', 'activitypub' ) }
|
||||||
|
</div>
|
||||||
|
<div className="apfmd__button-group">
|
||||||
|
<input type="text" value={ resource } readOnly />
|
||||||
|
<Button ref={ ref }>
|
||||||
|
<Icon icon={ buttonIcon } />
|
||||||
|
{ __( 'Copy', 'activitypub' ) }
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="apmfd__section">
|
||||||
|
<h4>{ __( 'Your Profile', 'activitypub' ) }</h4>
|
||||||
|
<div className="apfmd-description">
|
||||||
|
{ createInterpolateElement(
|
||||||
|
__( 'Or, if you know your own profile, we can start things that way! (eg <code>https://example.com/yourusername</code> or <code>yourusername@example.com</code>)', 'activitypub' ),
|
||||||
|
{ code: <code /> }
|
||||||
|
) }
|
||||||
|
</div>
|
||||||
|
<div className="apfmd__button-group">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={ remoteProfile }
|
||||||
|
onKeyDown={ ( event ) => { event?.code === 'Enter' && retrieveAndFollow() } }
|
||||||
|
onChange={ e => setRemoteProfile( e.target.value ) }
|
||||||
|
/>
|
||||||
|
<Button onClick={ retrieveAndFollow }>{ buttonText }</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function FollowMe( { selectedUser, style, backgroundColor, id, useId = false } ) {
|
||||||
|
const [ profile, setProfile ] = useState( getNormalizedProfile() );
|
||||||
|
const userId = selectedUser === 'site' ? 0 : selectedUser;
|
||||||
|
const popupStyles = getPopupStyles( style );
|
||||||
|
const wrapperProps = useId ? { id } : {};
|
||||||
|
function setProfileData( profile ) {
|
||||||
|
setProfile( getNormalizedProfile( profile ) );
|
||||||
|
}
|
||||||
|
useEffect( () => {
|
||||||
|
fetchProfile( userId ).then( setProfileData );
|
||||||
|
}, [ userId ] );
|
||||||
|
|
||||||
|
return(
|
||||||
|
<div { ...wrapperProps }>
|
||||||
|
<ButtonStyle selector={ `#${ id }` } style={ style } backgroundColor={ backgroundColor } />
|
||||||
|
<Profile profile={ profile } userId={ userId } popupStyles={ popupStyles } />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
5
src/follow-me/index.js
Normal file
5
src/follow-me/index.js
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
import { registerBlockType } from '@wordpress/blocks';
|
||||||
|
import { people } from '@wordpress/icons';
|
||||||
|
import edit from './edit';
|
||||||
|
const save = () => null;
|
||||||
|
registerBlockType( 'activitypub/follow-me', { edit, save, icon: people } );
|
67
src/follow-me/style.scss
Normal file
67
src/follow-me/style.scss
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
.editor-styles-wrapper, .activitypub-follow-me-block-wrapper {
|
||||||
|
.activitypub-profile {
|
||||||
|
display: flex;
|
||||||
|
align-items: self-start;
|
||||||
|
padding: 1rem;
|
||||||
|
|
||||||
|
.activitypub-profile__avatar {
|
||||||
|
height: 75px;
|
||||||
|
width: 75px;
|
||||||
|
margin-right: 1rem;
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
.activitypub-profile__content {
|
||||||
|
flex: 1;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
.activitypub-profile__name, .activitypub-profile__handle {
|
||||||
|
margin: 0;
|
||||||
|
line-height: 1.2;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
.activitypub-profile__name {
|
||||||
|
font-size: 1.25em;
|
||||||
|
}
|
||||||
|
.activitypub-profile__follow {
|
||||||
|
margin-left: 1rem;
|
||||||
|
align-self: center;
|
||||||
|
background-color: var(--wp--preset--color--black);
|
||||||
|
color: var(--wp--preset--color--white);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.activitypub-follow-me__dialog {
|
||||||
|
max-width: 30em;
|
||||||
|
h4 {
|
||||||
|
line-height: 1;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
.apmfd__section {
|
||||||
|
margin-bottom: 2em;
|
||||||
|
}
|
||||||
|
.apfmd-description {
|
||||||
|
font-size: var( --wp--preset--font-size--normal, .75rem );
|
||||||
|
margin: 0.33em 0 1em;
|
||||||
|
}
|
||||||
|
.apfmd__button-group {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
margin-right: .5em;
|
||||||
|
height: 21px;
|
||||||
|
width: 21px;
|
||||||
|
}
|
||||||
|
|
||||||
|
input {
|
||||||
|
flex: 1;
|
||||||
|
padding: {
|
||||||
|
left: 1em;
|
||||||
|
right: 1em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
16
src/follow-me/view.js
Normal file
16
src/follow-me/view.js
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
import { render } from '@wordpress/element';
|
||||||
|
import domReady from '@wordpress/dom-ready';
|
||||||
|
import FollowMe from './follow-me';
|
||||||
|
|
||||||
|
let id = 1;
|
||||||
|
function getUniqueId() {
|
||||||
|
return `activitypub-follow-me-block-${ id++ }`;
|
||||||
|
}
|
||||||
|
|
||||||
|
domReady( () => {
|
||||||
|
// iterate over a nodelist
|
||||||
|
[].forEach.call( document.querySelectorAll( '.activitypub-follow-me-block-wrapper' ), ( element ) => {
|
||||||
|
const attrs = JSON.parse( element.dataset.attrs );
|
||||||
|
render( <FollowMe { ...attrs } id={ getUniqueId() } useId={ true } />, element );
|
||||||
|
} );
|
||||||
|
} );
|
|
@ -1,31 +1,9 @@
|
||||||
import { SelectControl, RangeControl, PanelBody, TextControl } from '@wordpress/components';
|
import { SelectControl, RangeControl, PanelBody, TextControl } from '@wordpress/components';
|
||||||
import { useSelect } from '@wordpress/data';
|
import { useState, useEffect } from '@wordpress/element';
|
||||||
import { useMemo, useState } from '@wordpress/element';
|
|
||||||
import { InspectorControls, useBlockProps } from '@wordpress/block-editor';
|
import { InspectorControls, useBlockProps } from '@wordpress/block-editor';
|
||||||
import { __ } from '@wordpress/i18n';
|
import { __ } from '@wordpress/i18n';
|
||||||
import { Followers } from './followers';
|
import { Followers } from './followers';
|
||||||
|
import { useUserOptions } from '../shared/use-user-options';
|
||||||
const enabled = window._activityPubOptions?.enabled;
|
|
||||||
|
|
||||||
function useUserOptions() {
|
|
||||||
const users = enabled?.users ? useSelect( ( select ) => select( 'core' ).getUsers( { who: 'authors' } ) ) : [];
|
|
||||||
return useMemo( () => {
|
|
||||||
if ( ! users ) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
const withBlogUser = enabled?.site ? [ {
|
|
||||||
label: __( 'Whole Site', 'activitypub' ),
|
|
||||||
value: 'site'
|
|
||||||
} ] : [];
|
|
||||||
return users.reduce( ( acc, user ) => {
|
|
||||||
acc.push({
|
|
||||||
label: user.name,
|
|
||||||
value: user.id
|
|
||||||
} );
|
|
||||||
return acc;
|
|
||||||
}, withBlogUser );
|
|
||||||
}, [ users ] );
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function Edit( { attributes, setAttributes } ) {
|
export default function Edit( { attributes, setAttributes } ) {
|
||||||
const { order, per_page, selectedUser, title } = attributes;
|
const { order, per_page, selectedUser, title } = attributes;
|
||||||
|
@ -43,6 +21,17 @@ export default function Edit( { attributes, setAttributes } ) {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
useEffect( () => {
|
||||||
|
// if there are no users yet, do nothing
|
||||||
|
if ( ! usersOptions.length ) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// ensure that the selected user is in the list of options, if not, select the first available user
|
||||||
|
if ( ! usersOptions.find( ( { value } ) => value === selectedUser ) ) {
|
||||||
|
setAttributes( { selectedUser: usersOptions[ 0 ].value } );
|
||||||
|
}
|
||||||
|
}, [ selectedUser, usersOptions ] );
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div { ...blockProps }>
|
<div { ...blockProps }>
|
||||||
<InspectorControls key="setting">
|
<InspectorControls key="setting">
|
||||||
|
|
|
@ -64,7 +64,7 @@ export function Followers( {
|
||||||
setTotal( data.totalItems );
|
setTotal( data.totalItems );
|
||||||
setFollowers( data.orderedItems );
|
setFollowers( data.orderedItems );
|
||||||
} )
|
} )
|
||||||
.catch( ( error ) => console.error( error ) );
|
.catch( () => {} );
|
||||||
}, [ userId, per_page, order, page ] );
|
}, [ userId, per_page, order, page ] );
|
||||||
return (
|
return (
|
||||||
<div className={ "activitypub-follower-block " + className }>
|
<div className={ "activitypub-follower-block " + className }>
|
||||||
|
|
24
src/shared/use-user-options.js
Normal file
24
src/shared/use-user-options.js
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
import { __ } from '@wordpress/i18n';
|
||||||
|
import { useSelect } from '@wordpress/data';
|
||||||
|
import { useMemo } from '@wordpress/element';
|
||||||
|
const enabled = window._activityPubOptions?.enabled;
|
||||||
|
|
||||||
|
export function useUserOptions() {
|
||||||
|
const users = enabled?.users ? useSelect( ( select ) => select( 'core' ).getUsers( { who: 'authors' } ) ) : [];
|
||||||
|
return useMemo( () => {
|
||||||
|
if ( ! users ) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
const withBlogUser = enabled?.site ? [ {
|
||||||
|
label: __( 'Whole Site', 'activitypub' ),
|
||||||
|
value: 'site'
|
||||||
|
} ] : [];
|
||||||
|
return users.reduce( ( acc, user ) => {
|
||||||
|
acc.push({
|
||||||
|
label: user.name,
|
||||||
|
value: `${ user.id }` // casting to string because that's how the attribute is stored by Gutenberg
|
||||||
|
} );
|
||||||
|
return acc;
|
||||||
|
}, withBlogUser );
|
||||||
|
}, [ users ] );
|
||||||
|
}
|
|
@ -6,7 +6,7 @@
|
||||||
<h1><?php \esc_html_e( 'ActivityPub', 'activitypub' ); ?></h1>
|
<h1><?php \esc_html_e( 'ActivityPub', 'activitypub' ); ?></h1>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<nav class="activitypub-settings-tabs-wrapper hide-if-no-js" aria-label="<?php \esc_attr_e( 'Secondary menu', 'activitypub' ); ?>">
|
<nav class="activitypub-settings-tabs-wrapper" aria-label="<?php \esc_attr_e( 'Secondary menu', 'activitypub' ); ?>">
|
||||||
<a href="<?php echo \esc_url_raw( admin_url( 'options-general.php?page=activitypub' ) ); ?>" class="activitypub-settings-tab <?php echo \esc_attr( $args['welcome'] ); ?>">
|
<a href="<?php echo \esc_url_raw( admin_url( 'options-general.php?page=activitypub' ) ); ?>" class="activitypub-settings-tab <?php echo \esc_attr( $args['welcome'] ); ?>">
|
||||||
<?php \esc_html_e( 'Welcome', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Welcome', 'activitypub' ); ?>
|
||||||
</a>
|
</a>
|
||||||
|
|
|
@ -8,15 +8,13 @@
|
||||||
'followers' => 'active',
|
'followers' => 'active',
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
$table = new \Activitypub\Table\Followers();
|
||||||
|
$follower_count = $table->get_user_count();
|
||||||
|
// translators: The follower count.
|
||||||
|
$followers_template = _n( 'Your blog profile currently has %s follower.', 'Your blog profile currently has %s followers.', $follower_count, 'activitypub' );
|
||||||
?>
|
?>
|
||||||
|
|
||||||
<div class="wrap activitypub-followers-page">
|
<div class="wrap activitypub-followers-page">
|
||||||
<h1><?php \esc_html_e( 'Followers', 'activitypub' ); ?></h1>
|
<p><?php \printf( \esc_html( $followers_template ), \esc_attr( $follower_count ) ); ?></p>
|
||||||
|
|
||||||
<?php $table = new \Activitypub\Table\Followers(); ?>
|
|
||||||
|
|
||||||
<?php // translators: The follower count. ?>
|
|
||||||
<p><?php \printf( \esc_html__( 'You currently have %s followers.', 'activitypub' ), \esc_attr( $table->get_user_count() ) ); ?></p>
|
|
||||||
|
|
||||||
<form method="get">
|
<form method="get">
|
||||||
<input type="hidden" name="page" value="activitypub" />
|
<input type="hidden" name="page" value="activitypub" />
|
||||||
|
|
|
@ -11,60 +11,41 @@
|
||||||
?>
|
?>
|
||||||
|
|
||||||
<div class="activitypub-settings activitypub-settings-page hide-if-no-js">
|
<div class="activitypub-settings activitypub-settings-page hide-if-no-js">
|
||||||
<div class="box">
|
|
||||||
<h3><?php \esc_html_e( 'Troubleshooting', 'activitypub' ); ?></h3>
|
|
||||||
<p>
|
|
||||||
<?php
|
|
||||||
echo \wp_kses(
|
|
||||||
\sprintf(
|
|
||||||
// translators:
|
|
||||||
\__( 'If you have problems using this plugin, please check the <a href="%s">Site Health</a> to ensure that your site is compatible and/or use the "Help" tab (in the top right of the settings pages).', 'activitypub' ),
|
|
||||||
\esc_url_raw( \admin_url( 'site-health.php' ) )
|
|
||||||
),
|
|
||||||
'default'
|
|
||||||
);
|
|
||||||
?>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<form method="post" action="options.php">
|
<form method="post" action="options.php">
|
||||||
<?php \settings_fields( 'activitypub' ); ?>
|
<?php \settings_fields( 'activitypub' ); ?>
|
||||||
|
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3><?php \esc_html_e( 'Users', 'activitypub' ); ?></h3>
|
<h3><?php \esc_html_e( 'Profiles', 'activitypub' ); ?></h3>
|
||||||
|
|
||||||
<p><?php \esc_html_e( 'All user related settings.', 'activitypub' ); ?></p>
|
|
||||||
|
|
||||||
<table class="form-table">
|
<table class="form-table">
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row">
|
<th scope="row">
|
||||||
<?php \esc_html_e( 'Enable/disable Users by Type', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Enable profiles by type', 'activitypub' ); ?>
|
||||||
</th>
|
</th>
|
||||||
<td>
|
<td>
|
||||||
<p>
|
<p>
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" name="activitypub_enable_users" id="activitypub_enable_users" value="1" <?php echo \checked( '1', \get_option( 'activitypub_enable_users', '1' ) ); ?> />
|
<input type="checkbox" name="activitypub_enable_users" id="activitypub_enable_users" value="1" <?php echo \checked( '1', \get_option( 'activitypub_enable_users', '1' ) ); ?> />
|
||||||
<?php \esc_html_e( 'Enable Authors', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Enable authors', 'activitypub' ); ?>
|
||||||
</label>
|
</label>
|
||||||
</p>
|
</p>
|
||||||
<p class="description">
|
<p class="description">
|
||||||
<?php echo \wp_kses( \__( 'Every Author on this Blog (with the <code>publish_posts</code> capability) gets his own ActivityPub enabled Profile.', 'activitypub' ), array( 'code' => array() ) ); ?>
|
<?php echo \wp_kses( \__( 'Every author on this blog (with the <code>publish_posts</code> capability) gets their own ActivityPub profile.', 'activitypub' ), array( 'code' => array() ) ); ?>
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<label>
|
<label>
|
||||||
<input type="checkbox" name="activitypub_enable_blog_user" id="activitypub_enable_blog_user" value="1" <?php echo \checked( '1', \get_option( 'activitypub_enable_blog_user', '0' ) ); ?> />
|
<input type="checkbox" name="activitypub_enable_blog_user" id="activitypub_enable_blog_user" value="1" <?php echo \checked( '1', \get_option( 'activitypub_enable_blog_user', '0' ) ); ?> />
|
||||||
<?php \esc_html_e( 'Enable Blog-User', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Enable blog', 'activitypub' ); ?>
|
||||||
</label>
|
</label>
|
||||||
</p>
|
</p>
|
||||||
<p class="description">
|
<p class="description">
|
||||||
<?php \esc_html_e( 'Your Blog becomes an ActivityPub compatible Profile.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Your blog becomes an ActivityPub profile.', 'activitypub' ); ?>
|
||||||
</p>
|
</p>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row">
|
<th scope="row">
|
||||||
<?php \esc_html_e( 'Change Blog-User Identifier', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Change blog profile ID', 'activitypub' ); ?>
|
||||||
</th>
|
</th>
|
||||||
<td>
|
<td>
|
||||||
<label for="activitypub_blog_user_identifier">
|
<label for="activitypub_blog_user_identifier">
|
||||||
|
@ -72,7 +53,12 @@
|
||||||
@<?php echo esc_html( \wp_parse_url( \home_url(), PHP_URL_HOST ) ); ?>
|
@<?php echo esc_html( \wp_parse_url( \home_url(), PHP_URL_HOST ) ); ?>
|
||||||
</label>
|
</label>
|
||||||
<p class="description">
|
<p class="description">
|
||||||
<?php \esc_html_e( 'This Blog-User will federate all posts written on your Blog, regardless of the User who posted it.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'This profile name will federate all posts written on your blog, regardless of the author who posted it.', 'activitypub' ); ?>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<strong>
|
||||||
|
<?php \esc_html_e( 'Please avoid using an existing author’s name as the blog profile ID. Fediverse platforms might use caching and this could break the functionality completely.', 'activitypub' ); ?>
|
||||||
|
</strong>
|
||||||
</p>
|
</p>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
@ -84,14 +70,11 @@
|
||||||
|
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3><?php \esc_html_e( 'Activities', 'activitypub' ); ?></h3>
|
<h3><?php \esc_html_e( 'Activities', 'activitypub' ); ?></h3>
|
||||||
|
|
||||||
<p><?php \esc_html_e( 'All activity related settings.', 'activitypub' ); ?></p>
|
|
||||||
|
|
||||||
<table class="form-table">
|
<table class="form-table">
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row">
|
<th scope="row">
|
||||||
<?php \esc_html_e( 'Post-Content', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Post content', 'activitypub' ); ?>
|
||||||
</th>
|
</th>
|
||||||
<td>
|
<td>
|
||||||
<p>
|
<p>
|
||||||
|
@ -130,7 +113,7 @@
|
||||||
<?php \esc_html_e( 'Custom', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Custom', 'activitypub' ); ?>
|
||||||
-
|
-
|
||||||
<span class="description">
|
<span class="description">
|
||||||
<?php \esc_html_e( 'Use the text-area below, to customize your activities.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Use the text area below, to customize your activities.', 'activitypub' ); ?>
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
</p>
|
</p>
|
||||||
|
@ -237,7 +220,7 @@
|
||||||
</th>
|
</th>
|
||||||
<td>
|
<td>
|
||||||
<p>
|
<p>
|
||||||
<label><input type="checkbox" name="activitypub_use_hashtags" id="activitypub_use_hashtags" value="1" <?php echo \checked( '1', \get_option( 'activitypub_use_hashtags', '1' ) ); ?> /> <?php echo wp_kses( \__( 'Add hashtags in the content as native tags and replace the <code>#tag</code> with the tag-link. <strong>This feature is experimental! Please disable it, if you find any HTML or CSS errors.</strong>', 'activitypub' ), 'default' ); ?></label>
|
<label><input type="checkbox" name="activitypub_use_hashtags" id="activitypub_use_hashtags" value="1" <?php echo \checked( '1', \get_option( 'activitypub_use_hashtags', '1' ) ); ?> /> <?php echo wp_kses( \__( 'Add hashtags in the content as native tags and replace the <code>#tag</code> with the tag link. <strong>This feature is experimental! Please disable it, if you find any HTML or CSS errors.</strong>', 'activitypub' ), 'default' ); ?></label>
|
||||||
</p>
|
</p>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
@ -249,9 +232,6 @@
|
||||||
|
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3><?php \esc_html_e( 'Server', 'activitypub' ); ?></h3>
|
<h3><?php \esc_html_e( 'Server', 'activitypub' ); ?></h3>
|
||||||
|
|
||||||
<p><?php \esc_html_e( 'Server related settings.', 'activitypub' ); ?></p>
|
|
||||||
|
|
||||||
<table class="form-table">
|
<table class="form-table">
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
|
|
|
@ -1,7 +1,12 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
$follower_count = \Activitypub\Collection\Followers::count_followers( \get_current_user_id() );
|
||||||
|
// translators: The follower count.
|
||||||
|
$followers_template = _n( 'Your author profile currently has %s follower.', 'Your author profile currently has %s followers.', $follower_count, 'activitypub' );
|
||||||
|
?>
|
||||||
<div class="wrap">
|
<div class="wrap">
|
||||||
<h1><?php \esc_html_e( 'Followers', 'activitypub' ); ?></h1>
|
<h1><?php \esc_html_e( 'Author Followers', 'activitypub' ); ?></h1>
|
||||||
<?php // translators: The follower count. ?>
|
<p><?php \printf( \esc_html( $followers_template ), \esc_attr( $follower_count ) ); ?></p>
|
||||||
<p><?php \printf( \esc_html__( 'You currently have %s followers.', 'activitypub' ), \esc_attr( \Activitypub\Collection\Followers::count_followers( \get_current_user_id() ) ) ); ?></p>
|
|
||||||
|
|
||||||
<?php $table = new \Activitypub\Table\Followers(); ?>
|
<?php $table = new \Activitypub\Table\Followers(); ?>
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@ $user = \Activitypub\Collection\Users::get_by_id( \get_current_user_id() ); ?>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<th scope="row">
|
<th scope="row">
|
||||||
<label><?php \esc_html_e( 'Profile identifier', 'activitypub' ); ?></label>
|
<label><?php \esc_html_e( 'Profile URL', 'activitypub' ); ?></label>
|
||||||
</th>
|
</th>
|
||||||
<td>
|
<td>
|
||||||
<p>
|
<p>
|
||||||
|
@ -15,7 +15,7 @@ $user = \Activitypub\Collection\Users::get_by_id( \get_current_user_id() ); ?>
|
||||||
<code><?php echo \esc_url( $user->get_url() ); ?></code>
|
<code><?php echo \esc_url( $user->get_url() ); ?></code>
|
||||||
</p>
|
</p>
|
||||||
<?php // translators: the webfinger resource ?>
|
<?php // translators: the webfinger resource ?>
|
||||||
<p class="description"><?php \printf( \esc_html__( 'Try to follow "@%s" by searching for it on Mastodon,Friendica & Co.', 'activitypub' ), \esc_html( $user->get_resource() ) ); ?></p>
|
<p class="description"><?php \printf( \esc_html__( 'Follow "@%s" by searching for it on Mastodon, Friendica, etc.', 'activitypub' ), \esc_html( $user->get_resource() ) ); ?></p>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="activitypub-user-description-wrap">
|
<tr class="activitypub-user-description-wrap">
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h2><?php \esc_html_e( 'Welcome', 'activitypub' ); ?></h2>
|
<h2><?php \esc_html_e( 'Welcome', 'activitypub' ); ?></h2>
|
||||||
|
|
||||||
<p><?php echo wp_kses( \__( 'With ActivityPub your blog becomes part of a federated social network. This means you can share and talk to everyone using the <strong>ActivityPub</strong> protocol, including users of <strong>Friendica</strong>, <strong>Pleroma</strong>, <strong>Pixelfed</strong> and <strong>Mastodon</strong>.', 'activitypub' ), array( 'strong' => array() ) ); ?></p>
|
<p><?php echo wp_kses( \__( 'Enter the fediverse with <strong>ActivityPub</strong>, broadcasting your blog to a wider audience. Attract followers, deliver updates, and receive comments from a diverse user base on <strong>Mastodon</strong>, <strong>Friendica</strong>, <strong>Pleroma</strong>, <strong>Pixelfed</strong>, and all <strong>ActivityPub</strong>-compliant platforms.', 'activitypub' ), array( 'strong' => array() ) ); ?></p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<?php
|
<?php
|
||||||
|
@ -22,9 +22,9 @@
|
||||||
$blog_user = new \Activitypub\Model\Blog_User();
|
$blog_user = new \Activitypub\Model\Blog_User();
|
||||||
?>
|
?>
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3><?php \esc_html_e( 'Blog Account', 'activitypub' ); ?></h3>
|
<h3><?php \esc_html_e( 'Blog profile', 'activitypub' ); ?></h3>
|
||||||
<p>
|
<p>
|
||||||
<?php \esc_html_e( 'People can follow your Blog by using:', 'activitypub' ); ?>
|
<?php \esc_html_e( 'People can follow your blog by using:', 'activitypub' ); ?>
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<label for="activitypub-blog-identifier"><?php \esc_html_e( 'Username', 'activitypub' ); ?></label>
|
<label for="activitypub-blog-identifier"><?php \esc_html_e( 'Username', 'activitypub' ); ?></label>
|
||||||
|
@ -33,17 +33,17 @@
|
||||||
<input type="text" class="regular-text" id="activitypub-blog-identifier" value="<?php echo \esc_attr( $blog_user->get_resource() ); ?>" readonly />
|
<input type="text" class="regular-text" id="activitypub-blog-identifier" value="<?php echo \esc_attr( $blog_user->get_resource() ); ?>" readonly />
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<label for="activitypub-blog-url"><?php \esc_html_e( 'Profile-URL', 'activitypub' ); ?></label>
|
<label for="activitypub-blog-url"><?php \esc_html_e( 'Profile URL', 'activitypub' ); ?></label>
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<input type="text" class="regular-text" id="activitypub-blog-url" value="<?php echo \esc_attr( $blog_user->get_url() ); ?>" readonly />
|
<input type="text" class="regular-text" id="activitypub-blog-url" value="<?php echo \esc_attr( $blog_user->get_url() ); ?>" readonly />
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<?php \esc_html_e( 'This Blog-User will federate all posts written on your Blog, regardless of the User who posted it.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'This blog profile will federate all posts written on your blog, regardless of the author who posted it.', 'activitypub' ); ?>
|
||||||
<p>
|
<p>
|
||||||
<p>
|
<p>
|
||||||
<a href="<?php echo \esc_url_raw( \admin_url( '/options-general.php?page=activitypub&tab=settings' ) ); ?>">
|
<a href="<?php echo \esc_url_raw( \admin_url( '/options-general.php?page=activitypub&tab=settings' ) ); ?>">
|
||||||
<?php \esc_html_e( 'Customize Blog-User on Settings page.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Customize the blog profile', 'activitypub' ); ?>
|
||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -54,9 +54,9 @@
|
||||||
$user = \Activitypub\Collection\Users::get_by_id( wp_get_current_user()->ID );
|
$user = \Activitypub\Collection\Users::get_by_id( wp_get_current_user()->ID );
|
||||||
?>
|
?>
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3><?php \esc_html_e( 'Personal Account', 'activitypub' ); ?></h3>
|
<h3><?php \esc_html_e( 'Author profile', 'activitypub' ); ?></h3>
|
||||||
<p>
|
<p>
|
||||||
<?php \esc_html_e( 'People can follow you by using your Username:', 'activitypub' ); ?>
|
<?php \esc_html_e( 'People can follow you by using your author name:', 'activitypub' ); ?>
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<label for="activitypub-user-identifier"><?php \esc_html_e( 'Username', 'activitypub' ); ?></label>
|
<label for="activitypub-user-identifier"><?php \esc_html_e( 'Username', 'activitypub' ); ?></label>
|
||||||
|
@ -65,17 +65,17 @@
|
||||||
<input type="text" class="regular-text" id="activitypub-user-identifier" value="<?php echo \esc_attr( $user->get_resource() ); ?>" readonly />
|
<input type="text" class="regular-text" id="activitypub-user-identifier" value="<?php echo \esc_attr( $user->get_resource() ); ?>" readonly />
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<label for="activitypub-user-url"><?php \esc_html_e( 'Profile-URL', 'activitypub' ); ?></label>
|
<label for="activitypub-user-url"><?php \esc_html_e( 'Profile URL', 'activitypub' ); ?></label>
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<input type="text" class="regular-text" id="activitypub-user-url" value="<?php echo \esc_attr( $user->get_url() ); ?>" readonly />
|
<input type="text" class="regular-text" id="activitypub-user-url" value="<?php echo \esc_attr( $user->get_url() ); ?>" readonly />
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
<?php \esc_html_e( 'Users who can not access this settings page will find their username on the "Edit Profile" page.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Authors who can not access this settings page will find their username on the "Edit Profile" page.', 'activitypub' ); ?>
|
||||||
<p>
|
<p>
|
||||||
<p>
|
<p>
|
||||||
<a href="<?php echo \esc_url_raw( \admin_url( '/profile.php#activitypub' ) ); ?>">
|
<a href="<?php echo \esc_url_raw( \admin_url( '/profile.php#activitypub' ) ); ?>">
|
||||||
<?php \esc_html_e( 'Customize Username on "Edit Profile" page.', 'activitypub' ); ?>
|
<?php \esc_html_e( 'Customize username on "Edit Profile" page.', 'activitypub' ); ?>
|
||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -87,9 +87,9 @@
|
||||||
<?php
|
<?php
|
||||||
echo wp_kses(
|
echo wp_kses(
|
||||||
\sprintf(
|
\sprintf(
|
||||||
// translators:
|
/* translators: the placeholder is the Site Health URL */
|
||||||
\__(
|
\__(
|
||||||
'If you have problems using this plugin, please check the <a href="%s">Site Health</a> to ensure that your site is compatible and/or use the "Help" tab (in the top right of the settings pages).',
|
'If you have problems using this plugin, please check the <a href="%s">Site Health</a> page to ensure that your site is compatible and/or use the "Help" tab (in the top right of the settings pages).',
|
||||||
'activitypub'
|
'activitypub'
|
||||||
),
|
),
|
||||||
\esc_url_raw( admin_url( 'site-health.php' ) )
|
\esc_url_raw( admin_url( 'site-health.php' ) )
|
||||||
|
|
|
@ -35,14 +35,15 @@ ENDPRE;
|
||||||
array( 'test', 'test' ),
|
array( 'test', 'test' ),
|
||||||
array( '#test', '#test' ),
|
array( '#test', '#test' ),
|
||||||
array( 'hallo #test test', 'hallo #test test' ),
|
array( 'hallo #test test', 'hallo #test test' ),
|
||||||
array( 'hallo #object test', 'hallo <a rel="tag" class="u-tag u-category" href="%s">#object</a> test' ),
|
array( 'hallo #object test', 'hallo <a rel="tag" class="hashtag u-tag u-category" href="%s">#object</a> test' ),
|
||||||
array( '#object test', '<a rel="tag" class="u-tag u-category" href="%s">#object</a> test' ),
|
array( '#object test', '<a rel="tag" class="hashtag u-tag u-category" href="%s">#object</a> test' ),
|
||||||
array( 'hallo <a href="http://test.test/#object">test</a> test', 'hallo <a href="http://test.test/#object">test</a> test' ),
|
array( 'hallo <a href="http://test.test/#object">test</a> test', 'hallo <a href="http://test.test/#object">test</a> test' ),
|
||||||
array( 'hallo <a href="http://test.test/#object">#test</a> test', 'hallo <a href="http://test.test/#object">#test</a> test' ),
|
array( 'hallo <a href="http://test.test/#object">#test</a> test', 'hallo <a href="http://test.test/#object">#test</a> test' ),
|
||||||
array( '<div>hallo #object test</div>', '<div>hallo <a rel="tag" class="u-tag u-category" href="%s">#object</a> test</div>' ),
|
array( '<div>hallo #object test</div>', '<div>hallo <a rel="tag" class="hashtag u-tag u-category" href="%s">#object</a> test</div>' ),
|
||||||
array( '<div>hallo #object</div>', '<div>hallo <a rel="tag" class="u-tag u-category" href="%s">#object</a></div>' ),
|
array( '<div>hallo #object</div>', '<div>hallo <a rel="tag" class="hashtag u-tag u-category" href="%s">#object</a></div>' ),
|
||||||
array( '<div>#object</div>', '<div>#object</div>' ),
|
array( '<div>#object</div>', '<div><a rel="tag" class="hashtag u-tag u-category" href="%s">#object</a></div>' ),
|
||||||
array( '<a>#object</a>', '<a>#object</a>' ),
|
array( '<a>#object</a>', '<a>#object</a>' ),
|
||||||
|
array( '<!-- #object -->', '<!-- #object -->' ),
|
||||||
array( '<div style="color: #ccc;">object</a>', '<div style="color: #ccc;">object</a>' ),
|
array( '<div style="color: #ccc;">object</a>', '<div style="color: #ccc;">object</a>' ),
|
||||||
array( $code, $code ),
|
array( $code, $code ),
|
||||||
array( $style, $style ),
|
array( $style, $style ),
|
||||||
|
|
|
@ -33,6 +33,7 @@ ENDPRE;
|
||||||
array( 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/author/matthias-pfefferle/">@pfefferle@notiz.blog</a> test', 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/author/matthias-pfefferle/">@pfefferle@notiz.blog</a> test' ),
|
array( 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/author/matthias-pfefferle/">@pfefferle@notiz.blog</a> test', 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/author/matthias-pfefferle/">@pfefferle@notiz.blog</a> test' ),
|
||||||
array( 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/@pfefferle/">@pfefferle@notiz.blog</a> test', 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/@pfefferle/">@pfefferle@notiz.blog</a> test' ),
|
array( 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/@pfefferle/">@pfefferle@notiz.blog</a> test', 'hallo <a rel="mention" class="u-url mention" href="https://notiz.blog/@pfefferle/">@pfefferle@notiz.blog</a> test' ),
|
||||||
array( 'hallo <img src="abc" alt="https://notiz.blog/@pfefferle/" title="@pfefferle@notiz.blog"/> test', 'hallo <img src="abc" alt="https://notiz.blog/@pfefferle/" title="@pfefferle@notiz.blog"/> test' ),
|
array( 'hallo <img src="abc" alt="https://notiz.blog/@pfefferle/" title="@pfefferle@notiz.blog"/> test', 'hallo <img src="abc" alt="https://notiz.blog/@pfefferle/" title="@pfefferle@notiz.blog"/> test' ),
|
||||||
|
array( '<!-- @pfefferle@notiz.blog -->', '<!-- @pfefferle@notiz.blog -->' ),
|
||||||
array( $code, $code ),
|
array( $code, $code ),
|
||||||
array( $pre, $pre ),
|
array( $pre, $pre ),
|
||||||
);
|
);
|
||||||
|
|
|
@ -19,5 +19,9 @@ class Test_Activitypub_Post extends WP_UnitTestCase {
|
||||||
$activitypub_post = \Activitypub\Transformer\Post::transform( get_post( $post ) )->to_object();
|
$activitypub_post = \Activitypub\Transformer\Post::transform( get_post( $post ) )->to_object();
|
||||||
|
|
||||||
$this->assertEquals( $permalink, $activitypub_post->get_id() );
|
$this->assertEquals( $permalink, $activitypub_post->get_id() );
|
||||||
|
|
||||||
|
$cached = \get_post_meta( $post, 'activitypub_canonical_url', true );
|
||||||
|
|
||||||
|
$this->assertEquals( $cached, $activitypub_post->get_id() );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,13 +39,13 @@ class Test_Activitypub_Signature_Verification extends WP_UnitTestCase {
|
||||||
|
|
||||||
// Start verification
|
// Start verification
|
||||||
// parse_signature_header, get_signed_data, get_public_key
|
// parse_signature_header, get_signed_data, get_public_key
|
||||||
$signature_block = Activitypub\Signature::parse_signature_header( $headers['signature'] );
|
$signature_block = Activitypub\Signature::parse_signature_header( $headers['signature'][0] );
|
||||||
$signed_headers = $signature_block['headers'];
|
$signed_headers = $signature_block['headers'];
|
||||||
$signed_data = Activitypub\Signature::get_signed_data( $signed_headers, $signature_block, $headers );
|
$signed_data = Activitypub\Signature::get_signed_data( $signed_headers, $signature_block, $headers );
|
||||||
|
|
||||||
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
||||||
|
|
||||||
$public_key = $user->get__public_key();
|
$public_key = Activitypub\Signature::get_public_key_for( $user->get__id() );
|
||||||
|
|
||||||
// signature_verification
|
// signature_verification
|
||||||
$verified = \openssl_verify( $signed_data, $signature_block['signature'], $public_key, 'rsa-sha256' ) > 0;
|
$verified = \openssl_verify( $signed_data, $signature_block['signature'], $public_key, 'rsa-sha256' ) > 0;
|
||||||
|
@ -57,7 +57,7 @@ class Test_Activitypub_Signature_Verification extends WP_UnitTestCase {
|
||||||
'pre_get_remote_metadata_by_actor',
|
'pre_get_remote_metadata_by_actor',
|
||||||
function( $json, $actor ) {
|
function( $json, $actor ) {
|
||||||
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
||||||
$public_key = $user->get__public_key();
|
$public_key = Activitypub\Signature::get_public_key_for( $user->get__id() );
|
||||||
// return ActivityPub Profile with signature
|
// return ActivityPub Profile with signature
|
||||||
return array(
|
return array(
|
||||||
'id' => $actor,
|
'id' => $actor,
|
||||||
|
|
110
tests/test-class-activitypub-signature.php
Normal file
110
tests/test-class-activitypub-signature.php
Normal file
|
@ -0,0 +1,110 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class Test_Activitypub_Signature extends WP_UnitTestCase {
|
||||||
|
public function test_signature_creation() {
|
||||||
|
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
$public_key = Activitypub\Signature::get_public_key_for( $user->get__id() );
|
||||||
|
$private_key = Activitypub\Signature::get_private_key_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_signature_legacy() {
|
||||||
|
// check user
|
||||||
|
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
||||||
|
|
||||||
|
$public_key = 'public key ' . $user->get__id();
|
||||||
|
$private_key = 'private key ' . $user->get__id();
|
||||||
|
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_public_key', $public_key );
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_private_key', $private_key );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
|
||||||
|
// check application user
|
||||||
|
$user = Activitypub\Collection\Users::get_by_id( -1 );
|
||||||
|
|
||||||
|
$public_key = 'public key ' . $user->get__id();
|
||||||
|
$private_key = 'private key ' . $user->get__id();
|
||||||
|
|
||||||
|
add_option( 'activitypub_application_user_public_key', $public_key );
|
||||||
|
add_option( 'activitypub_application_user_private_key', $private_key );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
|
||||||
|
// check blog user
|
||||||
|
\define( 'ACTIVITYPUB_DISABLE_BLOG_USER', false );
|
||||||
|
$user = Activitypub\Collection\Users::get_by_id( 0 );
|
||||||
|
|
||||||
|
$public_key = 'public key ' . $user->get__id();
|
||||||
|
$private_key = 'private key ' . $user->get__id();
|
||||||
|
|
||||||
|
add_option( 'activitypub_blog_user_public_key', $public_key );
|
||||||
|
add_option( 'activitypub_blog_user_private_key', $private_key );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_signature_consistancy() {
|
||||||
|
// check user
|
||||||
|
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
||||||
|
|
||||||
|
$public_key = 'public key ' . $user->get__id();
|
||||||
|
$private_key = 'private key ' . $user->get__id();
|
||||||
|
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_public_key', $public_key );
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_private_key', $private_key );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_public_key', $public_key . '-update' );
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_private_key', $private_key . '-update' );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
}
|
||||||
|
|
||||||
|
public function test_signature_consistancy2() {
|
||||||
|
$user = Activitypub\Collection\Users::get_by_id( 1 );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
$public_key = Activitypub\Signature::get_public_key_for( $user->get__id() );
|
||||||
|
$private_key = Activitypub\Signature::get_private_key_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_public_key', 'test' );
|
||||||
|
update_user_meta( $user->get__id(), 'magic_sig_private_key', 'test' );
|
||||||
|
|
||||||
|
$key_pair = Activitypub\Signature::get_keypair_for( $user->get__id() );
|
||||||
|
|
||||||
|
$this->assertNotEmpty( $key_pair );
|
||||||
|
$this->assertEquals( $key_pair['public_key'], $public_key );
|
||||||
|
$this->assertEquals( $key_pair['private_key'], $private_key );
|
||||||
|
}
|
||||||
|
}
|
|
@ -43,6 +43,11 @@ class Test_Db_Activitypub_Followers extends WP_UnitTestCase {
|
||||||
'name' => 'úser2',
|
'name' => 'úser2',
|
||||||
'preferredUsername' => 'user2',
|
'preferredUsername' => 'user2',
|
||||||
),
|
),
|
||||||
|
'error@example.com' => array(
|
||||||
|
'url' => 'https://error.example.com',
|
||||||
|
'name' => 'error',
|
||||||
|
'preferredUsername' => 'error',
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
public function set_up() {
|
public function set_up() {
|
||||||
|
@ -97,6 +102,27 @@ class Test_Db_Activitypub_Followers extends WP_UnitTestCase {
|
||||||
$this->assertContains( $follower2, $db_followers2 );
|
$this->assertContains( $follower2, $db_followers2 );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function test_add_follower_error() {
|
||||||
|
$pre_http_request = new MockAction();
|
||||||
|
add_filter( 'pre_http_request', array( $pre_http_request, 'filter' ), 10, 3 );
|
||||||
|
|
||||||
|
$follower = 'error@example.com';
|
||||||
|
|
||||||
|
$result = \Activitypub\Collection\Followers::add_follower( 1, $follower );
|
||||||
|
|
||||||
|
$this->assertTrue( is_wp_error( $result ) );
|
||||||
|
|
||||||
|
$follower2 = 'https://error.example.com';
|
||||||
|
|
||||||
|
$result = \Activitypub\Collection\Followers::add_follower( 1, $follower2 );
|
||||||
|
|
||||||
|
$this->assertTrue( is_wp_error( $result ) );
|
||||||
|
|
||||||
|
$db_followers = \Activitypub\Collection\Followers::get_followers( 1 );
|
||||||
|
|
||||||
|
$this->assertEmpty( $db_followers );
|
||||||
|
}
|
||||||
|
|
||||||
public function test_get_follower() {
|
public function test_get_follower() {
|
||||||
$followers = array( 'https://example.com/author/jon' );
|
$followers = array( 'https://example.com/author/jon' );
|
||||||
$followers2 = array( 'https://user2.example.com' );
|
$followers2 = array( 'https://user2.example.com' );
|
||||||
|
@ -268,6 +294,30 @@ class Test_Db_Activitypub_Followers extends WP_UnitTestCase {
|
||||||
$this->assertCount( 1, $meta );
|
$this->assertCount( 1, $meta );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function test_migration() {
|
||||||
|
$pre_http_request = new MockAction();
|
||||||
|
add_filter( 'pre_http_request', array( $pre_http_request, 'filter' ), 10, 3 );
|
||||||
|
|
||||||
|
$followers = array(
|
||||||
|
'https://example.com/author/jon',
|
||||||
|
'https://example.og/errors',
|
||||||
|
'https://example.org/author/doe',
|
||||||
|
'http://sally.example.org',
|
||||||
|
'https://error.example.com',
|
||||||
|
'https://example.net/error',
|
||||||
|
);
|
||||||
|
|
||||||
|
$user_id = 1;
|
||||||
|
|
||||||
|
add_user_meta( $user_id, 'activitypub_followers', $followers, true );
|
||||||
|
|
||||||
|
\Activitypub\Migration::maybe_migrate();
|
||||||
|
|
||||||
|
$db_followers = \Activitypub\Collection\Followers::get_followers( 1 );
|
||||||
|
|
||||||
|
$this->assertCount( 3, $db_followers );
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @dataProvider extract_name_from_uri_content_provider
|
* @dataProvider extract_name_from_uri_content_provider
|
||||||
*/
|
*/
|
||||||
|
|
Loading…
Reference in a new issue