2024-02-29 19:12:04 +00:00
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
2024-05-08 20:34:46 +01:00
use std ::borrow ::Cow ;
2024-08-12 16:17:25 -04:00
use std ::path ::Path ;
2024-02-29 19:12:04 +00:00
use std ::path ::PathBuf ;
2024-03-01 21:34:13 +00:00
use std ::sync ::Arc ;
2024-02-29 19:12:04 +00:00
use deno_ast ::TextChange ;
2024-07-19 15:56:07 -04:00
use deno_config ::deno_json ::FmtOptionsConfig ;
2024-05-08 20:34:46 +01:00
use deno_core ::anyhow ::anyhow ;
2024-02-29 19:12:04 +00:00
use deno_core ::anyhow ::bail ;
use deno_core ::anyhow ::Context ;
use deno_core ::error ::AnyError ;
use deno_core ::futures ::FutureExt ;
use deno_core ::futures ::StreamExt ;
use deno_core ::serde_json ;
2024-05-08 20:34:46 +01:00
use deno_core ::ModuleSpecifier ;
use deno_runtime ::deno_node ;
2024-06-10 19:56:43 -04:00
use deno_semver ::package ::PackageReq ;
2024-05-08 20:34:46 +01:00
use indexmap ::IndexMap ;
2024-02-29 19:12:04 +00:00
use jsonc_parser ::ast ::ObjectProp ;
use jsonc_parser ::ast ::Value ;
use crate ::args ::AddFlags ;
use crate ::args ::CacheSetting ;
use crate ::args ::Flags ;
2024-08-12 16:17:25 -04:00
use crate ::args ::RemoveFlags ;
2024-02-29 19:12:04 +00:00
use crate ::factory ::CliFactory ;
use crate ::file_fetcher ::FileFetcher ;
2024-03-01 21:34:13 +00:00
use crate ::jsr ::JsrFetchResolver ;
2024-03-06 13:24:15 +00:00
use crate ::npm ::NpmFetchResolver ;
2024-02-29 19:12:04 +00:00
2024-05-08 20:34:46 +01:00
enum DenoConfigFormat {
Json ,
Jsonc ,
}
impl DenoConfigFormat {
fn from_specifier ( spec : & ModuleSpecifier ) -> Result < Self , AnyError > {
let file_name = spec
. path_segments ( )
. ok_or_else ( | | anyhow! ( " Empty path in deno config specifier: {spec} " ) ) ?
. last ( )
. unwrap ( ) ;
match file_name {
" deno.json " = > Ok ( Self ::Json ) ,
" deno.jsonc " = > Ok ( Self ::Jsonc ) ,
_ = > bail! ( " Unsupported deno config file: {file_name} " ) ,
}
}
}
enum DenoOrPackageJson {
2024-07-19 15:56:07 -04:00
Deno ( Arc < deno_config ::deno_json ::ConfigFile > , DenoConfigFormat ) ,
2024-06-26 17:24:10 -04:00
Npm ( Arc < deno_node ::PackageJson > , Option < FmtOptionsConfig > ) ,
2024-05-08 20:34:46 +01:00
}
impl DenoOrPackageJson {
fn specifier ( & self ) -> Cow < ModuleSpecifier > {
match self {
Self ::Deno ( d , .. ) = > Cow ::Borrowed ( & d . specifier ) ,
Self ::Npm ( n , .. ) = > Cow ::Owned ( n . specifier ( ) ) ,
}
}
/// Returns the existing imports/dependencies from the config.
fn existing_imports ( & self ) -> Result < IndexMap < String , String > , AnyError > {
match self {
DenoOrPackageJson ::Deno ( deno , .. ) = > {
if let Some ( imports ) = deno . json . imports . clone ( ) {
match serde_json ::from_value ( imports ) {
Ok ( map ) = > Ok ( map ) ,
Err ( err ) = > {
bail! ( " Malformed \" imports \" configuration: {err} " )
}
}
} else {
Ok ( Default ::default ( ) )
}
}
DenoOrPackageJson ::Npm ( npm , .. ) = > {
Ok ( npm . dependencies . clone ( ) . unwrap_or_default ( ) )
}
}
}
fn fmt_options ( & self ) -> FmtOptionsConfig {
match self {
DenoOrPackageJson ::Deno ( deno , .. ) = > deno
. to_fmt_config ( )
. ok ( )
. map ( | f | f . options )
. unwrap_or_default ( ) ,
DenoOrPackageJson ::Npm ( _ , config ) = > config . clone ( ) . unwrap_or_default ( ) ,
}
}
fn imports_key ( & self ) -> & 'static str {
match self {
DenoOrPackageJson ::Deno ( .. ) = > " imports " ,
DenoOrPackageJson ::Npm ( .. ) = > " dependencies " ,
}
}
fn file_name ( & self ) -> & 'static str {
match self {
DenoOrPackageJson ::Deno ( _ , format ) = > match format {
DenoConfigFormat ::Json = > " deno.json " ,
DenoConfigFormat ::Jsonc = > " deno.jsonc " ,
} ,
DenoOrPackageJson ::Npm ( .. ) = > " package.json " ,
}
}
fn is_npm ( & self ) -> bool {
matches! ( self , Self ::Npm ( .. ) )
}
/// Get the preferred config file to operate on
/// given the flags. If no config file is present,
/// creates a `deno.json` file - in this case
/// we also return a new `CliFactory` that knows about
/// the new config
2024-07-23 19:00:48 -04:00
fn from_flags ( flags : Arc < Flags > ) -> Result < ( Self , CliFactory ) , AnyError > {
let factory = CliFactory ::from_flags ( flags . clone ( ) ) ;
let options = factory . cli_options ( ) ? ;
2024-07-19 15:56:07 -04:00
let start_dir = & options . start_dir ;
2024-05-08 20:34:46 +01:00
2024-07-19 15:56:07 -04:00
match ( start_dir . maybe_deno_json ( ) , start_dir . maybe_pkg_json ( ) ) {
2024-05-08 20:34:46 +01:00
// when both are present, for now,
// default to deno.json
( Some ( deno ) , Some ( _ ) | None ) = > Ok ( (
DenoOrPackageJson ::Deno (
deno . clone ( ) ,
DenoConfigFormat ::from_specifier ( & deno . specifier ) ? ,
) ,
factory ,
) ) ,
( None , Some ( package_json ) ) if options . enable_future_features ( ) = > {
Ok ( ( DenoOrPackageJson ::Npm ( package_json . clone ( ) , None ) , factory ) )
}
( None , Some ( _ ) | None ) = > {
std ::fs ::write ( options . initial_cwd ( ) . join ( " deno.json " ) , " {} \n " )
. context ( " Failed to create deno.json file " ) ? ;
2024-07-19 15:56:07 -04:00
drop ( factory ) ; // drop to prevent use
2024-05-08 20:34:46 +01:00
log ::info! ( " Created deno.json configuration file. " ) ;
2024-07-23 19:00:48 -04:00
let factory = CliFactory ::from_flags ( flags . clone ( ) ) ;
let options = factory . cli_options ( ) ? . clone ( ) ;
2024-07-19 15:56:07 -04:00
let start_dir = & options . start_dir ;
2024-05-08 20:34:46 +01:00
Ok ( (
DenoOrPackageJson ::Deno (
2024-07-19 15:56:07 -04:00
start_dir . maybe_deno_json ( ) . cloned ( ) . ok_or_else ( | | {
2024-07-03 20:54:33 -04:00
anyhow! ( " config not found, but it was just created " )
} ) ? ,
2024-05-08 20:34:46 +01:00
DenoConfigFormat ::Json ,
) ,
2024-07-03 20:54:33 -04:00
factory ,
2024-05-08 20:34:46 +01:00
) )
}
}
}
}
fn package_json_dependency_entry (
selected : SelectedPackage ,
) -> ( String , String ) {
if let Some ( npm_package ) = selected . package_name . strip_prefix ( " npm: " ) {
( npm_package . into ( ) , selected . version_req )
} else if let Some ( jsr_package ) = selected . package_name . strip_prefix ( " jsr: " ) {
let jsr_package = jsr_package . strip_prefix ( '@' ) . unwrap_or ( jsr_package ) ;
let scope_replaced = jsr_package . replace ( '/' , " __ " ) ;
let version_req =
format! ( " npm:@jsr/ {scope_replaced} @ {} " , selected . version_req ) ;
( selected . import_name , version_req )
} else {
( selected . package_name , selected . version_req )
}
}
2024-08-09 16:29:11 +02:00
#[ derive(Clone, Copy) ]
/// The name of the subcommand invoking the `add` operation.
pub enum AddCommandName {
Add ,
Install ,
}
impl std ::fmt ::Display for AddCommandName {
fn fmt ( & self , f : & mut std ::fmt ::Formatter < '_ > ) -> std ::fmt ::Result {
match self {
AddCommandName ::Add = > write! ( f , " add " ) ,
AddCommandName ::Install = > write! ( f , " install " ) ,
}
}
}
2024-07-23 19:00:48 -04:00
pub async fn add (
flags : Arc < Flags > ,
add_flags : AddFlags ,
2024-08-09 16:29:11 +02:00
cmd_name : AddCommandName ,
2024-07-23 19:00:48 -04:00
) -> Result < ( ) , AnyError > {
2024-05-08 20:34:46 +01:00
let ( config_file , cli_factory ) =
DenoOrPackageJson ::from_flags ( flags . clone ( ) ) ? ;
2024-02-29 19:12:04 +00:00
2024-05-08 20:34:46 +01:00
let config_specifier = config_file . specifier ( ) ;
if config_specifier . scheme ( ) ! = " file " {
2024-02-29 19:12:04 +00:00
bail! ( " Can't add dependencies to a remote configuration file " ) ;
}
2024-05-08 20:34:46 +01:00
let config_file_path = config_specifier . to_file_path ( ) . unwrap ( ) ;
2024-02-29 19:12:04 +00:00
2024-06-03 17:17:08 -04:00
let http_client = cli_factory . http_client_provider ( ) ;
2024-02-29 19:12:04 +00:00
let mut selected_packages = Vec ::with_capacity ( add_flags . packages . len ( ) ) ;
let mut package_reqs = Vec ::with_capacity ( add_flags . packages . len ( ) ) ;
2024-06-10 19:56:43 -04:00
for entry_text in add_flags . packages . iter ( ) {
let req = AddPackageReq ::parse ( entry_text ) . with_context ( | | {
format! ( " Failed to parse package required: {} " , entry_text )
} ) ? ;
2024-02-29 19:12:04 +00:00
package_reqs . push ( req ) ;
}
let deps_http_cache = cli_factory . global_http_cache ( ) ? ;
let mut deps_file_fetcher = FileFetcher ::new (
deps_http_cache . clone ( ) ,
CacheSetting ::ReloadAll ,
true ,
http_client . clone ( ) ,
Default ::default ( ) ,
None ,
) ;
deps_file_fetcher . set_download_log_level ( log ::Level ::Trace ) ;
2024-06-03 17:17:08 -04:00
let deps_file_fetcher = Arc ::new ( deps_file_fetcher ) ;
2024-03-06 13:24:15 +00:00
let jsr_resolver = Arc ::new ( JsrFetchResolver ::new ( deps_file_fetcher . clone ( ) ) ) ;
let npm_resolver = Arc ::new ( NpmFetchResolver ::new ( deps_file_fetcher ) ) ;
2024-02-29 19:12:04 +00:00
let package_futures = package_reqs
. into_iter ( )
2024-03-01 21:34:13 +00:00
. map ( move | package_req | {
2024-03-06 13:24:15 +00:00
find_package_and_select_version_for_req (
jsr_resolver . clone ( ) ,
npm_resolver . clone ( ) ,
package_req ,
)
. boxed_local ( )
2024-02-29 19:12:04 +00:00
} )
. collect ::< Vec < _ > > ( ) ;
let stream_of_futures = deno_core ::futures ::stream ::iter ( package_futures ) ;
2024-08-18 17:02:32 -04:00
let mut buffered = stream_of_futures . buffered ( 10 ) ;
2024-02-29 19:12:04 +00:00
while let Some ( package_and_version_result ) = buffered . next ( ) . await {
let package_and_version = package_and_version_result ? ;
match package_and_version {
2024-08-09 16:29:11 +02:00
PackageAndVersion ::NotFound {
package : package_name ,
found_npm_package ,
package_req ,
} = > {
if found_npm_package {
bail! ( " {} was not found, but a matching npm package exists. Did you mean `{}`? " , crate ::colors ::red ( package_name ) , crate ::colors ::yellow ( format! ( " deno {cmd_name} npm: {package_req} " ) ) ) ;
} else {
bail! ( " {} was not found. " , crate ::colors ::red ( package_name ) ) ;
}
2024-02-29 19:12:04 +00:00
}
PackageAndVersion ::Selected ( selected ) = > {
selected_packages . push ( selected ) ;
}
}
}
2024-04-19 04:48:15 +09:00
let config_file_contents = {
let contents = tokio ::fs ::read_to_string ( & config_file_path ) . await . unwrap ( ) ;
if contents . trim ( ) . is_empty ( ) {
" {} \n " . into ( )
} else {
contents
}
} ;
2024-02-29 19:12:04 +00:00
let ast = jsonc_parser ::parse_to_ast (
& config_file_contents ,
& Default ::default ( ) ,
& Default ::default ( ) ,
) ? ;
let obj = match ast . value {
Some ( Value ::Object ( obj ) ) = > obj ,
_ = > bail! ( " Failed updating config file due to no object. " ) ,
} ;
2024-08-20 16:32:01 -04:00
if obj . get_string ( " importMap " ) . is_some ( ) {
bail! (
concat! (
" `deno add` is not supported when configuration file contains an \" importMap \" field. " ,
" Inline the import map into the Deno configuration file. \n " ,
" at {} " ,
) ,
config_specifier
) ;
}
2024-05-08 20:34:46 +01:00
let mut existing_imports = config_file . existing_imports ( ) ? ;
2024-02-29 19:12:04 +00:00
2024-05-08 20:34:46 +01:00
let is_npm = config_file . is_npm ( ) ;
2024-02-29 19:12:04 +00:00
for selected_package in selected_packages {
log ::info! (
2024-08-08 15:25:05 +01:00
" Add {}{}{} " ,
crate ::colors ::green ( & selected_package . package_name ) ,
crate ::colors ::gray ( " @ " ) ,
selected_package . selected_version
2024-02-29 19:12:04 +00:00
) ;
2024-05-08 20:34:46 +01:00
if is_npm {
let ( name , version ) = package_json_dependency_entry ( selected_package ) ;
existing_imports . insert ( name , version )
} else {
existing_imports . insert (
selected_package . import_name ,
format! (
" {}@{} " ,
selected_package . package_name , selected_package . version_req
) ,
)
} ;
2024-02-29 19:12:04 +00:00
}
let mut import_list : Vec < ( String , String ) > =
existing_imports . into_iter ( ) . collect ( ) ;
import_list . sort_by ( | ( k1 , _ ) , ( k2 , _ ) | k1 . cmp ( k2 ) ) ;
let generated_imports = generate_imports ( import_list ) ;
2024-05-08 20:34:46 +01:00
let fmt_config_options = config_file . fmt_options ( ) ;
2024-02-29 19:12:04 +00:00
let new_text = update_config_file_content (
obj ,
& config_file_contents ,
generated_imports ,
fmt_config_options ,
2024-05-08 20:34:46 +01:00
config_file . imports_key ( ) ,
config_file . file_name ( ) ,
2024-02-29 19:12:04 +00:00
) ;
tokio ::fs ::write ( & config_file_path , new_text )
. await
. context ( " Failed to update configuration file " ) ? ;
2024-06-26 17:24:10 -04:00
// clear the previously cached package.json from memory before reloading it
2024-07-25 19:08:14 -04:00
node_resolver ::PackageJsonThreadLocalCache ::clear ( ) ;
2024-05-08 20:34:46 +01:00
// make a new CliFactory to pick up the updated config file
2024-07-23 19:00:48 -04:00
let cli_factory = CliFactory ::from_flags ( flags ) ;
2024-05-08 20:34:46 +01:00
// cache deps
2024-08-12 16:17:25 -04:00
crate ::module_loader ::load_top_level_deps ( & cli_factory ) . await ? ;
2024-02-29 19:12:04 +00:00
Ok ( ( ) )
}
struct SelectedPackage {
import_name : String ,
package_name : String ,
version_req : String ,
2024-08-08 15:25:05 +01:00
selected_version : String ,
2024-02-29 19:12:04 +00:00
}
enum PackageAndVersion {
2024-08-09 16:29:11 +02:00
NotFound {
package : String ,
found_npm_package : bool ,
package_req : PackageReq ,
} ,
2024-02-29 19:12:04 +00:00
Selected ( SelectedPackage ) ,
}
async fn find_package_and_select_version_for_req (
2024-03-01 21:34:13 +00:00
jsr_resolver : Arc < JsrFetchResolver > ,
2024-03-06 13:24:15 +00:00
npm_resolver : Arc < NpmFetchResolver > ,
2024-02-29 19:12:04 +00:00
add_package_req : AddPackageReq ,
) -> Result < PackageAndVersion , AnyError > {
2024-06-10 19:56:43 -04:00
match add_package_req . value {
AddPackageReqValue ::Jsr ( req ) = > {
2024-03-01 21:34:13 +00:00
let jsr_prefixed_name = format! ( " jsr: {} " , & req . name ) ;
2024-06-10 19:56:43 -04:00
let Some ( nv ) = jsr_resolver . req_to_nv ( & req ) . await else {
2024-08-09 16:29:11 +02:00
if npm_resolver . req_to_nv ( & req ) . await . is_some ( ) {
return Ok ( PackageAndVersion ::NotFound {
package : jsr_prefixed_name ,
found_npm_package : true ,
package_req : req ,
} ) ;
}
return Ok ( PackageAndVersion ::NotFound {
package : jsr_prefixed_name ,
found_npm_package : false ,
package_req : req ,
} ) ;
2024-03-01 21:34:13 +00:00
} ;
let range_symbol = if req . version_req . version_text ( ) . starts_with ( '~' ) {
'~'
} else {
'^'
} ;
Ok ( PackageAndVersion ::Selected ( SelectedPackage {
2024-06-10 19:56:43 -04:00
import_name : add_package_req . alias ,
2024-03-01 21:34:13 +00:00
package_name : jsr_prefixed_name ,
version_req : format ! ( " {}{} " , range_symbol , & nv . version ) ,
2024-08-08 15:25:05 +01:00
selected_version : nv . version . to_string ( ) ,
2024-03-01 21:34:13 +00:00
} ) )
2024-02-29 19:12:04 +00:00
}
2024-06-10 19:56:43 -04:00
AddPackageReqValue ::Npm ( req ) = > {
2024-03-06 13:24:15 +00:00
let npm_prefixed_name = format! ( " npm: {} " , & req . name ) ;
2024-06-10 19:56:43 -04:00
let Some ( nv ) = npm_resolver . req_to_nv ( & req ) . await else {
2024-08-09 16:29:11 +02:00
return Ok ( PackageAndVersion ::NotFound {
package : npm_prefixed_name ,
found_npm_package : false ,
package_req : req ,
} ) ;
2024-03-06 13:24:15 +00:00
} ;
let range_symbol = if req . version_req . version_text ( ) . starts_with ( '~' ) {
'~'
} else {
'^'
} ;
Ok ( PackageAndVersion ::Selected ( SelectedPackage {
2024-06-10 19:56:43 -04:00
import_name : add_package_req . alias ,
2024-03-06 13:24:15 +00:00
package_name : npm_prefixed_name ,
version_req : format ! ( " {}{} " , range_symbol , & nv . version ) ,
2024-08-08 15:25:05 +01:00
selected_version : nv . version . to_string ( ) ,
2024-03-06 13:24:15 +00:00
} ) )
2024-02-29 19:12:04 +00:00
}
}
}
2024-06-10 19:56:43 -04:00
#[ derive(Debug, PartialEq, Eq) ]
enum AddPackageReqValue {
Jsr ( PackageReq ) ,
Npm ( PackageReq ) ,
}
#[ derive(Debug, PartialEq, Eq) ]
struct AddPackageReq {
alias : String ,
value : AddPackageReqValue ,
}
impl AddPackageReq {
pub fn parse ( entry_text : & str ) -> Result < Self , AnyError > {
enum Prefix {
Jsr ,
Npm ,
}
fn parse_prefix ( text : & str ) -> ( Option < Prefix > , & str ) {
if let Some ( text ) = text . strip_prefix ( " jsr: " ) {
( Some ( Prefix ::Jsr ) , text )
} else if let Some ( text ) = text . strip_prefix ( " npm: " ) {
( Some ( Prefix ::Npm ) , text )
} else {
( None , text )
}
}
// parse the following:
// - alias@npm:<package_name>
// - other_alias@npm:<package_name>
// - @alias/other@jsr:<package_name>
fn parse_alias ( entry_text : & str ) -> Option < ( & str , & str ) > {
for prefix in [ " npm: " , " jsr: " ] {
let Some ( location ) = entry_text . find ( prefix ) else {
continue ;
} ;
let prefix = & entry_text [ .. location ] ;
if let Some ( alias ) = prefix . strip_suffix ( '@' ) {
return Some ( ( alias , & entry_text [ location .. ] ) ) ;
}
}
None
}
let ( maybe_prefix , entry_text ) = parse_prefix ( entry_text ) ;
let ( prefix , maybe_alias , entry_text ) = match maybe_prefix {
Some ( prefix ) = > ( prefix , None , entry_text ) ,
None = > match parse_alias ( entry_text ) {
Some ( ( alias , text ) ) = > {
let ( maybe_prefix , entry_text ) = parse_prefix ( text ) ;
(
maybe_prefix . unwrap_or ( Prefix ::Jsr ) ,
Some ( alias . to_string ( ) ) ,
entry_text ,
)
}
None = > ( Prefix ::Jsr , None , entry_text ) ,
} ,
} ;
match prefix {
Prefix ::Jsr = > {
let package_req = PackageReq ::from_str ( entry_text ) ? ;
Ok ( AddPackageReq {
alias : maybe_alias . unwrap_or_else ( | | package_req . name . to_string ( ) ) ,
value : AddPackageReqValue ::Jsr ( package_req ) ,
} )
}
Prefix ::Npm = > {
let package_req = PackageReq ::from_str ( entry_text ) ? ;
Ok ( AddPackageReq {
alias : maybe_alias . unwrap_or_else ( | | package_req . name . to_string ( ) ) ,
value : AddPackageReqValue ::Npm ( package_req ) ,
} )
}
}
}
2024-02-29 19:12:04 +00:00
}
fn generate_imports ( packages_to_version : Vec < ( String , String ) > ) -> String {
let mut contents = vec! [ ] ;
let len = packages_to_version . len ( ) ;
for ( index , ( package , version ) ) in packages_to_version . iter ( ) . enumerate ( ) {
// TODO(bartlomieju): fix it, once we start support specifying version on the cli
contents . push ( format! ( " \" {} \" : \" {} \" " , package , version ) ) ;
if index ! = len - 1 {
contents . push ( " , " . to_string ( ) ) ;
}
}
contents . join ( " \n " )
}
2024-08-12 16:17:25 -04:00
fn remove_from_config (
config_path : & Path ,
keys : & [ & 'static str ] ,
packages_to_remove : & [ String ] ,
removed_packages : & mut Vec < String > ,
fmt_options : & FmtOptionsConfig ,
) -> Result < ( ) , AnyError > {
let mut json : serde_json ::Value =
serde_json ::from_slice ( & std ::fs ::read ( config_path ) ? ) ? ;
for key in keys {
let Some ( obj ) = json . get_mut ( * key ) . and_then ( | v | v . as_object_mut ( ) ) else {
continue ;
} ;
for package in packages_to_remove {
if obj . shift_remove ( package ) . is_some ( ) {
removed_packages . push ( package . clone ( ) ) ;
}
}
}
let config = serde_json ::to_string_pretty ( & json ) ? ;
let config =
crate ::tools ::fmt ::format_json ( config_path , & config , fmt_options )
. ok ( )
. flatten ( )
. unwrap_or ( config ) ;
std ::fs ::write ( config_path , config )
. context ( " Failed to update configuration file " ) ? ;
Ok ( ( ) )
}
pub async fn remove (
flags : Arc < Flags > ,
remove_flags : RemoveFlags ,
) -> Result < ( ) , AnyError > {
let ( config_file , factory ) = DenoOrPackageJson ::from_flags ( flags . clone ( ) ) ? ;
let options = factory . cli_options ( ) ? ;
let start_dir = & options . start_dir ;
let fmt_config_options = config_file . fmt_options ( ) ;
let mut removed_packages = Vec ::new ( ) ;
if let Some ( deno_json ) = start_dir . maybe_deno_json ( ) {
remove_from_config (
& deno_json . specifier . to_file_path ( ) . unwrap ( ) ,
& [ " imports " ] ,
& remove_flags . packages ,
& mut removed_packages ,
& fmt_config_options ,
) ? ;
}
if let Some ( pkg_json ) = start_dir . maybe_pkg_json ( ) {
remove_from_config (
& pkg_json . path ,
& [ " dependencies " , " devDependencies " ] ,
& remove_flags . packages ,
& mut removed_packages ,
& fmt_config_options ,
) ? ;
}
if removed_packages . is_empty ( ) {
log ::info! ( " No packages were removed " ) ;
} else {
for package in & removed_packages {
log ::info! ( " Removed {} " , crate ::colors ::green ( package ) ) ;
}
// Update deno.lock
node_resolver ::PackageJsonThreadLocalCache ::clear ( ) ;
let cli_factory = CliFactory ::from_flags ( flags ) ;
crate ::module_loader ::load_top_level_deps ( & cli_factory ) . await ? ;
}
Ok ( ( ) )
}
2024-02-29 19:12:04 +00:00
fn update_config_file_content (
obj : jsonc_parser ::ast ::Object ,
config_file_contents : & str ,
generated_imports : String ,
fmt_options : FmtOptionsConfig ,
2024-05-08 20:34:46 +01:00
imports_key : & str ,
file_name : & str ,
2024-02-29 19:12:04 +00:00
) -> String {
let mut text_changes = vec! [ ] ;
2024-05-08 20:34:46 +01:00
match obj . get ( imports_key ) {
2024-02-29 19:12:04 +00:00
Some ( ObjectProp {
value : Value ::Object ( lit ) ,
..
} ) = > text_changes . push ( TextChange {
range : ( lit . range . start + 1 ) .. ( lit . range . end - 1 ) ,
new_text : generated_imports ,
} ) ,
None = > {
let insert_position = obj . range . end - 1 ;
text_changes . push ( TextChange {
range : insert_position .. insert_position ,
2024-03-26 21:40:24 +00:00
// NOTE(bartlomieju): adding `\n` here to force the formatter to always
2024-08-02 13:26:54 +02:00
// produce a config file that is multiline, like so:
2024-03-26 21:40:24 +00:00
// ```
// {
// "imports": {
// "<package_name>": "<registry>:<package_name>@<semver>"
// }
// }
2024-05-08 20:34:46 +01:00
new_text : format ! ( " \" {imports_key} \" : {{ \n {generated_imports} }} " ) ,
2024-02-29 19:12:04 +00:00
} )
}
2024-05-08 20:34:46 +01:00
// we verified the shape of `imports`/`dependencies` above
2024-02-29 19:12:04 +00:00
Some ( _ ) = > unreachable! ( ) ,
}
let new_text =
deno_ast ::apply_text_changes ( config_file_contents , text_changes ) ;
crate ::tools ::fmt ::format_json (
2024-05-08 20:34:46 +01:00
& PathBuf ::from ( file_name ) ,
2024-02-29 19:12:04 +00:00
& new_text ,
& fmt_options ,
)
. ok ( )
. map ( | formatted_text | formatted_text . unwrap_or_else ( | | new_text . clone ( ) ) )
. unwrap_or ( new_text )
}
2024-06-10 19:56:43 -04:00
#[ cfg(test) ]
mod test {
use deno_semver ::VersionReq ;
use super ::* ;
#[ test ]
fn test_parse_add_package_req ( ) {
assert_eq! (
AddPackageReq ::parse ( " jsr:foo " ) . unwrap ( ) ,
AddPackageReq {
alias : " foo " . to_string ( ) ,
value : AddPackageReqValue ::Jsr ( PackageReq ::from_str ( " foo " ) . unwrap ( ) )
}
) ;
assert_eq! (
AddPackageReq ::parse ( " alias@jsr:foo " ) . unwrap ( ) ,
AddPackageReq {
alias : " alias " . to_string ( ) ,
value : AddPackageReqValue ::Jsr ( PackageReq ::from_str ( " foo " ) . unwrap ( ) )
}
) ;
assert_eq! (
AddPackageReq ::parse ( " @alias/pkg@npm:foo " ) . unwrap ( ) ,
AddPackageReq {
alias : " @alias/pkg " . to_string ( ) ,
value : AddPackageReqValue ::Npm ( PackageReq ::from_str ( " foo " ) . unwrap ( ) )
}
) ;
assert_eq! (
AddPackageReq ::parse ( " @alias/pkg@jsr:foo " ) . unwrap ( ) ,
AddPackageReq {
alias : " @alias/pkg " . to_string ( ) ,
value : AddPackageReqValue ::Jsr ( PackageReq ::from_str ( " foo " ) . unwrap ( ) )
}
) ;
assert_eq! (
AddPackageReq ::parse ( " alias@jsr:foo@^1.5.0 " ) . unwrap ( ) ,
AddPackageReq {
alias : " alias " . to_string ( ) ,
value : AddPackageReqValue ::Jsr (
PackageReq ::from_str ( " foo@^1.5.0 " ) . unwrap ( )
)
}
) ;
assert_eq! (
AddPackageReq ::parse ( " @scope/pkg@tag " ) . unwrap ( ) ,
AddPackageReq {
alias : " @scope/pkg " . to_string ( ) ,
value : AddPackageReqValue ::Jsr ( PackageReq {
name : " @scope/pkg " . to_string ( ) ,
// this is a tag
version_req : VersionReq ::parse_from_specifier ( " tag " ) . unwrap ( ) ,
} ) ,
}
) ;
}
}