mirror of
https://github.com/openai/codex.git
synced 2026-04-28 02:11:08 +03:00
103 lines
3.0 KiB
Rust
103 lines
3.0 KiB
Rust
use super::*;
|
|
use chrono::TimeZone;
|
|
use chrono::Utc;
|
|
use pretty_assertions::assert_eq;
|
|
use serde::Serialize;
|
|
|
|
fn fake_jwt(payload: serde_json::Value) -> String {
|
|
#[derive(Serialize)]
|
|
struct Header {
|
|
alg: &'static str,
|
|
typ: &'static str,
|
|
}
|
|
let header = Header {
|
|
alg: "none",
|
|
typ: "JWT",
|
|
};
|
|
|
|
fn b64url_no_pad(bytes: &[u8]) -> String {
|
|
base64::engine::general_purpose::URL_SAFE_NO_PAD.encode(bytes)
|
|
}
|
|
|
|
let header_b64 = b64url_no_pad(&serde_json::to_vec(&header).unwrap());
|
|
let payload_b64 = b64url_no_pad(&serde_json::to_vec(&payload).unwrap());
|
|
let signature_b64 = b64url_no_pad(b"sig");
|
|
format!("{header_b64}.{payload_b64}.{signature_b64}")
|
|
}
|
|
|
|
#[test]
|
|
fn id_token_info_parses_email_and_plan() {
|
|
let fake_jwt = fake_jwt(serde_json::json!({
|
|
"email": "user@example.com",
|
|
"https://api.openai.com/auth": {
|
|
"chatgpt_plan_type": "pro"
|
|
}
|
|
}));
|
|
|
|
let info = parse_chatgpt_jwt_claims(&fake_jwt).expect("should parse");
|
|
assert_eq!(info.email.as_deref(), Some("user@example.com"));
|
|
assert_eq!(info.get_chatgpt_plan_type().as_deref(), Some("Pro"));
|
|
}
|
|
|
|
#[test]
|
|
fn id_token_info_parses_go_plan() {
|
|
let fake_jwt = fake_jwt(serde_json::json!({
|
|
"email": "user@example.com",
|
|
"https://api.openai.com/auth": {
|
|
"chatgpt_plan_type": "go"
|
|
}
|
|
}));
|
|
|
|
let info = parse_chatgpt_jwt_claims(&fake_jwt).expect("should parse");
|
|
assert_eq!(info.email.as_deref(), Some("user@example.com"));
|
|
assert_eq!(info.get_chatgpt_plan_type().as_deref(), Some("Go"));
|
|
}
|
|
|
|
#[test]
|
|
fn id_token_info_handles_missing_fields() {
|
|
let fake_jwt = fake_jwt(serde_json::json!({ "sub": "123" }));
|
|
|
|
let info = parse_chatgpt_jwt_claims(&fake_jwt).expect("should parse");
|
|
assert!(info.email.is_none());
|
|
assert!(info.get_chatgpt_plan_type().is_none());
|
|
}
|
|
|
|
#[test]
|
|
fn jwt_expiration_parses_exp_claim() {
|
|
let fake_jwt = fake_jwt(serde_json::json!({
|
|
"exp": 1_700_000_000_i64,
|
|
}));
|
|
|
|
let expires_at = parse_jwt_expiration(&fake_jwt).expect("should parse");
|
|
assert_eq!(expires_at, Utc.timestamp_opt(1_700_000_000, 0).single());
|
|
}
|
|
|
|
#[test]
|
|
fn jwt_expiration_handles_missing_exp() {
|
|
let fake_jwt = fake_jwt(serde_json::json!({ "sub": "123" }));
|
|
|
|
let expires_at = parse_jwt_expiration(&fake_jwt).expect("should parse");
|
|
assert_eq!(expires_at, None);
|
|
}
|
|
|
|
#[test]
|
|
fn jwt_expiration_rejects_malformed_jwt() {
|
|
let err = parse_jwt_expiration("not-a-jwt").expect_err("should fail");
|
|
assert_eq!(err.to_string(), "invalid ID token format");
|
|
}
|
|
|
|
#[test]
|
|
fn workspace_account_detection_matches_workspace_plans() {
|
|
let workspace = IdTokenInfo {
|
|
chatgpt_plan_type: Some(PlanType::Known(KnownPlan::Business)),
|
|
..IdTokenInfo::default()
|
|
};
|
|
assert_eq!(workspace.is_workspace_account(), true);
|
|
|
|
let personal = IdTokenInfo {
|
|
chatgpt_plan_type: Some(PlanType::Known(KnownPlan::Pro)),
|
|
..IdTokenInfo::default()
|
|
};
|
|
assert_eq!(personal.is_workspace_account(), false);
|
|
}
|