2024-01-01 14:58:21 -05:00
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
2022-08-20 01:37:05 +02:00
2024-11-21 00:03:11 +00:00
use crate ::args ::DenoSubcommand ;
use crate ::args ::Flags ;
2022-08-20 01:37:05 +02:00
use crate ::args ::InitFlags ;
2024-11-21 00:03:11 +00:00
use crate ::args ::PackagesAllowedScripts ;
use crate ::args ::PermissionFlags ;
use crate ::args ::RunFlags ;
2022-12-08 06:34:28 +01:00
use crate ::colors ;
2024-11-21 00:03:11 +00:00
use color_print ::cformat ;
use color_print ::cstr ;
2024-11-25 14:42:34 +00:00
use deno_config ::deno_json ::NodeModulesDirMode ;
2022-08-29 20:19:54 +02:00
use deno_core ::anyhow ::Context ;
use deno_core ::error ::AnyError ;
2024-07-10 06:43:34 +05:30
use deno_core ::serde_json ::json ;
2024-11-21 00:03:11 +00:00
use deno_runtime ::WorkerExecutionMode ;
2022-09-02 17:59:36 +02:00
use log ::info ;
2024-11-21 00:03:11 +00:00
use std ::io ::IsTerminal ;
2022-08-20 01:37:05 +02:00
use std ::io ::Write ;
use std ::path ::Path ;
2024-11-21 00:03:11 +00:00
pub async fn init_project ( init_flags : InitFlags ) -> Result < i32 , AnyError > {
if let Some ( package ) = & init_flags . package {
return init_npm ( package , init_flags . package_args ) . await ;
}
2022-08-20 01:37:05 +02:00
let cwd =
std ::env ::current_dir ( ) . context ( " Can't read current working directory. " ) ? ;
let dir = if let Some ( dir ) = & init_flags . dir {
let dir = cwd . join ( dir ) ;
std ::fs ::create_dir_all ( & dir ) ? ;
dir
} else {
cwd
} ;
2024-08-08 17:54:39 +01:00
if init_flags . serve {
create_file (
& dir ,
" main.ts " ,
2024-10-29 14:37:21 +09:00
r #" import { serveDir } from " @ std / http " ;
2024-08-08 17:54:39 +01:00
2024-10-29 14:37:21 +09:00
const userPagePattern = new URLPattern ( { pathname : " /users/:id " } ) ;
const staticPathPattern = new URLPattern ( { pathname : " /static/* " } ) ;
2024-08-08 17:54:39 +01:00
export default {
fetch ( req ) {
2024-10-29 14:37:21 +09:00
const url = new URL ( req . url ) ;
if ( url . pathname = = = " / " ) {
return new Response ( " Home page " ) ;
}
const userPageMatch = userPagePattern . exec ( url ) ;
if ( userPageMatch ) {
return new Response ( userPageMatch . pathname . groups . id ) ;
}
if ( staticPathPattern . test ( url ) ) {
return serveDir ( req ) ;
}
return new Response ( " Not found " , { status : 404 } ) ;
2024-08-08 17:54:39 +01:00
} ,
} satisfies Deno . ServeDefaultExport ;
" #,
) ? ;
create_file (
& dir ,
" main_test.ts " ,
r #" import { assertEquals } from " @ std / assert " ;
import server from " ./main.ts " ;
Deno . test ( async function serverFetch ( ) {
const req = new Request ( " https://deno.land " ) ;
const res = await server . fetch ( req ) ;
assertEquals ( await res . text ( ) , " Home page " ) ;
} ) ;
Deno . test ( async function serverFetchNotFound ( ) {
const req = new Request ( " https://deno.land/404 " ) ;
const res = await server . fetch ( req ) ;
assertEquals ( res . status , 404 ) ;
} ) ;
Deno . test ( async function serverFetchUsers ( ) {
const req = new Request ( " https://deno.land/users/123 " ) ;
const res = await server . fetch ( req ) ;
assertEquals ( await res . text ( ) , " 123 " ) ;
} ) ;
Deno . test ( async function serverFetchStatic ( ) {
2024-09-02 19:01:36 -04:00
const req = new Request ( " https://deno.land/static/hello.js " ) ;
2024-08-08 17:54:39 +01:00
const res = await server . fetch ( req ) ;
2024-09-02 19:01:36 -04:00
assertEquals ( await res . text ( ) , ' console . log ( " Hello, world! " ) ; \ n ' ) ;
assertEquals ( res . headers . get ( " content-type " ) , " text/javascript; charset=UTF-8 " ) ;
2024-08-08 17:54:39 +01:00
} ) ;
" #,
) ? ;
2024-09-02 19:01:36 -04:00
let static_dir = dir . join ( " static " ) ;
std ::fs ::create_dir_all ( & static_dir ) ? ;
create_file (
& static_dir ,
" hello.js " ,
r #" console.log( " Hello , world ! " );
" #,
) ? ;
2024-08-08 17:54:39 +01:00
create_json_file (
& dir ,
" deno.json " ,
& json! ( {
" tasks " : {
" dev " : " deno serve --watch -R main.ts " ,
} ,
" imports " : {
" @std/assert " : " jsr:@std/assert@1 " ,
" @std/http " : " jsr:@std/http@1 " ,
}
} ) ,
) ? ;
} else if init_flags . lib {
2024-07-10 06:43:34 +05:30
// Extract the directory name to use as the project name
let project_name = dir
. file_name ( )
. unwrap_or_else ( | | dir . as_os_str ( ) )
. to_str ( )
. unwrap ( ) ;
create_file (
& dir ,
" mod.ts " ,
r #" export function add(a: number, b: number): number {
return a + b ;
}
" #,
) ? ;
create_file (
& dir ,
" mod_test.ts " ,
2024-07-14 13:22:43 -04:00
r #" import { assertEquals } from " @ std / assert " ;
2024-07-10 06:43:34 +05:30
import { add } from " ./mod.ts " ;
Deno . test ( function addTest ( ) {
assertEquals ( add ( 2 , 3 ) , 5 ) ;
} ) ;
" #,
) ? ;
create_json_file (
& dir ,
" deno.json " ,
& json! ( {
" name " : project_name ,
2024-07-14 13:22:43 -04:00
" version " : " 0.1.0 " ,
2024-08-08 16:28:18 +02:00
" exports " : " ./mod.ts " ,
2024-07-10 06:43:34 +05:30
" tasks " : {
" dev " : " deno test --watch mod.ts "
2024-07-14 13:22:43 -04:00
} ,
2024-08-16 09:12:52 -04:00
" license " : " MIT " ,
2024-07-14 13:22:43 -04:00
" imports " : {
" @std/assert " : " jsr:@std/assert@1 "
} ,
2024-07-10 06:43:34 +05:30
} ) ,
) ? ;
} else {
create_file (
& dir ,
" main.ts " ,
r #" export function add(a: number, b: number): number {
return a + b ;
}
2024-07-12 22:04:27 +03:00
// Learn more at https://docs.deno.com/runtime/manual/examples/module_metadata#concepts
2024-07-10 06:43:34 +05:30
if ( import . meta . main ) {
console . log ( " Add 2 + 3 = " , add ( 2 , 3 ) ) ;
}
" #,
) ? ;
create_file (
& dir ,
" main_test.ts " ,
2024-07-14 13:22:43 -04:00
r #" import { assertEquals } from " @ std / assert " ;
2024-07-10 06:43:34 +05:30
import { add } from " ./main.ts " ;
2022-08-20 01:37:05 +02:00
2024-07-10 06:43:34 +05:30
Deno . test ( function addTest ( ) {
assertEquals ( add ( 2 , 3 ) , 5 ) ;
} ) ;
" #,
) ? ;
create_json_file (
& dir ,
" deno.json " ,
& json! ( {
" tasks " : {
" dev " : " deno run --watch main.ts "
2024-07-14 13:22:43 -04:00
} ,
" imports " : {
" @std/assert " : " jsr:@std/assert@1 "
2024-07-10 06:43:34 +05:30
}
} ) ,
) ? ;
}
2022-12-08 06:34:28 +01:00
info! ( " ✅ {} " , colors ::green ( " Project initialized " ) ) ;
info! ( " " ) ;
info! ( " {} " , colors ::gray ( " Run these commands to get started " ) ) ;
info! ( " " ) ;
2022-08-20 01:37:05 +02:00
if let Some ( dir ) = init_flags . dir {
2022-09-02 17:59:36 +02:00
info! ( " cd {} " , dir ) ;
2022-12-08 06:34:28 +01:00
info! ( " " ) ;
2022-08-20 01:37:05 +02:00
}
2024-08-08 17:54:39 +01:00
if init_flags . serve {
info! ( " {} " , colors ::gray ( " # Run the server " ) ) ;
info! ( " deno serve -R main.ts " ) ;
info! ( " " ) ;
info! (
" {} " ,
colors ::gray ( " # Run the server and watch for file changes " )
) ;
info! ( " deno task dev " ) ;
info! ( " " ) ;
info! ( " {} " , colors ::gray ( " # Run the tests " ) ) ;
2024-09-02 19:01:36 -04:00
info! ( " deno test -R " ) ;
2024-08-08 17:54:39 +01:00
} else if init_flags . lib {
2024-07-10 06:43:34 +05:30
info! ( " {} " , colors ::gray ( " # Run the tests " ) ) ;
info! ( " deno test " ) ;
info! ( " " ) ;
info! (
" {} " ,
colors ::gray ( " # Run the tests and watch for file changes " )
) ;
info! ( " deno task dev " ) ;
info! ( " " ) ;
info! ( " {} " , colors ::gray ( " # Publish to JSR (dry run) " ) ) ;
info! ( " deno publish --dry-run " ) ;
} else {
info! ( " {} " , colors ::gray ( " # Run the program " ) ) ;
info! ( " deno run main.ts " ) ;
info! ( " " ) ;
info! (
" {} " ,
colors ::gray ( " # Run the program and watch for file changes " )
) ;
info! ( " deno task dev " ) ;
info! ( " " ) ;
info! ( " {} " , colors ::gray ( " # Run the tests " ) ) ;
info! ( " deno test " ) ;
}
2024-11-21 00:03:11 +00:00
Ok ( 0 )
}
2024-11-28 15:07:32 +01:00
fn npm_name_to_create_package ( name : & str ) -> String {
let mut s = " npm: " . to_string ( ) ;
let mut scoped = false ;
let mut create = false ;
for ( i , ch ) in name . char_indices ( ) {
if i = = 0 {
if ch = = '@' {
scoped = true ;
} else {
create = true ;
s . push_str ( " create- " ) ;
}
} else if scoped {
if ch = = '/' {
scoped = false ;
create = true ;
s . push_str ( " /create- " ) ;
continue ;
} else if ch = = '@' & & ! create {
scoped = false ;
create = true ;
s . push_str ( " /create@ " ) ;
continue ;
}
}
s . push ( ch ) ;
}
if ! create {
s . push_str ( " /create " ) ;
}
s
}
2024-11-21 00:03:11 +00:00
async fn init_npm ( name : & str , args : Vec < String > ) -> Result < i32 , AnyError > {
2024-11-28 15:07:32 +01:00
let script_name = npm_name_to_create_package ( name ) ;
2024-11-21 00:03:11 +00:00
fn print_manual_usage ( script_name : & str , args : & [ String ] ) -> i32 {
log ::info! ( " {} " , cformat! ( " You can initialize project manually by running <u>deno run {} {}</> and applying desired permissions. " , script_name , args . join ( " " ) ) ) ;
1
}
if std ::io ::stdin ( ) . is_terminal ( ) {
log ::info! (
cstr! ( " ⚠️ Do you fully trust <y>{}</> package? Deno will invoke code from it with all permissions. Do you want to continue? <p(245)>[y/n]</> " ) ,
script_name
) ;
loop {
let _ = std ::io ::stdout ( ) . write ( b " > " ) ? ;
std ::io ::stdout ( ) . flush ( ) ? ;
let mut answer = String ::new ( ) ;
if std ::io ::stdin ( ) . read_line ( & mut answer ) . is_ok ( ) {
let answer = answer . trim ( ) . to_ascii_lowercase ( ) ;
if answer ! = " y " {
return Ok ( print_manual_usage ( & script_name , & args ) ) ;
} else {
break ;
}
}
}
} else {
return Ok ( print_manual_usage ( & script_name , & args ) ) ;
}
let new_flags = Flags {
permissions : PermissionFlags {
allow_all : true ,
.. Default ::default ( )
} ,
allow_scripts : PackagesAllowedScripts ::All ,
argv : args ,
2024-11-25 14:42:34 +00:00
node_modules_dir : Some ( NodeModulesDirMode ::Auto ) ,
2024-11-21 00:03:11 +00:00
subcommand : DenoSubcommand ::Run ( RunFlags {
script : script_name ,
.. Default ::default ( )
} ) ,
.. Default ::default ( )
} ;
crate ::tools ::run ::run_script (
WorkerExecutionMode ::Run ,
new_flags . into ( ) ,
None ,
)
. await
2022-08-20 01:37:05 +02:00
}
2024-07-10 06:43:34 +05:30
fn create_json_file (
dir : & Path ,
filename : & str ,
value : & deno_core ::serde_json ::Value ,
) -> Result < ( ) , AnyError > {
let mut text = deno_core ::serde_json ::to_string_pretty ( value ) ? ;
text . push ( '\n' ) ;
create_file ( dir , filename , & text )
}
fn create_file (
dir : & Path ,
filename : & str ,
content : & str ,
) -> Result < ( ) , AnyError > {
let path = dir . join ( filename ) ;
if path . exists ( ) {
info! (
" ℹ ️ {}" ,
colors ::gray ( format! ( " Skipped creating {filename} as it already exists " ) )
) ;
Ok ( ( ) )
} else {
let mut file = std ::fs ::OpenOptions ::new ( )
. write ( true )
. create_new ( true )
. open ( path )
. with_context ( | | format! ( " Failed to create {filename} file " ) ) ? ;
file . write_all ( content . as_bytes ( ) ) ? ;
Ok ( ( ) )
}
}
2024-11-28 15:07:32 +01:00
#[ cfg(test) ]
mod test {
use crate ::tools ::init ::npm_name_to_create_package ;
#[ test ]
fn npm_name_to_create_package_test ( ) {
// See https://docs.npmjs.com/cli/v8/commands/npm-init#description
assert_eq! (
npm_name_to_create_package ( " foo " ) ,
" npm:create-foo " . to_string ( )
) ;
assert_eq! (
npm_name_to_create_package ( " foo@1.0.0 " ) ,
" npm:create-foo@1.0.0 " . to_string ( )
) ;
assert_eq! (
npm_name_to_create_package ( " @foo " ) ,
" npm:@foo/create " . to_string ( )
) ;
assert_eq! (
npm_name_to_create_package ( " @foo@1.0.0 " ) ,
" npm:@foo/create@1.0.0 " . to_string ( )
) ;
assert_eq! (
npm_name_to_create_package ( " @foo/bar " ) ,
" npm:@foo/create-bar " . to_string ( )
) ;
assert_eq! (
npm_name_to_create_package ( " @foo/bar@1.0.0 " ) ,
" npm:@foo/create-bar@1.0.0 " . to_string ( )
) ;
}
}