2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2023-01-25 21:13:40 +01:00
|
|
|
|
2024-02-24 00:21:09 -05:00
|
|
|
use deno_core::anyhow::Context;
|
2023-01-25 21:13:40 +01:00
|
|
|
use deno_core::error::AnyError;
|
|
|
|
use deno_core::serde_json;
|
|
|
|
use deno_core::url::Url;
|
2024-06-06 23:37:53 -04:00
|
|
|
use deno_runtime::deno_permissions::PermissionsContainer;
|
2023-01-25 21:13:40 +01:00
|
|
|
use import_map::ImportMap;
|
|
|
|
use import_map::ImportMapDiagnostic;
|
|
|
|
use log::warn;
|
|
|
|
|
2023-01-25 16:51:04 -05:00
|
|
|
use super::ConfigFile;
|
|
|
|
use crate::file_fetcher::FileFetcher;
|
|
|
|
|
2024-02-24 00:21:09 -05:00
|
|
|
pub async fn resolve_import_map(
|
|
|
|
specified_specifier: Option<&Url>,
|
2023-01-25 16:51:04 -05:00
|
|
|
maybe_config_file: Option<&ConfigFile>,
|
|
|
|
file_fetcher: &FileFetcher,
|
2024-02-24 00:21:09 -05:00
|
|
|
) -> Result<Option<ImportMap>, AnyError> {
|
|
|
|
if let Some(specifier) = specified_specifier {
|
|
|
|
resolve_import_map_from_specifier(specifier.clone(), file_fetcher)
|
|
|
|
.await
|
|
|
|
.with_context(|| format!("Unable to load '{}' import map", specifier))
|
|
|
|
.map(Some)
|
|
|
|
} else if let Some(config_file) = maybe_config_file {
|
|
|
|
let maybe_url_and_value = config_file
|
|
|
|
.to_import_map_value(|specifier| {
|
|
|
|
let specifier = specifier.clone();
|
|
|
|
async move {
|
|
|
|
let file = file_fetcher
|
2024-04-18 21:43:28 -04:00
|
|
|
.fetch(&specifier, &PermissionsContainer::allow_all())
|
2024-02-24 00:21:09 -05:00
|
|
|
.await?
|
|
|
|
.into_text_decoded()?;
|
|
|
|
Ok(file.source.to_string())
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.await
|
|
|
|
.with_context(|| {
|
|
|
|
format!(
|
|
|
|
"Unable to resolve import map in '{}'",
|
|
|
|
config_file.specifier
|
|
|
|
)
|
|
|
|
})?;
|
|
|
|
match maybe_url_and_value {
|
|
|
|
Some((url, value)) => {
|
|
|
|
import_map_from_value(url.into_owned(), value).map(Some)
|
|
|
|
}
|
|
|
|
None => Ok(None),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn resolve_import_map_from_specifier(
|
|
|
|
specifier: Url,
|
|
|
|
file_fetcher: &FileFetcher,
|
2023-01-25 16:51:04 -05:00
|
|
|
) -> Result<ImportMap, AnyError> {
|
|
|
|
let value: serde_json::Value = if specifier.scheme() == "data" {
|
2024-01-31 22:15:22 -05:00
|
|
|
let data_url_text =
|
2024-02-24 00:21:09 -05:00
|
|
|
deno_graph::source::RawDataUrl::parse(&specifier)?.decode()?;
|
2024-01-31 22:15:22 -05:00
|
|
|
serde_json::from_str(&data_url_text)?
|
2023-01-25 16:51:04 -05:00
|
|
|
} else {
|
2024-02-24 00:21:09 -05:00
|
|
|
let file = file_fetcher
|
2024-04-18 21:43:28 -04:00
|
|
|
.fetch(&specifier, &PermissionsContainer::allow_all())
|
2024-02-24 00:21:09 -05:00
|
|
|
.await?
|
|
|
|
.into_text_decoded()?;
|
|
|
|
serde_json::from_str(&file.source)?
|
2023-01-25 16:51:04 -05:00
|
|
|
};
|
|
|
|
import_map_from_value(specifier, value)
|
|
|
|
}
|
|
|
|
|
2023-11-17 02:28:38 +01:00
|
|
|
pub fn import_map_from_value(
|
2024-02-24 00:21:09 -05:00
|
|
|
specifier: Url,
|
2023-01-25 21:13:40 +01:00
|
|
|
json_value: serde_json::Value,
|
|
|
|
) -> Result<ImportMap, AnyError> {
|
|
|
|
debug_assert!(
|
|
|
|
!specifier.as_str().contains("../"),
|
|
|
|
"Import map specifier incorrectly contained ../: {}",
|
|
|
|
specifier.as_str()
|
|
|
|
);
|
|
|
|
let result = import_map::parse_from_value(specifier, json_value)?;
|
|
|
|
print_import_map_diagnostics(&result.diagnostics);
|
|
|
|
Ok(result.import_map)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_import_map_diagnostics(diagnostics: &[ImportMapDiagnostic]) {
|
|
|
|
if !diagnostics.is_empty() {
|
|
|
|
warn!(
|
|
|
|
"Import map diagnostics:\n{}",
|
|
|
|
diagnostics
|
|
|
|
.iter()
|
2023-01-27 10:43:16 -05:00
|
|
|
.map(|d| format!(" - {d}"))
|
2023-01-25 21:13:40 +01:00
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.join("\n")
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2024-04-23 16:21:06 +01:00
|
|
|
|
|
|
|
pub fn enhance_import_map_value_with_workspace_members(
|
|
|
|
mut import_map_value: serde_json::Value,
|
|
|
|
workspace_members: &[deno_config::WorkspaceMemberConfig],
|
|
|
|
) -> serde_json::Value {
|
|
|
|
let mut imports =
|
|
|
|
if let Some(imports) = import_map_value.get("imports").as_ref() {
|
|
|
|
imports.as_object().unwrap().clone()
|
|
|
|
} else {
|
|
|
|
serde_json::Map::new()
|
|
|
|
};
|
|
|
|
|
|
|
|
for workspace_member in workspace_members {
|
|
|
|
let name = &workspace_member.package_name;
|
|
|
|
let version = &workspace_member.package_version;
|
|
|
|
// Don't override existings, explicit imports
|
|
|
|
if imports.contains_key(name) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
imports.insert(
|
|
|
|
name.to_string(),
|
|
|
|
serde_json::Value::String(format!("jsr:{}@^{}", name, version)),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
import_map_value["imports"] = serde_json::Value::Object(imports);
|
|
|
|
::import_map::ext::expand_import_map_value(import_map_value)
|
|
|
|
}
|