Skip to content

Commit 7d91bcc

Browse files
committed
Duplicate GitHub OAuth info to a linked_accounts table (deploy 1)
That has a `provider` column that will (for now) always be set to 0, which corresponds to `AccountProvider::Github`. The table's primary key is (provider, account_id), which corresponds to (0, gh_id). This constraint will mean a particular GitHub/GitLab/etc account, identified from the provider by an ID, may only be associated with one crates.io user record, but a crates.io user record could (eventually) have *both* a GitHub *and* a GitLab account associated with it (or two GitHub accounts, even!) This is the first step of many to eventually allow for crates.io accounts linked to other OAuth providers in addition/instead of GitHub. No code aside from one test is reading from the linked accounts table at this time. No backfill has been done yet. No handling of creating/associating multiple OAuth accounts with one crates.io account has been done yet.
1 parent 65df7f1 commit 7d91bcc

File tree

10 files changed

+156
-6
lines changed

10 files changed

+156
-6
lines changed

crates/crates_io_database/src/models/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ pub use self::krate::{Crate, CrateName, NewCrate, RecentCrateDownloads};
1414
pub use self::owner::{CrateOwner, Owner, OwnerKind};
1515
pub use self::team::{NewTeam, Team};
1616
pub use self::token::ApiToken;
17-
pub use self::user::{NewUser, User};
17+
pub use self::user::{AccountProvider, LinkedAccount, NewUser, User};
1818
pub use self::version::{NewVersion, TopVersions, Version};
1919

2020
pub mod helpers;

crates/crates_io_database/src/models/user.rs

+27-2
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,8 @@ use diesel_async::{AsyncPgConnection, RunQueryDsl};
88
use secrecy::SecretString;
99

1010
use crate::models::{Crate, CrateOwner, Email, Owner, OwnerKind};
11-
use crate::schema::{crate_owners, emails, users};
12-
use crates_io_diesel_helpers::lower;
11+
use crate::schema::{crate_owners, emails, linked_accounts, users};
12+
use crates_io_diesel_helpers::{lower, pg_enum};
1313

1414
/// The model representing a row in the `users` database table.
1515
#[derive(Clone, Debug, Queryable, Identifiable, Selectable)]
@@ -78,6 +78,31 @@ impl User {
7878
}
7979
}
8080

81+
// Supported OAuth providers. Currently only GitHub.
82+
pg_enum! {
83+
pub enum AccountProvider {
84+
Github = 0,
85+
}
86+
}
87+
88+
/// Represents an OAuth account record linked to a user record.
89+
#[derive(Associations, Identifiable, Selectable, Queryable, Debug, Clone)]
90+
#[diesel(
91+
table_name = linked_accounts,
92+
check_for_backend(diesel::pg::Pg),
93+
primary_key(provider, account_id),
94+
belongs_to(User),
95+
)]
96+
pub struct LinkedAccount {
97+
pub user_id: i32,
98+
pub provider: AccountProvider,
99+
pub account_id: i32, // corresponds to user.gh_id
100+
#[diesel(deserialize_as = String)]
101+
pub access_token: SecretString, // corresponds to user.gh_access_token
102+
pub login: String, // corresponds to user.gh_login
103+
pub avatar: Option<String>, // corresponds to user.gh_avatar
104+
}
105+
81106
/// Represents a new user record insertable to the `users` table
82107
#[derive(Insertable, Debug, Builder)]
83108
#[diesel(table_name = users, check_for_backend(diesel::pg::Pg))]

crates/crates_io_database/src/schema.rs

+46
Original file line numberDiff line numberDiff line change
@@ -593,6 +593,50 @@ diesel::table! {
593593
}
594594
}
595595

596+
diesel::table! {
597+
/// Representation of the `linked_accounts` table.
598+
///
599+
/// (Automatically generated by Diesel.)
600+
linked_accounts (provider, account_id) {
601+
/// The `user_id` column of the `linked_accounts` table.
602+
///
603+
/// Its SQL type is `Int4`.
604+
///
605+
/// (Automatically generated by Diesel.)
606+
user_id -> Int4,
607+
/// The `provider` column of the `linked_accounts` table.
608+
///
609+
/// Its SQL type is `Int4`.
610+
///
611+
/// (Automatically generated by Diesel.)
612+
provider -> Int4,
613+
/// The `account_id` column of the `linked_accounts` table.
614+
///
615+
/// Its SQL type is `Int4`.
616+
///
617+
/// (Automatically generated by Diesel.)
618+
account_id -> Int4,
619+
/// The `access_token` column of the `linked_accounts` table.
620+
///
621+
/// Its SQL type is `Varchar`.
622+
///
623+
/// (Automatically generated by Diesel.)
624+
access_token -> Varchar,
625+
/// The `login` column of the `linked_accounts` table.
626+
///
627+
/// Its SQL type is `Varchar`.
628+
///
629+
/// (Automatically generated by Diesel.)
630+
login -> Varchar,
631+
/// The `avatar` column of the `linked_accounts` table.
632+
///
633+
/// Its SQL type is `Nullable<Varchar>`.
634+
///
635+
/// (Automatically generated by Diesel.)
636+
avatar -> Nullable<Varchar>,
637+
}
638+
}
639+
596640
diesel::table! {
597641
/// Representation of the `metadata` table.
598642
///
@@ -1064,6 +1108,7 @@ diesel::joinable!(dependencies -> versions (version_id));
10641108
diesel::joinable!(emails -> users (user_id));
10651109
diesel::joinable!(follows -> crates (crate_id));
10661110
diesel::joinable!(follows -> users (user_id));
1111+
diesel::joinable!(linked_accounts -> users (user_id));
10671112
diesel::joinable!(publish_limit_buckets -> users (user_id));
10681113
diesel::joinable!(publish_rate_overrides -> users (user_id));
10691114
diesel::joinable!(readme_renderings -> versions (version_id));
@@ -1092,6 +1137,7 @@ diesel::allow_tables_to_appear_in_same_query!(
10921137
emails,
10931138
follows,
10941139
keywords,
1140+
linked_accounts,
10951141
metadata,
10961142
processed_log_files,
10971143
publish_limit_buckets,

crates/crates_io_database_dump/src/dump-db.toml

+10
Original file line numberDiff line numberDiff line change
@@ -154,6 +154,16 @@ keyword = "public"
154154
crates_cnt = "public"
155155
created_at = "public"
156156

157+
[linked_accounts.columns]
158+
user_id = "public"
159+
provider = "public"
160+
account_id = "public"
161+
access_token = "private"
162+
login = "public"
163+
avatar = "public"
164+
[linked_accounts.column_defaults]
165+
access_token = "''"
166+
157167
[metadata.columns]
158168
total_downloads = "public"
159169

crates/crates_io_database_dump/src/snapshots/[email protected]

+2
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,15 @@
11
---
22
source: crates/crates_io_database_dump/src/lib.rs
33
expression: content
4+
snapshot_kind: text
45
---
56
BEGIN ISOLATION LEVEL REPEATABLE READ, READ ONLY;
67

78
\copy "categories" ("category", "crates_cnt", "created_at", "description", "id", "path", "slug") TO 'data/categories.csv' WITH CSV HEADER
89
\copy "crate_downloads" ("crate_id", "downloads") TO 'data/crate_downloads.csv' WITH CSV HEADER
910
\copy "crates" ("created_at", "description", "documentation", "homepage", "id", "max_features", "max_upload_size", "name", "readme", "repository", "updated_at") TO 'data/crates.csv' WITH CSV HEADER
1011
\copy "keywords" ("crates_cnt", "created_at", "id", "keyword") TO 'data/keywords.csv' WITH CSV HEADER
12+
\copy "linked_accounts" ("account_id", "avatar", "login", "provider", "user_id") TO 'data/linked_accounts.csv' WITH CSV HEADER
1113
\copy "metadata" ("total_downloads") TO 'data/metadata.csv' WITH CSV HEADER
1214
\copy "reserved_crate_names" ("name") TO 'data/reserved_crate_names.csv' WITH CSV HEADER
1315
\copy "teams" ("avatar", "github_id", "id", "login", "name", "org_id") TO 'data/teams.csv' WITH CSV HEADER

crates/crates_io_database_dump/src/snapshots/[email protected]

+7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
---
22
source: crates/crates_io_database_dump/src/lib.rs
33
expression: content
4+
snapshot_kind: text
45
---
56
BEGIN;
67
-- Disable triggers on each table.
@@ -9,6 +10,7 @@ BEGIN;
910
ALTER TABLE "crate_downloads" DISABLE TRIGGER ALL;
1011
ALTER TABLE "crates" DISABLE TRIGGER ALL;
1112
ALTER TABLE "keywords" DISABLE TRIGGER ALL;
13+
ALTER TABLE "linked_accounts" DISABLE TRIGGER ALL;
1214
ALTER TABLE "metadata" DISABLE TRIGGER ALL;
1315
ALTER TABLE "reserved_crate_names" DISABLE TRIGGER ALL;
1416
ALTER TABLE "teams" DISABLE TRIGGER ALL;
@@ -23,6 +25,7 @@ BEGIN;
2325

2426
-- Set defaults for non-nullable columns not included in the dump.
2527

28+
ALTER TABLE "linked_accounts" ALTER COLUMN "access_token" SET DEFAULT '';
2629
ALTER TABLE "users" ALTER COLUMN "gh_access_token" SET DEFAULT '';
2730

2831
-- Truncate all tables.
@@ -31,6 +34,7 @@ BEGIN;
3134
TRUNCATE "crate_downloads" RESTART IDENTITY CASCADE;
3235
TRUNCATE "crates" RESTART IDENTITY CASCADE;
3336
TRUNCATE "keywords" RESTART IDENTITY CASCADE;
37+
TRUNCATE "linked_accounts" RESTART IDENTITY CASCADE;
3438
TRUNCATE "metadata" RESTART IDENTITY CASCADE;
3539
TRUNCATE "reserved_crate_names" RESTART IDENTITY CASCADE;
3640
TRUNCATE "teams" RESTART IDENTITY CASCADE;
@@ -52,6 +56,7 @@ BEGIN;
5256
\copy "crate_downloads" ("crate_id", "downloads") FROM 'data/crate_downloads.csv' WITH CSV HEADER
5357
\copy "crates" ("created_at", "description", "documentation", "homepage", "id", "max_features", "max_upload_size", "name", "readme", "repository", "updated_at") FROM 'data/crates.csv' WITH CSV HEADER
5458
\copy "keywords" ("crates_cnt", "created_at", "id", "keyword") FROM 'data/keywords.csv' WITH CSV HEADER
59+
\copy "linked_accounts" ("account_id", "avatar", "login", "provider", "user_id") FROM 'data/linked_accounts.csv' WITH CSV HEADER
5560
\copy "metadata" ("total_downloads") FROM 'data/metadata.csv' WITH CSV HEADER
5661
\copy "reserved_crate_names" ("name") FROM 'data/reserved_crate_names.csv' WITH CSV HEADER
5762
\copy "teams" ("avatar", "github_id", "id", "login", "name", "org_id") FROM 'data/teams.csv' WITH CSV HEADER
@@ -66,6 +71,7 @@ BEGIN;
6671

6772
-- Drop the defaults again.
6873

74+
ALTER TABLE "linked_accounts" ALTER COLUMN "access_token" DROP DEFAULT;
6975
ALTER TABLE "users" ALTER COLUMN "gh_access_token" DROP DEFAULT;
7076

7177
-- Reenable triggers on each table.
@@ -74,6 +80,7 @@ BEGIN;
7480
ALTER TABLE "crate_downloads" ENABLE TRIGGER ALL;
7581
ALTER TABLE "crates" ENABLE TRIGGER ALL;
7682
ALTER TABLE "keywords" ENABLE TRIGGER ALL;
83+
ALTER TABLE "linked_accounts" ENABLE TRIGGER ALL;
7784
ALTER TABLE "metadata" ENABLE TRIGGER ALL;
7885
ALTER TABLE "reserved_crate_names" ENABLE TRIGGER ALL;
7986
ALTER TABLE "teams" ENABLE TRIGGER ALL;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
DROP TABLE linked_accounts;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
CREATE TABLE linked_accounts (
2+
user_id INTEGER NOT NULL REFERENCES users (id) ON DELETE CASCADE,
3+
provider INTEGER NOT NULL,
4+
account_id INTEGER NOT NULL,
5+
access_token VARCHAR NOT NULL,
6+
login VARCHAR NOT NULL,
7+
avatar VARCHAR,
8+
PRIMARY KEY (provider, account_id)
9+
);

src/controllers/session.rs

+21-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ use axum::extract::{FromRequestParts, Query};
22
use axum::Json;
33
use axum_extra::json;
44
use axum_extra::response::ErasedJson;
5+
use diesel::insert_into;
56
use diesel::prelude::*;
67
use diesel_async::scoped_futures::ScopedFutureExt;
78
use diesel_async::{AsyncConnection, AsyncPgConnection, RunQueryDsl};
@@ -12,8 +13,8 @@ use crate::app::AppState;
1213
use crate::controllers::user::update::UserConfirmEmail;
1314
use crate::email::Emails;
1415
use crate::middleware::log_request::RequestLogExt;
15-
use crate::models::{NewEmail, NewUser, User};
16-
use crate::schema::users;
16+
use crate::models::{AccountProvider, NewEmail, NewUser, User};
17+
use crate::schema::{linked_accounts, users};
1718
use crate::util::diesel::is_read_only_error;
1819
use crate::util::errors::{bad_request, server_error, AppResult};
1920
use crate::views::EncodableMe;
@@ -175,6 +176,24 @@ async fn create_or_update_user(
175176
async move {
176177
let user = new_user.insert_or_update(conn).await?;
177178

179+
// To assist in eventually someday allowing OAuth with more than GitHub, also
180+
// write the GitHub info to the `linked_accounts` table. Nothing currently reads
181+
// from this table. Only log errors but don't fail login if this writing fails.
182+
if let Err(e) = insert_into(linked_accounts::table)
183+
.values((
184+
linked_accounts::user_id.eq(user.id),
185+
linked_accounts::provider.eq(AccountProvider::Github),
186+
linked_accounts::account_id.eq(user.gh_id),
187+
linked_accounts::access_token.eq(&new_user.gh_access_token),
188+
linked_accounts::login.eq(&user.gh_login),
189+
linked_accounts::avatar.eq(&user.gh_avatar),
190+
))
191+
.execute(conn)
192+
.await
193+
{
194+
info!("Error inserting linked_accounts record: {e}");
195+
}
196+
178197
// To send the user an account verification email
179198
if let Some(user_email) = email {
180199
let new_email = NewEmail::builder()

src/tests/user.rs

+32-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
use crate::controllers::session;
2-
use crate::models::{ApiToken, Email, User};
2+
use crate::models::{AccountProvider, ApiToken, Email, LinkedAccount, User};
3+
use crate::schema::linked_accounts;
34
use crate::tests::util::github::next_gh_id;
45
use crate::tests::util::{MockCookieUser, RequestHelper};
56
use crate::tests::TestApp;
@@ -265,3 +266,33 @@ async fn test_existing_user_email() -> anyhow::Result<()> {
265266

266267
Ok(())
267268
}
269+
270+
// To assist in eventually someday allowing OAuth with more than GitHub, verify that we're starting
271+
// to also write the GitHub info to the `linked_accounts` table. Nothing currently reads from this
272+
// table other than this test.
273+
#[tokio::test(flavor = "multi_thread")]
274+
async fn also_write_to_linked_accounts() -> anyhow::Result<()> {
275+
let (app, _) = TestApp::init().empty().await;
276+
let mut conn = app.db_conn().await;
277+
278+
let user = app.db_new_user("arbitrary_username").await;
279+
let u = user.as_model();
280+
281+
let linked_accounts = linked_accounts::table
282+
.filter(linked_accounts::provider.eq(AccountProvider::Github))
283+
.filter(linked_accounts::login.eq(&u.gh_login))
284+
.load::<LinkedAccount>(&mut conn)
285+
.await
286+
.unwrap();
287+
288+
assert_eq!(linked_accounts.len(), 1);
289+
let linked_account = &linked_accounts[0];
290+
assert_eq!(linked_account.user_id, u.id);
291+
assert_eq!(linked_account.account_id, u.gh_id);
292+
assert_eq!(
293+
linked_account.access_token.expose_secret(),
294+
u.gh_access_token.expose_secret()
295+
);
296+
297+
Ok(())
298+
}

0 commit comments

Comments
 (0)