use jsonwebtoken::jwk::{JwkSet, Jwk, CommonParameters, RSAKeyParameters, AlgorithmParameters}; use serde_json::json; use std::env; pub fn get_lti_private_key() -> jsonwebtoken::EncodingKey { let key_str = env::var("LTI_PRIVATE_KEY").unwrap_or_else(|_| { let dev_key_path = "services/lms-service/dev_keys/lti_private.pem"; std::fs::read_to_string(dev_key_path).unwrap_or_else(|_| { // Return a dummy key or handle error gracefully in production // For now, we'll return a string that will likely fail decoding if used, // but allows the service to start if LTI is not used. tracing::warn!("LTI private key not found at {} and LTI_PRIVATE_KEY is not set.", dev_key_path); String::new() }) }); if key_str.is_empty() { // Handle the empty key case - maybe return a specialized error or a dummy key // that fails later. jsonwebtoken::EncodingKey::from_rsa_pem usually expects valid PEM. // We'll use a dummy valid-looking but useless PEM if it's empty to avoid panic on startup // but it will fail on actual LTI usage. return jsonwebtoken::EncodingKey::from_rsa_pem(b"-----BEGIN RSA PRIVATE KEY-----\nMIIEpAIBAAKCAQEA7f...dummy...\n-----END RSA PRIVATE KEY-----").expect("Dummy key failed"); } jsonwebtoken::EncodingKey::from_rsa_pem(key_str.as_bytes()).expect("Invalid LTI private key format") } pub fn get_lti_jwks() -> JwkSet { let n = env::var("LTI_JWK_N").unwrap_or_else(|_| { "weIdo6QklIJW77oEAd0NvX_L1e6mFRpHbSrhWjEJTfQDzLdNV84zPfu-rP-IJdWlvrtO2F_dHHah0ilNRZCaAwPXNqS6L57OrYJjxeDXKWnnfaVw4uUT1aDGFcXQ55Bbf05-N28aj26NEXh9WQVqO6L8XRrleRUgJtb8MBAWovxKi3CBJ_lFVYe31cPeAOCaEF_xzeMVEmJt3fbSewsUIrB7jD8F3YOcu8h_QGAc9tn9uxMfBJv2XZoGHCtMQUGG07iZtoSKBYGrWf5rBc7PsCF_VuQzlO9cf13jgQ2rcfcU3LwC_gp4A9RYnv_ymaHELz0kALKBtBxj1XU7QdLrsw".to_string() }); let jwk = Jwk { common: CommonParameters { public_key_use: Some(jsonwebtoken::jwk::PublicKeyUse::Signature), key_operations: None, key_algorithm: Some(jsonwebtoken::jwk::KeyAlgorithm::RS256), key_id: Some("openccb-lti-key-1".to_string()), x509_url: None, x509_chain: None, x509_sha1_fingerprint: None, x509_sha256_fingerprint: None, }, algorithm: AlgorithmParameters::RSA(RSAKeyParameters { key_type: jsonwebtoken::jwk::RSAKeyType::RSA, n, e: "AQAB".to_string(), }), }; JwkSet { keys: vec![jwk] } } pub async fn lti_jwks_handler() -> axum::Json { let jwks = get_lti_jwks(); axum::Json(json!(jwks)) }