// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license. use std::fmt::Write; use std::io::IsTerminal; use std::path::Path; use std::path::PathBuf; use std::sync::Arc; use base64::prelude::BASE64_STANDARD; use base64::Engine; use deno_config::ConfigFile; use deno_core::anyhow; use deno_core::anyhow::anyhow; use deno_core::anyhow::bail; use deno_core::anyhow::Context; use deno_core::error::AnyError; use deno_core::serde_json; use deno_core::serde_json::json; use deno_core::url::Url; use deno_runtime::colors; use deno_runtime::deno_fetch::reqwest; use http::header::AUTHORIZATION; use http::header::CONTENT_ENCODING; use hyper::body::Bytes; use import_map::ImportMapWithDiagnostics; use serde::de::DeserializeOwned; use serde::Serialize; use sha2::Digest; use crate::args::Flags; use crate::args::PublishFlags; use crate::factory::CliFactory; use crate::http_util::HttpClient; use crate::util::import_map::ImportMapUnfurler; mod tar; enum AuthMethod { Interactive, Token(String), Oidc(OidcConfig), } struct OidcConfig { url: String, token: String, } struct PreparedPublishPackage { scope: String, package: String, version: String, tarball_hash: String, tarball: Bytes, } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] pub struct PublishingTaskError { pub code: String, pub message: String, } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] pub struct PublishingTask { pub id: String, pub status: String, pub error: Option, } async fn prepare_publish( initial_cwd: &Path, directory: PathBuf, ) -> Result { // TODO: handle publishing without deno.json let directory_path = initial_cwd.join(directory); // TODO: doesn't handle jsonc let deno_json_path = directory_path.join("deno.json"); let deno_json = ConfigFile::read(&deno_json_path).with_context(|| { format!( "Failed to read deno configuration file at {}", deno_json_path.display() ) })?; let Some(version) = deno_json.json.version.clone() else { bail!("{} is missing 'version' field", deno_json_path.display()); }; let Some(name) = deno_json.json.name.clone() else { bail!("{} is missing 'name' field", deno_json_path.display()); }; let Some(name) = name.strip_prefix('@') else { bail!("Invalid package name, use '@/ format"); }; let Some((scope, package_name)) = name.split_once('/') else { bail!("Invalid package name, use '@/ format"); }; // TODO: support `importMap` field in deno.json assert!(deno_json.to_import_map_path().is_none()); let deno_json_url = Url::from_file_path(&deno_json_path) .map_err(|_| anyhow!("deno.json path is not a valid file URL"))?; let ImportMapWithDiagnostics { import_map, .. } = import_map::parse_from_value( &deno_json_url, deno_json.to_import_map_value(), )?; let unfurler = ImportMapUnfurler::new(import_map); let tarball = tar::create_gzipped_tarball(directory_path, unfurler) .context("Failed to create a tarball")?; let tarball_hash_bytes: Vec = sha2::Sha256::digest(&tarball).iter().cloned().collect(); let mut tarball_hash = "sha256-".to_string(); for byte in tarball_hash_bytes { write!(&mut tarball_hash, "{:02x}", byte).unwrap(); } Ok(PreparedPublishPackage { scope: scope.to_string(), package: package_name.to_string(), version: version.to_string(), tarball_hash, tarball, }) } #[derive(Serialize)] #[serde(tag = "permission")] pub enum Permission<'s> { #[serde(rename = "package/publish", rename_all = "camelCase")] VersionPublish { scope: &'s str, package: &'s str, version: &'s str, tarball_hash: &'s str, }, } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] struct CreateAuthorizationResponse { verification_url: String, code: String, exchange_token: String, poll_interval: u64, } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] struct ExchangeAuthorizationResponse { token: String, user: User, } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] struct User { name: String, } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] struct ApiError { pub code: String, pub message: String, #[serde(skip)] pub x_deno_ray: Option, } impl std::fmt::Display for ApiError { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(f, "{} ({})", self.message, self.code)?; if let Some(x_deno_ray) = &self.x_deno_ray { write!(f, "[x-deno-ray: {}]", x_deno_ray)?; } Ok(()) } } impl std::fmt::Debug for ApiError { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { std::fmt::Display::fmt(self, f) } } impl std::error::Error for ApiError {} async fn parse_response( response: reqwest::Response, ) -> Result { let status = response.status(); let x_deno_ray = response .headers() .get("x-deno-ray") .and_then(|value| value.to_str().ok()) .map(|s| s.to_string()); let text = response.text().await.unwrap(); if !status.is_success() { match serde_json::from_str::(&text) { Ok(mut err) => { err.x_deno_ray = x_deno_ray; return Err(err); } Err(_) => { let err = ApiError { code: "unknown".to_string(), message: format!("{}: {}", status, text), x_deno_ray, }; return Err(err); } } } serde_json::from_str(&text).map_err(|err| ApiError { code: "unknown".to_string(), message: format!("Failed to parse response: {}, response: '{}'", err, text), x_deno_ray, }) } #[derive(serde::Deserialize)] #[serde(rename_all = "camelCase")] struct OidcTokenResponse { value: String, } async fn perform_publish( http_client: &Arc, packages: Vec, auth_method: AuthMethod, ) -> Result<(), AnyError> { let client = http_client.client()?; let registry_url = crate::cache::DENO_REGISTRY_URL.to_string(); let authorization = match auth_method { AuthMethod::Interactive => { let verifier = uuid::Uuid::new_v4().to_string(); let challenge = BASE64_STANDARD.encode(sha2::Sha256::digest(&verifier)); let permissions = packages .iter() .map(|package| Permission::VersionPublish { scope: &package.scope, package: &package.package, version: &package.version, tarball_hash: &package.tarball_hash, }) .collect::>(); let response = client .post(format!("{}authorizations", registry_url)) .json(&serde_json::json!({ "challenge": challenge, "permissions": permissions, })) .send() .await .context("Failed to create interactive authorization")?; let auth = parse_response::(response) .await .context("Failed to create interactive authorization")?; print!( "Visit {} to authorize publishing of", colors::cyan(format!("{}?code={}", auth.verification_url, auth.code)) ); if packages.len() > 1 { println!(" {} packages", packages.len()); } else { println!(" @{}/{}", packages[0].scope, packages[0].package); } println!("{}", colors::gray("Waiting...")); let interval = std::time::Duration::from_secs(auth.poll_interval); loop { tokio::time::sleep(interval).await; let response = client .post(format!("{}authorizations/exchange", registry_url)) .json(&serde_json::json!({ "exchangeToken": auth.exchange_token, "verifier": verifier, })) .send() .await .context("Failed to exchange authorization")?; let res = parse_response::(response).await; match res { Ok(res) => { println!( "{} {} {}", colors::green("Authorization successful."), colors::gray("Authenticated as"), colors::cyan(res.user.name) ); break format!("Bearer {}", res.token); } Err(err) => { if err.code == "authorizationPending" { continue; } else { return Err(err).context("Failed to exchange authorization"); } } } } } AuthMethod::Token(token) => format!("Bearer {}", token), AuthMethod::Oidc(oidc_config) => { let permissions = packages .iter() .map(|package| Permission::VersionPublish { scope: &package.scope, package: &package.package, version: &package.version, tarball_hash: &package.tarball_hash, }) .collect::>(); let audience = json!({ "permissions": permissions }).to_string(); let url = format!( "{}&audience={}", oidc_config.url, percent_encoding::percent_encode( audience.as_bytes(), percent_encoding::NON_ALPHANUMERIC ) ); let response = client .get(url) .bearer_auth(oidc_config.token) .send() .await .context("Failed to get OIDC token")?; let status = response.status(); let text = response.text().await.with_context(|| { format!("Failed to get OIDC token: status {}", status) })?; if !status.is_success() { bail!( "Failed to get OIDC token: status {}, response: '{}'", status, text ); } let OidcTokenResponse { value } = serde_json::from_str(&text) .with_context(|| { format!("Failed to parse OIDC token: '{}' (status {})", text, status) })?; format!("githuboidc {}", value) } }; for package in packages { println!( "{} @{}/{}@{} ...", colors::intense_blue("Publishing"), package.scope, package.package, package.version ); let url = format!( "{}scopes/{}/packages/{}/versions/{}", registry_url, package.scope, package.package, package.version ); let response = client .post(url) .header(AUTHORIZATION, &authorization) .header(CONTENT_ENCODING, "gzip") .body(package.tarball) .send() .await?; let mut task = parse_response::(response) .await .with_context(|| { format!( "Failed to publish @{}/{} at {}", package.scope, package.package, package.version ) })?; let interval = std::time::Duration::from_secs(2); while task.status != "success" && task.status != "failure" { tokio::time::sleep(interval).await; let resp = client .get(format!("{}publish_status/{}", registry_url, task.id)) .send() .await .with_context(|| { format!( "Failed to get publishing status for @{}/{} at {}", package.scope, package.package, package.version ) })?; task = parse_response::(resp) .await .with_context(|| { format!( "Failed to get publishing status for @{}/{} at {}", package.scope, package.package, package.version ) })?; } if let Some(error) = task.error { bail!( "{} @{}/{} at {}: {}", colors::red("Failed to publish"), package.scope, package.package, package.version, error.message ); } println!( "{} @{}/{}@{}", colors::green("Successfully published"), package.scope, package.package, package.version ); println!( "{}/@{}/{}/{}_meta.json", registry_url, package.scope, package.package, package.version ); } Ok(()) } fn get_gh_oidc_env_vars() -> Option> { if std::env::var("GITHUB_ACTIONS").unwrap_or_default() == "true" { let url = std::env::var("ACTIONS_ID_TOKEN_REQUEST_URL"); let token = std::env::var("ACTIONS_ID_TOKEN_REQUEST_TOKEN"); match (url, token) { (Ok(url), Ok(token)) => Some(Ok((url, token))), (Err(_), Err(_)) => Some(Err(anyhow::anyhow!( "No means to authenticate. Pass a token to `--token`, or enable tokenless publishing from GitHub Actions using OIDC. Learn more at https://deno.co/ghoidc" ))), _ => None, } } else { None } } pub async fn publish( flags: Flags, publish_flags: PublishFlags, ) -> Result<(), AnyError> { let cli_factory = CliFactory::from_flags(flags).await?; let auth_method = match publish_flags.token { Some(token) => AuthMethod::Token(token), None => match get_gh_oidc_env_vars() { Some(Ok((url, token))) => AuthMethod::Oidc(OidcConfig { url, token }), Some(Err(err)) => return Err(err), None if std::io::stdin().is_terminal() => AuthMethod::Interactive, None => { bail!("No means to authenticate. Pass a token to `--token`.") } }, }; let initial_cwd = std::env::current_dir().with_context(|| "Failed getting cwd.")?; let directory_path = initial_cwd.join(publish_flags.directory); // TODO: doesn't handle jsonc let deno_json_path = directory_path.join("deno.json"); let deno_json = ConfigFile::read(&deno_json_path).with_context(|| { format!( "Failed to read deno.json file at {}", deno_json_path.display() ) })?; let mut packages = Vec::with_capacity(std::cmp::max(1, deno_json.json.workspaces.len())); let members = &deno_json.json.workspaces; if members.is_empty() { packages.push(prepare_publish(&initial_cwd, directory_path).await?); } else { println!("Publishing a workspace..."); for member in members { let member_dir = directory_path.join(member); packages.push(prepare_publish(&initial_cwd, member_dir).await?); } } if packages.is_empty() { bail!("No packages to publish"); } perform_publish(cli_factory.http_client(), packages, auth_method).await }