Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix federation with GoToSocial and inconsistent KeyId in headers #2812

Merged
merged 16 commits into from
May 30, 2023

Conversation

hughrun
Copy link
Contributor

@hughrun hughrun commented Apr 10, 2023

Fixes #2801
Fixes #2794

This PR resolves a number of problems specific to GoToSocial federation but also potentially affecting other fediverse software.

Fix making assumptions about location of main-key

It is legitimate to use any url for the user's key id. We have been assuming this id is the user id plus a fragment ({userid}#main-key) but this is not always the case, notably in the case of GoToSocial it is at {userid}/main-key. This commit instead checks the remote user's information to see if the key id listed matches the key id of the message allegedly received from them.

This resolves most of the 401 errors when trying to communicate with GtS servers.

Fix Bookwyrm using wrong keyId in signed request headers

Whilst troubleshooting this it also became apparent that there is a mismatch between Bookwyrm users' keyId and the keyId we claim to be using in signed requests:

  • Actual keyId defined in User: {userid}/#main-key
  • keyId stated in signed request headers: {userid}#main-key

This commit changes the keyId listed in request headers to align with that assigned to users on creation.

Fix GoToSOcial DMs not appearing and single-user mentions not notifying

This error was caused by GtS sending the tag value for these posts as a single object (dict) rather than an array containing the object, and Bookwyrm assuming it would be an array (list).

This commit wraps the dict in a list where necessary.

Maintain backwards compatibility with legacy Bookwyrm instances

Aligning the keyId in the signatures with the real keyId of Bookwyrm users would break compatibility with all Bookwyrm versions before this change. The PR adds some logic to re-send the request with the old (wrong) keyId in the header if the first request is rejected. As Bookwyrm instances are updated this will become less of a problem.

hughrun added 2 commits April 10, 2023 17:32
Fixes bookwyrm-social#2801
Related to bookwyrm-social#2794

It is legitimate to use any url for the user's key id. We have been assuming this id is the user id plus a fragment (#key-id) but this is not always the case, notably in the case of GoToSocial it is at /key-id. This commit instead checks the remote user's information to see if the key id listed matches the key id of the message allegedly received from them.

Whilst troubleshooting this it also became apparent that there is a mismatch between Bookwyrm users' keyId and the KeyId we claim to be using in signed requests (there is a forward slash missing). Since everything after the slash is a fragment, this usually slips through but we should be consistent so I updated that.
bookwyrm/signatures.py Outdated Show resolved Hide resolved
hughrun added 3 commits April 10, 2023 20:34
Previously the 'tag' value in an activitypub object was assumed to be a List (array).
Some AP software sends 'tag' as a Dict (object) if there is only a single tag value.
It's somewhat debatable whether this is spec compliant but we should aim to be robust.
This commit puts an individual mention tag inside a list if necessary.
GoToSocial expects the 'name' value of a mention tag to have an initial '@' symbol. Mastodon doesn't seem to mind either way.
hughrun added 3 commits April 11, 2023 15:45
Bookwyrm keyIds are at `userpath/#main-key`, however when signing AP objects we have claimed in the headers that the keyId is at `userpath#main-key`.
This is incorrect, and makes GoToSocial's strict checking break.
Simply updating the signatures to use the correct KeyId breaks legacy Bookwyrm's signature checks, becuase it assumes that the keyId path is the same as the user path plus a fragment.
This commit allows for either option, by sending the request a second time with the incorrect keyId if sending with the correct one causes an error.
bookwyrm/models/status.py Outdated Show resolved Hide resolved
@hughrun hughrun requested a review from WesleyAC April 11, 2023 09:11
@hughrun hughrun added help wanted Extra attention is needed activitypub labels Apr 11, 2023
@hughrun hughrun requested a review from mouse-reeve April 11, 2023 09:11
bookwyrm/models/activitypub_mixin.py Outdated Show resolved Hide resolved
if not remote_user:
return False

if signature.key_id != remote_user.key_pair.remote_id:
if signature.key_id != f"{remote_user.remote_id}#main-key": # legacy Bookwyrm
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm a bit confused by this, isn't the legacy one the one with the /? And I thought remote_user.remote_id didn't have a /?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is incredibly confusing.

All Bookwyrm keyIds are at {username}/#main-key - this is set in models.User so it's impractical to change that because we can't retrospectively change people's keyIds.

However, the legacy signatures (falsely) assert that the keyId is at {username}#main-key. Both are equivalent if you want to hit a url, but are not if you want to compare strings.

So if the remote user is from another Bookwyrm instance that has not upgraded, it will send a signature asserting the remote user's keyId is at #main-key but when our updated instance goes and checks, it will see that the keyId is actually listed as /#main-key. This line is a bit of a hack where our instance says "hmm, maybe it's old Bookwyrm, what if I just pretend the last forward slash isn't there?".

hughrun added 2 commits April 13, 2023 13:21
GoToSocial sends 'tag' values as a single object if there is only one
user mentioned, rather than an array with an object inside it.

This causes Bookwyrm to reject the tag since it comes through as a
dict rather than a list.

This commit fixes this at the point the incoming AP object is transformed
so that "mention" tags are turned into a mention_user.
@hughrun hughrun removed the request for review from mouse-reeve April 13, 2023 03:28
@hughrun hughrun removed the help wanted Extra attention is needed label Apr 13, 2023
@hughrun hughrun marked this pull request as ready for review April 14, 2023 08:24
@hughrun hughrun requested a review from WesleyAC April 14, 2023 08:27
@hughrun hughrun changed the title GoToSocial fixes Fix federation with GoToSocial and inconsistent KeyId in headers Apr 14, 2023
@chdorner
Copy link
Member

Fix GoToSOcial DMs not appearing and single-user mentions not notifying

Haven’t looked too much into it yet but I think the same fix should also help with any interactions like boosting or liking BookWyrm posts from Takahē and those ActivityPub objects being properly handled on BookWyrm’s side.

@tsmethurst
Copy link

Cool, nice one! :) Thanks for making these changes, looking forward to hanging out with Bookwyrm peeps from GtS :)

@mouse-reeve
Copy link
Member

Is this ready to merge?

@hughrun
Copy link
Contributor Author

hughrun commented May 15, 2023

I think so but I'd really like someone to review and test it because it has a lot of potential to break things.

@kvibber
Copy link
Contributor

kvibber commented May 29, 2023

I brought this up to current main and ran it on a test instance for a bit. So far:

  • I'm able to look up and follow users on GoToSocial, Mastodon and other Bookwyrm instances, and they're able to look up and follow back.
  • statuses are federating outward to all three types of servers. (Sorry about the review spam on bw.social - I've deleted the test posts!)
  • statuses from users I follow on other Bookwyrm instances are showing up
  • mentions from GoToSocial are showing up in Bookwyrm, but not regular posts from a GTS user who's being followed
  • same with Mastodon, which might be a regression since I'm able to follow Mastodon users from Bookwyrm.social...but might just be something broken in my test server, because when I switch back to main, it still isn't picking up new statues from Mastodon users that it follows.

@hughrun
Copy link
Contributor Author

hughrun commented May 29, 2023

Thanks for testing!

  • mentions from GoToSocial are showing up in Bookwyrm, but not regular posts from a GTS user who's being followed
  • same with Mastodon, which might be a regression

This is intended behaviour - Bookwyrm has never displayed standard Notes from people you follow unless they mention you.

So it sounds like we're good to go @mouse-reeve !

@mouse-reeve mouse-reeve merged commit a4ccd45 into bookwyrm-social:main May 30, 2023
@hughrun hughrun deleted the gts branch November 5, 2023 22:27
@mirabilos
Copy link

Is this released? And if so, in 0.6.3?

I have many messages like…

current:timestamp="2023-12-20T20:15:34.663Z" func=server.glob..func1.Logger.func13.1 level=INFO latency="210.618µs" userAgent="python-requests/2.31.0 (BookWyrm/0.6.3; +https://outside.ofa.dog/)" method=POST statusCode=406 path=/users/mirabilos/inbox errors="Error #01: Content-Type application/activity+json; charset=utf-8 not acceptable, this endpoint accepts: [\"application/activity+json\" \"application/ld+json;profile=https://w3.org/ns/activitystreams\"]\n" requestID=0y4xx24c04001wfddvn0 msg="Not Acceptable: wrote 50B"

… and even one…

current:timestamp="2023-12-20T20:11:03.869Z" func=server.glob..func1.Logger.func13.1 level=INFO latency="23.278798ms" userAgent="python-requests/2.31.0 (BookWyrm/0.6.3; +https://outside.ofa.dog/)" method=GET statusCode=401 path=/users/mirabilos errors="Error #01: AuthenticateFederatedRequest: http request wasn't signed or http signature was invalid\n" requestID=4nkxn24c04001s3y0ef0 msg="Unauthorized: wrote 104B"

… in my GtS log and can’t follow my new Bookwyrm user from my GtS user (it stays on “following requested”, doesn’t get the accept).

@tsmethurst
Copy link

tsmethurst commented Dec 20, 2023

Former is not related to this, it's an issue with GtS perhaps too strictly enforcing POST content-type as given in the spec: https://www.w3.org/TR/activitypub/#server-to-server-interactions

We could fix this by trimming ; charset=utf-8 from incoming POST request headers on GoToSocial's side.

Latter error message looks like an unsigned GET to a user profile.

@mirabilos
Copy link

Ah, okay then, thanks. I can hold off further experiments until after the GtS vacation (and until the instance I’m using runs latest BookWyrm, too). I was just having an initial look at BookWyrm.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
7 participants