- Feature Name: crates-io-username-identity
- Start Date: 2026-04-07
- RFC PR: rust-lang/rfcs#3946
- Crates.io Issue: rust-lang/crates.io#13758
Summary
Someday, we would like to enable people to log in to crates.io with other services in addition to GitHub. This RFC is not yet about adding other services for login. It is proposing that crates.io change to have the concept of a “crates.io username” separate and possibly different from users’ GitHub usernames. Crates.io needs this change to make authenticating with different services possible while minimizing confusion.
🚨 After this RFC is accepted and implemented, you will still only be able to log in to crates.io via GitHub. This is a prerequisite of the eventual goal to add other methods of logging in. 🚨
The biggest changes to crates.io as a result of this RFC will be:
- There will be a crates.io username that may not always match the associated GitHub username
- Crates.io will no longer automatically update your crates.io username if you rename your GitHub account
Motivation
Crates.io’s code currently has a one-to-one mapping between crates.io accounts and GitHub accounts.
The URL https://crates.io/users/some_username displays the crates owned by the user with the
GitHub account some_username, and running cargo owner --add some_username adds some_username
as an owner of the current crate. Owners of a crate appear in the sidebar. Crate ownership conveys
trust.
Eventually (after future RFCs and additional work after this RFC), we’d like to add the ability to create crates.io accounts by logging in via OAuth with accounts from services other than GitHub, as well as associating OAuth accounts from multiple services to one crates.io account.
The same username on GitHub is not guaranteed to belong to the same person on other services, and one person’s usernames across different services are not guaranteed to be the same. When we add more services, crates.io’s codebase needs to be able to handle these situations and clearly convey crates.io user identities to minimize the possibility of confusion or deliberate impersonation.
Guide-level explanation
This section will address changes to what users will experience on the crates.io website and via
the cargo owner CLI.
Today, crates.io usernames always match the GitHub username of the account used to log in to crates.io (with exceptions for renamed or deleted GitHub accounts that will be discussed below). The link between a crates.io account and a GitHub account is public information.
After this change, there will be the concept of a crates.io username that may or may not match the GitHub username of the associated account. When there are multiple ways of logging in in the future, the crates.io username may or may not match the usernames on the other services. All existing active (that is, not deleted or renamed) accounts will have their crates.io username set to their current username, their GitHub username.
When you visit your account settings page, you will be able to edit your username to anything that isn’t already claimed as a crates.io username. Thus, crates.io usernames will become first-come-first-served as crate names are today. Crates.io admins will not change an account’s username without the consent of the current username holder, except in cases such as Code of Conduct or usage policy violations like impersonation (see Unresolved Questions about username squatting).
When you visit a user’s page at https://crates.io/users/example_username or see a user account
listed as an owner of a crate in the crate’s sidebar and the account’s crates.io username differs
from the GitHub username associated with the account, you will see some sort of icon to indicate
that something is different (this RFC will use the warning icon ⚠️, but we may decide to use
something more neutral such as an “information” ℹ️, to be determined during implementation).
Accompanying text will says something like “username does not match GitHub username”. Given that
the common case, and what people are used to being able to know, will be that the GitHub and
crates.io usernames will match, this will make it obvious in cases where that assumption does not
hold. We may decide after some transition period (say, 1-2 years) that the username mismatch
warning is no longer needed (especially once crates.io supports OAuth services other than GitHub).
If you run cargo owner --add example_username and the account’s crates.io username differs from
the GitHub username associated with the account, the command will error with a message similar to:
$ cargo owner --add example_username
error: username `example_username` is possibly ambiguous
Caused by:
The crates.io account `example_username` is associated with:
- https://github.com/something_else
- [any other accounts once we have that ability]
To confirm this is the account you want to add, please run one of the following:
$ cargo owner --add cratesio:example_username
$ cargo owner --add github:something_else
If this is not the account you want to add, verify the crates.io username of the account you want.
Returning an error and requesting the user re-run a command with a disambiguation prefix to confirm
is the easiest way to maintain compatibility with existing versions of Cargo. With some additional
work on Cargo, newer versions could be made that only require a y or n confirmation; see the
“Prior Art” section on Keybase for one possibility.
After this RFC is implemented, if you create an account on crates.io with an OAuth account (GitHub or otherwise), whether or not the associated OAuth account’s username is currently claimed on crates.io, you will be asked to register your crates.io account by choosing a username that hasn’t yet been taken on crates.io. The crates.io username field will be prefilled with the associated OAuth account’s username and an indication of whether that username is available on crates.io or not.
Renamed and deleted GitHub accounts
GitHub allows users to change their username (but keep the same GitHub ID number so that crates.io can know it’s the same account) or delete their account, which makes the username available for someone else to claim (with a different GitHub ID number than was previously associated with it).
Crates.io currently does not proactively check GitHub for account status, at the time of this RFC’s
drafting, but we hope to implement proactive
checking before this RFC is implemented and
independently of this RFC’s acceptance. Before this RFC is implemented, we would check with GitHub
for all renames/deletes to update crates.io’s usernames one last time. After this RFC is
implemented, the script would be changed to only update the oauth_github.username and not the
crates.io username.
There’s still a possibility of a window of time between when a user renames or deletes their GitHub account and when crates.io learns about it (depending on how often we check each account), during which the following scenarios could occur.
If someone takes these actions:
- Creates GitHub account with the username “example”
- Logs in to crates.io with that account so that they have the crates.io username “example”
- Renames their GitHub account to “something_else”
- Never logs in to crates.io with that GitHub account again
Their crates.io username will remain “example”, until such a point that they decide to log in to crates.io again. Currently, crates.io will then update their username in our database to match.
This RFC proposes decoupling GitHub account renaming from crates.io username completely, so that GitHub account renames do NOT automatically become crates.io account renames.
A similar situation occurs when a user deletes their GitHub account. The username “example” will then be available for someone else to claim on GitHub, but will remain claimed on crates.io.
If a different user does one of the following:
- Creates the GitHub account “example” and logs in to crates.io
- Tries to edit their username to “example”
- Logs in to crates.io with an account on some service other than GitHub with username “example”
At that point, crates.io will:
- See that the crates.io username “example” is taken
- Require the user with the GitHub username “example” to pick a different crates.io username
If the old “example” account had it via their associated GitHub account (and thus didn’t have the mismatch ⚠️ warning discussed above), then a new associated GitHub account logs in with the GitHub username “example” (and a different GitHub ID), at that point we know the GitHub account “example” does NOT belong to the crates.io account “example” and the crates.io account “example” should get the mismatch ⚠️ warning.
If a user manually changes their crates.io username to best_rust_programmer_ever (and doesn’t
have the matching GitHub account and thus has the warning symbol), and then later someone creates a
GitHub account with the username best_rust_programmer_ever and logs in to crates.io, the GitHub
user best_rust_programmer_ever will need to choose a different crates.io username. Both crates.io
accounts will have the warning symbol. The latter user may see this as unfair, but this is where
the first-come-first-serve policy should be enforced.
Crates.io username requirements
Crates.io usernames will largely use the same rules that GitHub usernames use today. All existing crates.io accounts will be valid under whatever rules we decide on.
Crates.io usernames must:
- Only contain alphanumeric characters
[a-zA-Z0-9], hyphens-, and underscores_1. - Be unique case insensitively and hyphen/underscore insensitively, much like crate names. That is,
uniqueness will be determined by normalizing case and normalizing hyphens and underscores
together. For example, the crates.io usernames
hello-thereandHello_Therewill be considered to be the same: once a user namedhello-thereexists, a user namedHello_Therewill not be allowed. - Not start with a hyphen or underscore. See Unresolved Questions for why not prohibiting ending with these characters or prohibiting two of these characters in a row.
- Not exceed 39 characters.
We have a list of reserved crate names that no one may register that includes top-level Rust standard library modules and keywords, reserved Windows filenames, and some swear words or slurs (which will never be exhaustive but contains the most common ones in English). We’ll have a similar list of reserved usernames that no one may use; GitHub’s Terms of Service is providing us some protection currently that we’d need to manage ourselves.
These requirements will be clearly documented on a page on crates.io as well as in the signup form when we are requiring the person to pick a crates.io username.
Crates.io account rename restrictions
The biggest concerns with allowing crates.io username changes are impersonation and resurrection attacks.
Impersonation is already possible and is already against crates.io policies, but of course we don’t want to make it easier to falsely gain the trust of crates.io users by pretending to be a well-known person. We plan to add typosquatting checks on usernames similar to those we’re already doing for crate names. We also plan to limit how often you can change your crates.io username (say, not more often than once every 30 days).
Resurrection attacks are a subset of impersonation, where a user named carols10cents, for
example, renames away from that username or deletes their account and another user claims the
carols10cents username to appear to be that person to users who don’t know about the rename or
deletion. We plan to limit the re-use of usernames, using a similar mechanism that we have today
that prevents re-use of a deleted crate name, so that no one could claim an abandonded username
for, say, 30 days. We will allow the person who changed away from a username to reclaim/“revert to”
that username within the re-use prevention period.
We also plan to mitigate the effectiveness of impersonation attacks by making the display of the linked accounts associated with a crates.io account very clear so that anyone is able to feel confident that the crates.io account has the same owner as the GitHub, GitLab, etc account they trust.
In the database, accessible by admins only, we will track history of username changes (starting from whenever the feature is implemented; we don’t have historical data of GitHub username changes). This could be useful for forensic investigation of accounts that may be attempting to impersonate other users. We could display historical usernames and their dates on a user’s page for transparency, but this would be problematic in cases such as someone transitioning and wanting to remove all association with their deadname (if their name was part of their crates.io username). We will update the privacy policy section on crates.io to make this retention clear, and we will delete even admin-only viewable information from the database on request.
Crates.io account and GitHub account link privacy
Currently, the knowledge that a crates.io account corresponds to a GitHub account is public information. This link is displayed on user pages and is present in the database dumps. This RFC is deliberately going to continue this status quo and publicly display the GitHub account that a crates.io account is linked with (whether or not the usernames match). We will add documentation in the signup process making it clear that knowledge of your GitHub account will be publicly associated with your crates.io account, and that you should not make a crates.io account with any GitHub account you do not want to have linked with your work on crates.io. See the Private Linked Accounts section under Future Possibilities for the possibility of adding a way to have a private linked account in the future.
Reference-level explanation
This section will address changes to crates.io’s HTTP API. It will not address low-level implementation details of the crates.io database schema or backend code changes; those will be worked out during implementation of this RFC.
User API
The find_user API is currently defined to respond to URLs in the form /api/v1/users/{user},
where {user} is currently the GitHub username that crates.io has been told about for that account.
This route would be changed such that {user} would be assumed to be the crates.io username only.
So for a user with crates.io username carols10cents and GitHub username carolgithub, the API
request /api/v1/users/carols10cents would return this user’s information, and requesting
/api/v1/users/carolgithub would return a 404 Not Found, because there is no crates.io username
carolgithub.
We could choose to have this route’s implementation attempt a lookup in the table of GitHub usernames (and eventually in other services’ tables when those are supported) if no crates.io username is found, but that seems like it could cause confusion.
See the Unresolved Questions section for possibly changing this API’s parameter to optionally accept a prefix with the username to allow lookup by GitHub (or other service) username.
/api/v1/users/{user} currently returns this information (for the carols10cents user):
{
"user": {
"id": 396, // crates.io database ID
"login": "carols10cents", // crates.io and GitHub username
"name": "Carol (Nichols || Goulding)", // display name (as set in GitHub)
"avatar": "https://avatars.githubusercontent.com/u/193874?v=4", // from GitHub
"url": "https://github.com/carols10cents" // assumes GitHub
}
}
This RFC would change the following:
loginwould be the crates.io usernamenamewould be deprecated and the UI would uselogin(we could continue to provide thenameattribute with its value set tologinto ease migration. Also see the “display name” Unresolved Question)- There would still be an
avatarURL returned; for how that is managed, see the Unresolved Questions urlwould be deprecated in favor of explicitly requesting linked account information (we could continue to provide a GitHub URL if the user has a linked GitHub account)- A field named
github_username_matches(with a value of a boolean that the user’s GitHub username is either the same as, or different from, their crates.io username)
So that the response eventually looks like this (without the values supporting deprecated fields
that we may choose to offer for more compatibility) for crates.io user carols10cents that has
GitHub username carolgithub:
{
"user": {
"id": 396, // same as before; crates.io database ID
"login": "carols10cents", // now the crates.io username
"avatar": "https://avatars.githubusercontent.com/u/193874?v=4",
"github_username_matches": false,
}
}
This request would require querying the users table and the oauth_github table, but not any
other services’ linked tables to limit database load by default. The github_username_matches
would be used to decide whether to show the ⚠️ warning about username mismatches discussed in the
Guide section.
If desired, the requester could instead use /api/v1/users/{user}?include=linked_accounts (much
like the current crate API allows for opt-in of returning related data) which would query all OAuth
tables and return all account information for this user once crates.io supports more services:
{
"user": {
"id": 396, // same as before; crates.io database ID
"login": "carols10cents", // now the crates.io username
"avatar": "https://avatars.githubusercontent.com/u/193874?v=4",
"github_username_matches": false,
},
"linked_accounts": [
{
"service": "github",
"login": "carolgithub",
"avatar": "https://avatars.githubusercontent.com/u/193874?v=4"
},
{
"service": "gitlab",
"login": "carols10cents"
"avatar": "https://secure.gravatar.com/avatar/5eefdbf7a532f1a36d5cdce703a3b346cadbebc6098c4fce5354af871f662f55"
}
]
}
The ?include=linked_accounts variant would be called by the frontend when visiting
https://crates.io/users/carols10cents, to be able to display all of a user’s linked accounts.
We would likely not request linked accounts for crate pages when displaying ownership information; we could add a way to view that information on a crate page on-demand, for example when hovering over an owner (or tapping on an icon next to the owner that the frontend shows when viewed on mobile devices), we could request the linked account information then and display a “detail card” for that owner showing the information on their linked accounts.
Owner APIs
The current API request for inviting user owners or adding team owners consists of a PUT request
to /api/v1/crates/[crate name]/owners with the following JSON (using a request to add user
some_user and team some_team from the some_org GitHub organization as an example):
{
"owners": [
"some_user",
"github:some_org:some_team"
]
}
The backend processes owner strings starting with github and containing two colons as an organization name and a team name; this behavior will be unchanged.
This request will begin to accept owners specified by strings containing one colon and starting
with cratesio2, github, and any other OAuth service we eventually add. An owner specification
of cratesio:some_user will only query users.username and not any other table. An owner
specification of github:some_user will only query oauth_github.login and not any other table.
As other services are added, we will add a prefix that can be used to only look up usernames in
that service’s table. If the username isn’t found in the specified table (say, the cratesio
prefix that specfies the users table), the request will return an error even if the username is
in another table (such as the oauth_github table, for this example).
If the owner specification doesn’t contain any colons, the behavior is similar to that of the users
API: we assume it’s a crates.io username and look it up in users.username only. We will also
query the oauth_github table to see if the crates.io username and GitHub username match. If they
do match, we will continue with adding this user as an owner. If they don’t match, we will return
an error containing information about the mismatch and asking the user to rerun the command with a
service prefix and colon in front of the username to ensure we’re adding the account they mean to
add.
An error response would look something like this:
{
"errors": [
{
"detail": "username `some_user` is possibly ambiguous. The crates.io account
`example_username` is associated with:
- https://github.com/something_else
- [any other accounts once we have that ability]
To confirm this is the account you want to add, please run one of the following:
$ cargo owner --add cratesio:example_username
$ cargo owner --add github:something_else
If this is not the account you want to add, verify the crates.io username of the account you want.
"
}
]
}
This maintains backwards compatibilty with existing cargo versions. We could do additional work
on Cargo and add more fields if a newer version of Cargo is making the request, to support a
“confirmation” flow as presented in the “Prior Art” section under Keybase.
The “remove owner” API would behave similarly as the “add owner” API - it will support cratesio:
or github: (etc) prefixes to usernames and will return an error if there is no current owner with
the specified username in the specified service’s table. If given a username without a prefix, the
“remove owner” API will only return an error if there are two current owners of the crate that have
the username on different services, and will then ask the user to rerun with a prefix to
disambiguate. That is, if the user runs cargo owner --remove some_user and there’s a crates.io
user with the users.username of some_user and a different account that has the GitHub user
some_user, the API will only return an error if both these accounts are owners of the crate the
request is being made about. If only one account is an owner, that account will be removed as an
owner.
Drawbacks
- Impedes signup flow if you have to choose a username or try multiple usernames before finding an available one
- Could cause confusion during signup and user lookup
- People who can’t or don’t want to have their GitHub username and crates.io username match will have a warning by their username that they might not want to have there and might imply there’s something wrong or untrustworthy about their account when that isn’t the case
- When crates.io was first created, only implementing GitHub OAuth for authentication was the quickest and lowest effort way to have a functional package registry in the rush to the Rust 1.0 launch. GitHub was, and still is to a slightly lesser extent, the most widely used platform for open source developers. GitHub profiles (as denoted by their username) convey identity which can confer trust. By allowing crates.io usernames to potentially diverge from GitHub usernames, and eventually to potentially have accounts only associated with accounts on service providers other than GitHub, crates.io loses any conferred identity verification or trust bestowed via association. Crates.io could also be more vulnerable to impersonation or account resurrection attacks, as discussed (with potential mitigations) in Crates.io account rename restrictions.
Rationale and alternatives
We could choose to diverge from crates.io’s current behavior more than proposed here, such as:
- We could force everyone to specify whether they mean the crates.io username or GitHub username
for every lookup, to force education that they’re no longer guaranteed to be the same. That is,
if someone runs
cargo owner --add example, we’d always return an error and ask them to reruncargo owner --add cratesio:exampleorcargo owner --add github:exampleexplicitly even if they both refer to the same account. This could be confusing for the most common case, but would be a way to force communication with people that something is changing. - We could implement the backend changes for this RFC but choose to wait to allow username editing until we have multiple ways of logging in.
“Disambiguation page” alternative
We could choose not to have a username field on the users table at all. When visiting
https://crates.io/users/example, crates.io would always look up usernames in all available
oauth_* tables. If only one user record was associated with oauth_* records that had the
username example, we’d display that user’s information. This would nicely handle the most common
case of having one user with a unique GitHub username.
If, however, there is one crates.io user record (id 1234) associated with the oauth_github
record that has the username example, and another crates.io user record (id 5678) associated
with the oauth_gitlab record that has the username example, we could show content similar to a
Wikipedia “disambiguation page”, something like:
There are multiple users with the username “example”. Did you mean:
and then you’d have to click an extra time. We could support direct linking to these users either
by including their crates.io user record ID (something like
https://crates.io/users/example/1234), or by including the name of the service where they hold
the username (something like https://crates.io/users/example/github).
For the cargo owner --add CLI, we could show similar disambiguation text and exit with an error:
$ cargo owner --add example
error: There are multiple users with the username "example".
If you meant https://github.com/example, rerun with `cargo owner --add github:example`.
If you meant https://gitlab.com/example, rerun with `cargo owner --add gitlab:example`.
The disambiguation page and extra specification for users who happen to have colliding usernames with users from other services is a bit more friction and annoyance for those people, through no real fault of their own. However, this case should be fairly rare.
Looking up a user would be more complex and require more queries to more tables, which may impact overall performance.
This idea may not match people’s expectations of how crates.io usernames work, causing confusion.
If a GitHub user with the username best_rust_programmer_ever was well known in Rust spaces but
actually never logged in to crates.io, but someone else claimed the username
best_rust_programmer_ever on GitLab and did log in to crates.io, the content on
https://crates.io/users/best_rust_programmer_ever would only show the GitLab user’s information
with no indication that the GitHub user even exists. We’d need to make the page content clear that
there was only a GitLab account attached, not a GitHub account attached as most people would expect
in most cases.
Prior art
Crates.io appears to be unique among the major OSS package registries in only offering GitHub OAuth, so there aren’t direct lessons we can draw from other ecosystems. Here are a few examples:
PyPI does not currently support changing a username. Instead, you can create a new account with the desired username, add the new account as a maintainer of all the projects your old account owns, and then delete the old account, which will have the same effect. There is no OAuth support.
npm (JavaScript) does not have any OAuth login mechanisms. Their policies state they are “extremely unlikely to transfer control of a username, as it is totally valid to be an npm user and never publish any packages”. It is not currently possible to change your npm username other than creating a new account and migrating data manually. When npm accounts are deleted, usernames become available for anyone to claim again after 30 days.
Maven Central (Java) allows you to create an account and log in via email or OAuth with Google, GitHub, or Microsoft. There is no way to rename, update or change your Maven Central username. If you want a different username, you have to create a new account. However, usernames don’t appear to be as important as they are on crates.io. Maven Central is organized around domain-based namespaces registered through DNS, and it’s the namespace that conveys authority.
Keybase is a service that tries to make working with public key cryptography
easier. They have ways of proving ownership of various accounts on other services to help people
ensure they’re communicating with the account that belongs to the intended person. Keybase also has
a CLI with a confirmation flow that we could use as inspiration for the cargo owner --add user
flow. See the Keybase documentation, under the heading “Step
3: the human review”:
Recall, in Step 2 your client proved “maria” has a number of identities, and it cryptographically verified all of them. Now you can review the usernames it verified, to determine if it’s the maria you wanted.
✔ maria2929 on twitter: https://twitter.com/2131231232133333... ✔ pasc4l_programmer on github: https://gist.github.com/pasc4... ✔ admin of mariah20.com via HTTPS: https://mariah20/keybase.tx... Is this the maria you wanted? [y/N]
With cargo owner --add, once we support multiple logins, the CLI could look something like this:
$ cargo owner --add carols10cents
Crates.io account `carols10cents` is associated with:
✔ https://github.com/carols10cents
✔ https://brand-new-code-hosting-platform.dev/some_other_username
Is this the `carols10cents` you wanted? [y/N]
Unresolved questions
-
How would we define “squatting” of usernames that would be clear cases for admins to make available again? Accounts that don’t have any crates and that don’t have any valid associated OAuth logins (that is, all the associated accounts have been deleted)?
- Selling names is already prohibited by crates.io’s usage policy but transferrability of a username may increase the perceived value and likelihood of username selling attempts. If this becomes a frequent problem, we should explore mitigation mechanisms that will depend on the nature of the incidents and how to avoid false positives that prevent legitimate use.
-
We currently have the concept of a “display name” that is associated with and managed through your GitHub “display name” that’s currently used on crate pages for owners and user pages. Do we want to have an authentication-independent “display name”? Or should we get rid of the “display name” concept completely and only show crates.io usernames everywhere? It seems like the display name would be a vector for possible impersonation (one that’s technically possible today. It is against crates.io policies, and as far as I can remember we haven’t had an account impersonation reported).
-
We are not yet committing to the support of organization/team owners from other services, but adding teams already requires specifying a literal
github:before theorg:teamwhen adding team owners so there shouldn’t be as much confusion around the identity of a team if we choose to add different team owners via different services. -
Should we have a URL for user pages that uses their crates.io ID number (ex:
https://crates.io/users/id/1234would run the querySELECT * FROM users WHERE id = 1234), and thus won’t change no matter what happens with the account’s username or connected accounts?- How would people discover what their crates.io ID is (without needing to look in the API
response)?
- Should we start displaying it on user pages?
- Should we start using these ID-based URLs as the canonical user URLs? That is, should
visiting
https://crates.io/users/carols10centsredirect tohttps://crates.io/users/id/396? - Should we accept it in the CLI, such as
cargo owner --add id:396? And/or should we accept GitHub IDs in the CLI, such ascargo owner --add github_id: 123456789?
- How would people discover what their crates.io ID is (without needing to look in the API
response)?
-
Should we have a URL for user pages or for the user API that allows specifying a prefix of an OAuth service, to allow for looking up a crates.io user when you only know, say, their GitHub username? For example, the user page at
https://crates.io/users/{user}calls thefind_userAPI, which is currently defined to respond to URLs in the form/api/v1/users/{user}where{user}is the username (which is currently the GitHub username but when this RFC is implemented will be the crates.io username).This route could be changed and expanded to allow disambiguation between GitHub, other services, and crates.io usernames. So for a user with crates.io username
carols10centsand GitHub usernamecarolgithub, these API requests could return the same information for this user:/api/v1/users/carols10cents // assumes this is the crates.io username /api/v1/users/cratesio:carols10cents /api/v1/users/github:carolgithubIs this behavior useful and do we want to commit to it in our public API? Do we want to offer it in an experimental form to see how/if it’s used?
-
Is there a way we could avoid having both hyphens and underscores in usernames and needing to normalize them together for uniqueness purposes, which can be confusing?
-
There are usernames in the database currently that contain two hyphens in a row or end in a hyphen. Example: @ra–. I suspect GitHub didn’t initially prohibit this, but now prohibit new accounts from doing this. If we wanted to disallow consecutive hyphens or ending in a hyphen, and someone with a legacy GitHub account like this who hasn’t signed up for crates.io before but does after this change, we could force them to have a crates.io username that doesn’t match their GitHub account. This would mean they’d get the ⚠️ which would be unfair because there’s no way they could match their GitHub account. Alternatively, we could allow consecutive hyphens or ending in hyphen only if that matches your GitHub account exactly. Would this be confusing? Is this a significant number of accounts worth handling specially?
-
Would it be useful to display on the crates.io website that a user had recently changed their username (without displaying what their old username was for privacy reasons, and for some value of “recently”)? This could alert people that they might not have the username of the person they want to add as a crate owner, for example.
-
For a user that recently changed their crates.io username, should we prevent that user from being added as an owner of a crate for a short holding period (1 day, for example) to potentially slow down/prevent impersonation attacks? How often would this be a false positive/annoyance?
Future possibilities
Email/password login
This functionality change would also enable a way of creating crates.io accounts without any associated identity/reputation, only an email address. But this opens more potential for spam and abuse as it’s easier to create anonymous email addresses than it is to maintain accounts in good standing on services like GitHub. When we choose which services to add as OAuth providers, we will assess in what ways the candidate services also provide these protections if we want to continue to have this benefit.
Avatars
Avatars are also indicators of identities; as we evolve crates.io’s capabilities with regards to authentication services, we should evolve how crates.io handles avatars, considering aspects such as:
- Once we have multiple authentication services, each possibly providing an avatar, which do we display when a crates.io user account has different accounts associated and different avatars?
- We currently don’t host avatar images; we don’t really want to as that would add maintenance, bandwidth, and moderation costs. However, hosting avatars ourselves would be the most privacy-preserving, and preferable to a service such as Gravatar that may unintentionally leak personal information.
- Could we let the user choose between:
- Avatar associated with a linked OAuth account, hosted by the linked service (like we do today with GitHub avatars)
- One placeholder image we provide (ex: Ferris)
- Impersonation via avatar is technically possible today with only GitHub unless GitHub policy enforcement disallows that, and I don’t think the decision on avatar resolution is as important as username resolution, but it might make implementation/database queries nicer if we make a similar decision with avatars as with usernames.
Private linked accounts
Currently, the link between a crates.io account and a GitHub account is public information both in the web interface and in the database dump. For this RFC, knowing which GitHub account is linked to which crates.io username will continue to be public information. In the future, we could implement an option to have a crates.io username that uses an authentication service for login but does not expose the link publicly. This feature deserves more thought, design, and its own RFC before implementation. For now, we are deliberately deciding to continue to have the link between a crates.io account and a GitHub account be public information.
-
Even though
github.comdoes not allow you to create an account with a username that contains an underscore, Enterprise Managed Users get a username that ends in_[enterprise shortcode]. We have accounts of this sort in crates.io’s database today. ↩ -
The
cratesioprefix may possibly becrates.io,crates_io,crates-io, or all of them, to be bikeshed during implementation. ↩