Skip to content

Instantly share code, notes, and snippets.

@rjhansen
Last active October 26, 2024 22:16
Show Gist options
  • Save rjhansen/67ab921ffb4084c865b3618d6955275f to your computer and use it in GitHub Desktop.
Save rjhansen/67ab921ffb4084c865b3618d6955275f to your computer and use it in GitHub Desktop.
SKS Keyserver Network Under Attack

SKS Keyserver Network Under Attack

This work is released under a Creative Commons Attribution-NoDerivatives 4.0 International License.

Terminological Note

"OpenPGP" refers to the OpenPGP protocol, in much the same way that HTML refers to the protocol that specifies how to write a web page. "GnuPG", "SequoiaPGP", "OpenPGP.js", and others are implementations of the OpenPGP protocol in the same way that Mozilla Firefox, Google Chromium, and Microsoft Edge refer to software packages that process HTML data.

Who am I?

Robert J. Hansen <rjh@sixdemonbag.org>. I maintain the GnuPG FAQ and unofficially hold the position of crisis communicator. This is not an official statement of the GnuPG project, but does come from someone with commit access to the GnuPG git repo.

Executive Summary

In the last week of June 2019 unknown actors deployed a certificate spamming attack against two high-profile contributors in the OpenPGP community (Robert J. Hansen and Daniel Kahn Gillmor, better known in the community as "rjh" and "dkg"). This attack exploited a defect in the OpenPGP protocol itself in order to "poison" rjh and dkg's OpenPGP certificates. Anyone who attempts to import a poisoned certificate into a vulnerable OpenPGP installation will very likely break their installation in hard-to-debug ways. Poisoned certificates are already on the SKS keyserver network. There is no reason to believe the attacker will stop at just poisoning two certificates. Further, given the ease of the attack and the highly publicized success of the attack, it is prudent to believe other certificates will soon be poisoned.

This attack cannot be mitigated by the SKS keyserver network in any reasonable time period. It is unlikely to be mitigated by the OpenPGP Working Group in any reasonable time period. Future releases of OpenPGP software will likely have some sort of mitigation, but there is no time frame. The best mitigation that can be applied at present is simple: stop retrieving data from the SKS keyserver network.

How Keyservers Work

When Phil Zimmermann first developed PGP ("Pretty Good Privacy") in the early 1990s there was a clear chicken and egg problem. Public key cryptography could revolutionize communications but required individuals possess each other's public keys. Over time terminology has shifted: now public key cryptography is mostly called "asymmetric cryptography" and public keys are more often called "public certificates", but the chicken-and-egg problem remains. To communicate privately, each party must have a small piece of public data with which to bootstrap a private communication channel.

Special software was written to facilitate the discovery and distribution of public certificates. Called "keyserver software", it can be thought of as analogous to a telephone directory. Users can search the keyserver by a variety of different criteria to discover public certificates which claim to belong to the desired user. The keyserver network does not attest to the accuracy of the information, however: that's left for each user to ascertain according to their own criteria.

Once a user has verified a certificate really and truly belongs to the person in question, they can affix an affidavit to the certificate attesting that they have reason to believe the certificate really belongs to the user in question.

For instance: John Hawley (john@example.org) and I (rjh@example.org) are good friends in real life. We have sat down face-to-face and confirmed certificates. I know with complete certainty a specific public certificate belongs to him; he knows with complete certainty a different one belongs to me. John also knows H. Peter Anvin (hpa@example.org) and has done the same with him. If I need to communicate privately with Peter, I can look him up in the keyserver. Whichever certificate bears an attestation by John, I can trust really belongs to Peter.

Keyserver Design Goals

In the early 1990s we were concerned repressive regimes would attempt to force keyserver operators to replace certificates with different ones of the government's choosing. (I speak from firsthand experience. I've been involved in the PGP community since 1992. I was there for these discussions.) We made a quick decision that keyservers would never, ever, ever, delete information. Keyservers could add information to existing certificates but could never, ever, ever, delete either a certificate or information about a certificate.

To meet this goal, we started running an international network of keyservers. Keyservers around the world would regularly communicate with each other to compare directories. If a government forced a keyserver operator to delete or modify a certificate, that would be discovered in the comparison step. The maimed keyserver would update itself with the content in the good keyserver's directory. This was a simple and effective solution to the problem of government censorship.

In the early 1990s this design seemed sound. It is not sound in 2019. We've known it has problems for well over a decade.

Why Hasn't It Been Fixed?

There are powerful technical and social factors inhibiting further keyserver development.

  • The software is Byzantine. The standard keyserver software is called SKS, for "Synchronizing Key Server". A bright fellow named Yaron Minsky devised a brilliant algorithm that could do reconciliations very quickly. It became the keystone of his Ph.D thesis, and he wrote SKS originally as a proof of concept of his idea. It's written in an unusual programming language called OCaml, and in a fairly idiosyncratic dialect of it at that. This is of course no problem for a proof of concept meant to support a Ph.D thesis, but for software that's deployed in the field it makes maintenance quite difficult. Not only do we need to be bright enough to understand an algorithm that's literally someone's Ph.D thesis, but we need expertise in obscure programming languages and strange programming customs.

  • The software is unmaintained. Due to the above, there is literally no one in the keyserver community who feels qualified to do a serious overhaul on the codebase.

  • Changing a design goal is not the same as fixing a bug. The design goal of the keyserver network is "baked into" essentially every part of the infrastructure. This isn't a case where there's a bug that's inhibiting the keyserver network from functioning correctly. Bugs are generally speaking fairly easy to fix once you know where the problem is. Changing design goals often requires an overhaul of such magnitude it may be better to just start over with a fresh sheet of paper.

  • There is no centralized authority in the keyserver network. The lack of centralized authority was a feature, not a bug. If there is no keyserver that controls the others, there is no single point of failure for a government to go after. On the other hand it also means that even after the software is overhauled and/or rewritten, each keyserver operator has to commit to making the upgrade and stomping out the difficulties that inevitably arise when new software is fielded. The confederated nature of the keyserver network makes changing the design goals even harder than it would normally be—and rest assured, it would normally be very hard!

The Vulnerabilities

The keyserver network is susceptible to a variety of attacks as a consequence of its write-only design. The keyserver network can be thought of as an extremely large, extremely reliable, extremely censorship-resistant distributed filesystem which anyone can write to.

Imagine if Dropbox allowed any Tom, Dick, or Harry to not only put information in your public Dropbox folder, but made it impossible for you to delete it. How would everyone from spammers to child pornographers abuse this?

Many of the same attacks are possible on the keyserver network. We have known about these vulnerabilities for well over a decade. Fixing the keyserver network is, however, problematic for the reasons listed above.

In order to limit the scope of this document a detailed breakdown of only one such vulnerability will be presented (see below).

The Certificate Spamming Attack

Consider public certificates. In order to make them easier to use, they have a list of attestations: statements from other people, represented by their own public certificates, that this certificate really belongs to the individual in question. In my example from before, John Hawley attested to H. Peter Anvin's certificate. When I looked for H. Peter Anvin's certificate I checked all the certificates which claimed to belong to him and selected the one John attested as being really his.

These attestations — what we call certificate signatures — can be made by anyone for any purpose. And once made, they never go away. Ever. Even when a certificate signature gets revoked the original remains on the certificate: all that happens is a second signature is affixed saying "don't trust the previous one I made".

The OpenPGP specification puts no limitation on how many signatures can be attached to a certificate. The keyserver network handles certificates with up to about 150,000 signatures.

GnuPG, on the other hand … doesn't. Any time GnuPG has to deal with such a spammed certificate, GnuPG grinds to a halt. It doesn't stop, per se, but it gets wedged for so long it is for all intents and purposes completely unusable.

My public certificate as found on the keyserver network now has just short of 150,000 signatures on it.

Further, pay attention to that phrase any time GnuPG has to deal with such a spammed certificate. If John were to ask GnuPG to verify my signature on H. Peter Anvin's certificate, GnuPG would attempt to comply and in the course of business would have to deal with my now-spammed certificate.

The Consequences

We've known for a decade this attack is possible. It's now here and it's devastating. There are a few major takeaways and all of them are bad.

  • If you fetch a poisoned certificate from the keyserver network, you will break your GnuPG installation.
  • Poisoned certificates cannot be deleted from the keyserver network.
  • The number of deliberately poisoned certificates, currently at only a few, will only rise over time.
  • We do not know whether the attackers are intent on poisoning other certificates.
  • We do not even know the scope of the damage.

That last one requires some explanation. Any certificate may be poisoned at any time, and is unlikely to be discovered until it breaks an OpenPGP installation.

The number one use of OpenPGP today is to verify downloaded packages for Linux-based operating systems, usually using a software tool called GnuPG. If someone were to poison a vendor's public certificate and upload it to the keyserver network, the next time a system administrator refreshed their keyring from the keyserver network the vendor's now-poisoned certificate would be downloaded. At that point upgrades become impossible because the authenticity of downloaded packages cannot be verified. Even downloading the vendor's certificate and re-importing it would be of no use, because GnuPG would choke trying to import the new certificate. It is not hard to imagine how motivated adversaries could employ this against a Linux-based computer network.

Mitigations

At present I (speaking only for myself) do not believe the global keyserver network is salvageable. High-risk users should stop using the keyserver network immediately.

Users who are confident editing their GnuPG configuration files should follow the following process:

  1. Open gpg.conf in a text editor. Ensure there is no line starting with keyserver. If there is, remove it.
  2. Open dirmngr.conf in a text editor. Add the line keyserver hkps://keys.openpgp.org to the end of it.

keys.openpgp.org is a new experimental keyserver which is not part of the keyserver network and has some features which make it resistant to this sort of attack. It is not a drop-in replacement: it has some limitations (for instance, its search functionality is sharply constrained). However, once you make this change you will be able to run gpg --refresh-keys with confidence.

Repairs

If you know which certificate is likely poisoned, try deleting it: this normally goes pretty quickly. If your OpenPGP installation becomes usable again, congratulations. Acquire a new unpoisoned copy of the certificate and import that.

If you don't know which certificate is poisoned, your best bet is to get a list of all your certificate IDs, delete your keyrings completely, and rebuild from scratch using known-good copies of the public certificates.

A Personal Postscript

dkg wrote a blog post about this. He sums up my feelings pretty well, so I'm going to quote him liberally with only a trivial correction.

I've spent a significant amount of time over the years trying to push the ecosystem into a more responsible posture with respect to OpenPGP certificates, and have clearly not been as successful at it or as fast as I wanted to be. Complex ecosystems can take time to move.

To have my own certificate directly spammed in this way felt surprisingly personal, as though someone was trying to attack or punish me, specifically. I can't know whether that's actually the case, of course, nor do I really want to. And the fact that Robert J. Hansen's certificate was also spammed makes me feel a little less like a singular or unique target, but I also don't feel particularly proud of feeling relieved that someone else is also being "punished" in addition to me.

But this report wouldn't be complete if I didn't mention that I've felt disheartened and demotivated by this situation. I'm a stubborn person, and I'm trying to make the best of the situation by being constructive about at least documenting the places that are most severely broken by this. But I've also found myself tempted to walk away from this ecosystem entirely because of this incident. I don't want to be too dramatic about this, but whoever did this basically experimented on me (and Rob) directly, and it's a pretty shitty thing to do.

If you're reading this, and you set this off, and you selected me specifically because of my role in the OpenPGP ecosystem, or because I wrote the abuse-resistant-keystore draft, or because I'm part of the Autocrypt project, then you should know that I care about making this stuff work for people. If you'd reached out to me to describe what you were planning to do, we could have done all of the above bug reporting and triage using demonstration certificates, and worked on it together. I would have happily helped. I still might! But because of the way this was done, I'm not feeling particularly happy right now. I hope that someone is, somewhere.

To which I'd like to add: I have never in my adult life wished violence on any human being. I have witnessed too much of it and its barbaric effects, stood by the graves of too many people cut down too young. I do not hate you and I do not wish any harm to befall you.

But if you get hit by a bus while crossing the street, I'll tell the driver everyone deserves a mulligan once in a while.

You fool. You absolute, unmitigated, unadulterated, complete and utter, fool.

Peace to everyone — including you, you son of a bitch.

— Rob

@706f6c6c7578
Copy link

I used gpg.mozilla.org, it was not affected by this attack.

And if the attackers wanted to, how long would it take them to attack any new keyserver? Zero time, essentially. So what's the point if telling people to use a server that hasn't been attacked yet? Is it to encourage new servers to be attacked?

what's different with keys.openpgp.org? keys.openpgp.org is not compatible with gnugp's keys. You cannot search or fetch new keys with this server. gpg.mozilla.org is not pool of SKS too.

Why do you think it is not compatible? And you can search for keys there:

https://keys.openpgp.org/search?q=

or put the key server URL in your config file and then GnuPG uses this server for fetching.

@NgoHuy
Copy link

NgoHuy commented Jul 17, 2019

I used gpg.mozilla.org, it was not affected by this attack.

And if the attackers wanted to, how long would it take them to attack any new keyserver? Zero time, essentially. So what's the point if telling people to use a server that hasn't been attacked yet? Is it to encourage new servers to be attacked?

what's different with keys.openpgp.org? keys.openpgp.org is not compatible with gnugp's keys. You cannot search or fetch new keys with this server. gpg.mozilla.org is not pool of SKS too.

Why do you think it is not compatible? And you can search for keys there:

https://keys.openpgp.org/search?q=

or put the key server URL in your config file and then GnuPG uses this server for fetching.

with latest gnupg, user should show this message.
image

@NgoHuy
Copy link

NgoHuy commented Jul 17, 2019

Sorry, I forgot to verify my email, then the key on pgp will be treated as non-identify.
It's fine now.

@nukeop
Copy link

nukeop commented Jul 17, 2019

sorry, that was me. I started vim, then I pressed something, and it just kept making those beeping noises and started saying something about certificates, signatures, and gpg. i have no idea what happened, sorry

@FranklinYu
Copy link

This vulnerability has been assigned to CVE-2019-13050. More description can be found in RedHat Knowledgebase.

@706f6c6c7578
Copy link

Howdy all, I'm interested in writing an sks-compatible keyserver implementation that can handle poisoned keys gracefully. I understand the HKP protocol, but I can't seem to find any documentation on the gossip/reconciliation protocol. All I can find is the academic paper, which is very hard to understand.
Does anyone understand the gossip protocol on a technical level and would you be willing to mentor me as I learn it?

@diafygi, you (and maybe @rjhansen?) might find this other keyserver implementation, Hockeypuck, of interest—looks like it's supposed to be protocol-compatible with SKS, but written in Go instead of OCaml: https://hockeypuck.github.io/

In case people have not seen it yet, the hockeypuck key server software, at least at the Ubuntu site, allows fake UAT packets been displayed.

https://keyserver.ubuntu.com/pks/lookup?search=0x7b96d396e6471601754be4db53b620d01ce0c630&fingerprint=on&op=vindex

@rozzin
Copy link

rozzin commented Dec 30, 2019

Howdy all, I'm interested in writing an sks-compatible keyserver implementation that can handle poisoned keys gracefully. I understand the HKP protocol, but I can't seem to find any documentation on the gossip/reconciliation protocol. All I can find is the academic paper, which is very hard to understand.
Does anyone understand the gossip protocol on a technical level and would you be willing to mentor me as I learn it?

@diafygi, you (and maybe @rjhansen?) might find this other keyserver implementation, Hockeypuck, of interest—looks like it's supposed to be protocol-compatible with SKS, but written in Go instead of OCaml: https://hockeypuck.github.io/

In case people have not seen it yet, the hockeypuck key server software, at least at the Ubuntu site, allows fake UAT packets been displayed.

https://keyserver.ubuntu.com/pks/lookup?search=0x7b96d396e6471601754be4db53b620d01ce0c630&fingerprint=on&op=vindex

Thanks @SAC001; would you mind please filing that in Hockeypuck's bug tracker? https://github.com/hockeypuck/hockeypuck/issues

And I just want to be clear that my point was not anything like "maybe Hockeypuck doesn't have these problems", but rather "maybe people who want to fix the problems but are intimidated by OCaml can be less intimidated by Go. :)

It looks like @cmars has been steadily hacking away on Hockeypuck; help in whatever way you can--one of those ways is to file bugs in the actual bugtracker so that developers can find them easily ;)

@706f6c6c7578
Copy link

Hi rozzin,

just submitted the issue to the bug tracker.

@cmars
Copy link

cmars commented Dec 31, 2019

Thanks for the bug report @SAC001. Now fixed in master; query responses will only contain uid, uat and subkey packets that have valid primary key self-signatures.

There is also a configuration option for only including self-signed packets in responses. This excludes WoT use cases, but it can protect clients like apt which are primarily concerned with signature verification.

This will go out in the next release. I'll put in a request to upgrade keyserver.ubuntu.com after we're back from the holiday break.

@dgoiko
Copy link

dgoiko commented Jan 30, 2020

Does anyone have a link to the thesis around?

@ssaavedra
Copy link

This is the reference DOI to it: https://dl.acm.org/doi/book/10.5555/935747

Since it's @yminsky's copyright, if he were so kind to make it available it would be great.

@hno
Copy link

hno commented Feb 7, 2020

A quick search turned up this alternate source for the paper where a PDF is available for free.

https://www.semanticscholar.org/paper/Spreading-rumors-cheaply%2C-quickly%2C-and-reliably-Schneider-Minsky/cfad58069fb2663758e5a97943de582984ad2bbb

@cmars
Copy link

cmars commented Dec 10, 2020

I've released Hockeypuck 2.1.0, which contains several new features that may be useful to mitigate these type of attacks. See the release link for details, but in a nutshell, I've added:

  • Configurable key length and packet size limits, with sensible defaults.
  • Configurable blacklist of primary key fingerprints.
  • Authenticated key management. This adds a couple of extra endpoints which allow a key owner to replace or delete their key, authenticated by self-signing the armored key in the request.

Blacklists and auth key management may also be of interest to keyserver operators subject to GDPR-related requests.

@adulau
Copy link

adulau commented Jan 19, 2021

Is there an official list of current malicious keys abusing the resources of PGP key servers?

  "2790943722612cf8d9c2db9213de25eed1bb5151",
  "a490d0f4d311a4153e2bb7cadbb802b258acd84f",
  "5c738727ee58786a777c4f1db5aa3fa3486ed7ad",
  "33d51b5621953173ab74b521bdca9f8e3a6c1785",

There is a partial list Hockeypuck changelog but do you know some more? It would be useful for PGP key server operators.

@KOLANICH
Copy link

KOLANICH commented Feb 17, 2021

I feel like the proper solution will be authenticating signatures from both parties.

So I guess it is time to just change the rules. In order for a signature to be considered valid, its key fingerprint must be present in a self-signed part of the "key" it signs. All the keys should be edited, but it is not an issue, they should be periodically re-signed anyway, and if an author doesn't care.

It would solve not only the issue of spamming but also the issue of unilateral violation of rules of signing another's "key"s.

I mean there are lot of people ready to sign random keys in the internet without proper identity and key checks (such as meeting personally, verifying identity of a person using natural face and voice biometrics built into every human being, then verifying integrity of the key, and only after that signing) because they say "but I trust them!" (which is an absolute violation of "key"-signing semantics).

@IzzySoft
Copy link

its key fingerprint must be present in a self-signed part of the "key" it signs

In that case, please make sure to have some "short and clear instructions" in an easy-to-find place. Else especially novice users will be overstretched (TBH, I use PGP for about 20 years now and couldn't tell out of my head).

As an alternative, wouldn't if be sufficient if the key owner self-uploads the key (IIRC that was the only way to get it in/updated anyway, requiring approval via a mail sent to the key's main email address)? Or is that "too weak" (as some "unwanted sigs" might sneak in unnoticed)?

@KOLANICH
Copy link

As an alternative, wouldn't if be sufficient if the key owner self-uploads the key (IIRC that was the only way to get it in/updated anyway, requiring approval via a mail sent to the key's main email address)? Or is that "too weak" (as some "unwanted sigs" might sneak in unnoticed)?

Of course no. I often upload to keys3rvers keys that don't belong to me, just for convenience of retrieval.

@IzzySoft
Copy link

Oh, then I must remember something wrongly. I thought with openpgp.net only the owner could upload (and needs to click the confirmation link sent by mail). Apologies then.

@KOLANICH
Copy link

I don't use openpgp.net so IDK how the things are going on there, my favorite keyservers are pgp.mit.edu and keyserver.ubuntu.com and at least on them and on the most of other keyservers using the same software one can upload public "keys" of other people.

@hno
Copy link

hno commented Mar 29, 2021

Does anyone have a link to the thesis around?

https://citeseerx.ist.psu.edu/viewdoc/summary?doi=10.1.1.20.5620

@GioF
Copy link

GioF commented Apr 1, 2021

So, as a newcomer to PGP that is interested on using the WoT, do any dangers on importing keys from the Hockeypuck or SKS keyservers related to key poisoning remain?

@yakamok
Copy link

yakamok commented Jun 10, 2021

Giof, the SKS keyservers are essentially dead now, you can use https://keys.openpgp.org/ instead to upload your public key and share.

@706f6c6c7578
Copy link

706f6c6c7578 commented Jun 10, 2021

@GioF Consider taking a look at https://keys.mailvelope.com/, because there you have full control over your pub key(s) and it keeps your WoT signatures, compared to https://keys.openpgp.org. Or If you like to use modern encryption tools forget PGP and switch, like a lot of people already did to modern 'age' https://github.com/FiloSottile/age. The author of age is a well known cryptographer and security lead for Golang and has on Twitler more followers than gnupg.org on Twitler... HTH!

@706f6c6c7578
Copy link

@KOLANICH that is perfectly fine if you still like to use GnuPG, I only mentioned age because many GnuPG users already switched to age. That should IMHO tell us something, you know. I also could have mentioned modern sequoia-pgp instead, but that is OpenPGP too and has a slightly higher learning curve, for beginners, than age. See also Mr. Zimmermann's Testimonial. https://sequoia-pgp.org/ Anyways, I will stay now on topic and hope that @GioF has found the right keyserver solution for himself and his communication partners!

@mdziczkowski
Copy link

Poisoned certificates cannot be deleted from the keyserver network.

can't the key become revoked ?

@muelli
Copy link

muelli commented Sep 12, 2021

Poisoned certificates cannot be deleted from the keyserver network.

can't the key become revoked ?

sure. But then they won't be deleted. That'd be silly as then nobody could learn about their revocation. The certifications could probably be stripped, though.

@duxsco
Copy link

duxsco commented Oct 11, 2021

For me, GnuPG's traditional Web of Trust approach is dead. I have written my thoughts including a "new" way to realise Web of Trust:
https://github.com/duxco?tab=repositories&q=gpg-&type=&language=&sort=

Perhaps, that's helpful for one or the other 🙂

@KOLANICH
Copy link

KOLANICH commented Oct 11, 2021

You need a class 3 S/MIME certificate signed by CAcert:

Thank you, but no. It is inacceptable for a decentralysed system to depend on CAs. Though it can be acceptable to depend on blockchain-based solutions.

@duxsco
Copy link

duxsco commented Oct 20, 2021

Poisoned certificates cannot be deleted from the keyserver network.

can't the key become revoked ?

sure. But then they won't be deleted. That'd be silly as then nobody could learn about their revocation. The certifications could probably be stripped, though.

The missing feature of deletion is the reason for hosting my own keyserver. I created my keypair with ed25519 for primary and nistp521 for subkeys. But, after figuring out that the nistp521 authentication subkey cannot be used for ssh I decided to use nistp384 for that subkey. Using something like hkps://keys.openpgp.org would make the obsolete nistp521 auth subkey linger around although that subkey is not needed for communication with others. This is not something that I wish for, because I want to keep my pubkey slim as long as it makes sense.

@Tachi107
Copy link

Tachi107 commented Aug 1, 2022

Reposting here too.

If you come across this thread and you're interested in what could be done to solve the issue, I'd recommend you to look into Attested Certifications (also called 1PA3PC)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment