2023-08-10 22:37:53 -04:00
|
|
|
import * as Env from "https://deno.land/std@0.194.0/dotenv/mod.ts";
|
|
|
|
import * as Arg from "https://deno.land/std@0.194.0/flags/mod.ts";
|
2023-08-01 10:19:39 -04:00
|
|
|
import { parse as JSONC } from "https://deno.land/x/jsonct@v0.1.0/mod.ts";
|
2023-08-01 17:23:19 -04:00
|
|
|
const RootFile = new URL(`file://${Deno.cwd().replaceAll("\\", "/")}`).toString();
|
|
|
|
const RootHost = import.meta.resolve("./");
|
2023-08-10 22:37:53 -04:00
|
|
|
let arg = await Arg.parse(Deno.args);
|
|
|
|
let env = await Env.load();
|
|
|
|
const collect =async(inKey:string, inArg:Record<string, string>, inEnv:Record<string, string>):Promise<string|undefined>=>
|
|
|
|
{
|
|
|
|
const scanArg = inArg[inKey];
|
|
|
|
const scanEnvFile = inEnv[inKey];
|
|
|
|
const scanEnvDeno = Deno.env.get(inKey);
|
|
|
|
|
|
|
|
if(scanArg)
|
|
|
|
{
|
|
|
|
console.log(`Using "${inKey}" from passed argument.`);
|
|
|
|
return scanArg;
|
|
|
|
}
|
|
|
|
if(scanEnvFile)
|
|
|
|
{
|
|
|
|
console.log(`Using "${inKey}" from .env file.`);
|
|
|
|
return scanEnvFile;
|
|
|
|
}
|
|
|
|
if(scanEnvDeno)
|
|
|
|
{
|
|
|
|
console.log(`Using "${inKey}" from environment variable.`);
|
|
|
|
return scanEnvDeno;
|
|
|
|
}
|
|
|
|
|
|
|
|
const scanUser = prompt(`No "${inKey}" found. Enter one here:`);
|
|
|
|
if(!scanUser || scanUser?.length < 3)
|
|
|
|
{
|
|
|
|
console.log("Exiting...");
|
|
|
|
Deno.exit();
|
|
|
|
}
|
|
|
|
return scanUser;
|
|
|
|
};
|
|
|
|
|
2023-08-01 17:23:19 -04:00
|
|
|
|
2023-08-01 22:42:23 -04:00
|
|
|
type ConfigCheck = {path?:string, text?:string, json?:Record<string, string|Record<string, string|string[]>>};
|
|
|
|
type ConfigCheckPair = [config:ConfigCheck, imports:ConfigCheck];
|
|
|
|
|
2023-08-01 10:19:39 -04:00
|
|
|
export async function HuntConfig()
|
|
|
|
{
|
|
|
|
let path:string, resp:Response, text="", json;
|
|
|
|
try
|
|
|
|
{
|
2023-08-01 15:30:31 -04:00
|
|
|
path = "deno.json"
|
2023-08-01 17:23:19 -04:00
|
|
|
resp = await fetch(RootFile + "/" + path);
|
2023-08-01 10:19:39 -04:00
|
|
|
text = await resp.text();
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2023-08-01 15:30:31 -04:00
|
|
|
path = "deno.jsonc";
|
2023-08-01 17:23:19 -04:00
|
|
|
resp = await fetch(RootFile + "/" + path);
|
2023-08-01 10:19:39 -04:00
|
|
|
text = await resp.text();
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2023-08-01 17:23:19 -04:00
|
|
|
path = RootFile+"/.vscode/settings.json"
|
2023-08-01 10:19:39 -04:00
|
|
|
resp = await fetch(path);
|
|
|
|
json = await resp.json();
|
|
|
|
path = json["deno.config"];
|
2023-08-01 15:30:31 -04:00
|
|
|
json = undefined;
|
2023-08-01 10:19:39 -04:00
|
|
|
if(path)
|
|
|
|
{
|
2023-08-01 22:42:23 -04:00
|
|
|
path = RootFile + "/" + path
|
|
|
|
resp = await fetch(path);
|
2023-08-01 10:19:39 -04:00
|
|
|
text = await resp.text();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
path = "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(text)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
json = JSONC(text);
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
2023-08-01 22:42:23 -04:00
|
|
|
// malformed config
|
2023-08-01 10:19:39 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-01 22:42:23 -04:00
|
|
|
let imports:ConfigCheck = {};
|
|
|
|
if(json && json.imports)
|
|
|
|
{
|
|
|
|
imports.json = json;
|
|
|
|
imports.text = JSON.stringify(json.imports);
|
|
|
|
imports.path = path;
|
|
|
|
}
|
|
|
|
else if(json && !json.imports && json.importMap)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
imports.path = RootFile + "/" + json.importMap;
|
|
|
|
resp = await fetch(imports.path);
|
|
|
|
imports.text = await resp.text();
|
|
|
|
imports.json = JSONC(text);
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
// malformed import map
|
|
|
|
}
|
|
|
|
}
|
2023-08-01 17:23:19 -04:00
|
|
|
|
2023-08-01 22:42:23 -04:00
|
|
|
return [{path, text, json}, imports] as ConfigCheckPair
|
2023-08-01 17:23:19 -04:00
|
|
|
}
|
|
|
|
|
2023-08-01 10:19:39 -04:00
|
|
|
export async function SubProcess(args:string[])
|
|
|
|
{
|
|
|
|
const command = new Deno.Command(
|
|
|
|
`deno`,
|
|
|
|
{
|
|
|
|
args,
|
|
|
|
stdin: "piped",
|
|
|
|
stdout: "piped"
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const child = command.spawn();
|
|
|
|
|
|
|
|
// open a file and pipe the subprocess output to it.
|
|
|
|
const writableStream = new WritableStream({
|
|
|
|
write(chunk: Uint8Array): Promise<void> {
|
|
|
|
Deno.stdout.write(chunk);
|
|
|
|
return Promise.resolve();
|
|
|
|
},
|
|
|
|
});
|
|
|
|
child.stdout.pipeTo(writableStream);
|
|
|
|
|
|
|
|
// manually close stdin
|
|
|
|
child.stdin.close();
|
|
|
|
const status = await child.status;
|
|
|
|
}
|
|
|
|
|
2023-08-01 17:23:19 -04:00
|
|
|
export async function Install(file:string, handler:(content:string)=>string = (s)=>s)
|
|
|
|
{
|
|
|
|
const pathFile = RootHost + "install__/" + file;
|
|
|
|
|
|
|
|
try{
|
|
|
|
const check = await Deno.readTextFile(Deno.cwd()+"/"+file);
|
2023-08-02 06:07:34 -04:00
|
|
|
const replace = confirm(`⚠️🚧 The file "${file}" already exists. Replace it?`);
|
|
|
|
if(replace)
|
2023-08-01 17:23:19 -04:00
|
|
|
{
|
|
|
|
throw("")
|
|
|
|
}
|
2023-08-02 06:07:34 -04:00
|
|
|
console.log(`Using pre-existing "${file}" for now.`);
|
2023-08-01 17:23:19 -04:00
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
const resp = await fetch(pathFile);
|
|
|
|
const text = await resp.text();
|
|
|
|
await Deno.writeTextFile(Deno.cwd()+"/"+file, handler(text));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
|
|
|
|
|
|
|
|
export async function Check()
|
2023-08-01 15:30:31 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
console.info(`👷 Checking your project`)
|
|
|
|
try
|
2023-08-01 15:30:31 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
let [config, imports] = await HuntConfig();
|
|
|
|
if(!config.path)
|
2023-08-01 15:30:31 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
//const resp1 = await Prompt(" ! No Deno configuration found. Create one? [y/n]");
|
|
|
|
const resp1 = confirm("🚨🚧 No Deno configuration found. Create one?");
|
|
|
|
if(resp1)
|
2023-08-01 17:23:19 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
const resp2 = confirm("⚠️🚧 Do you also want to add starter files?");
|
|
|
|
let replaceApp = "./path/to/app.tsx";
|
|
|
|
let replaceApi = "./path/to/api.tsx";
|
|
|
|
let replaceCommentApp = "// (required) module with default export ()=>React.JSX.Element";
|
|
|
|
let replaceCommentApi = "// (optional) module with default export (req:Request, url:URL)=>Promise<Response|false>";
|
|
|
|
if(resp2)
|
|
|
|
{
|
|
|
|
replaceApp = "./app.tsx";
|
|
|
|
replaceApi = "./api.tsx";
|
|
|
|
replaceCommentApp = "";
|
|
|
|
replaceCommentApi = "";
|
2023-08-01 15:30:31 -04:00
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
await Install("app.tsx");
|
|
|
|
await Install("api.tsx");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// config initialized with no app or api
|
|
|
|
}
|
|
|
|
|
|
|
|
await Install("deno.jsonc", (s)=>s
|
|
|
|
.replace("{{server}}", RootHost)
|
|
|
|
.replace("{{app}}", replaceApp)
|
|
|
|
.replace("{{api}}", replaceApi)
|
|
|
|
.replace("{{commentApp}}", replaceCommentApp)
|
|
|
|
.replace("{{commentApi}}", replaceCommentApi)
|
|
|
|
);
|
|
|
|
|
|
|
|
[config, imports] = await HuntConfig();
|
2023-08-01 17:23:19 -04:00
|
|
|
}
|
2023-08-01 22:42:23 -04:00
|
|
|
else
|
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
throw("⛔ Config is required.");
|
2023-08-01 22:42:23 -04:00
|
|
|
}
|
2023-08-01 15:30:31 -04:00
|
|
|
|
|
|
|
}
|
2023-08-01 22:42:23 -04:00
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
/*
|
|
|
|
const inputString = `Some text 'imports' more text { some content }`;
|
|
|
|
const regex = /(?<=(['"`])imports\1[^{}]*{)/;
|
2023-08-01 22:42:23 -04:00
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
const match = inputString.search(regex);
|
2023-08-01 22:42:23 -04:00
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
if (match !== -1) {
|
|
|
|
console.log("Index of '{':", match);
|
|
|
|
} else {
|
|
|
|
console.log("'{': Not found.");
|
|
|
|
}
|
|
|
|
*/
|
2023-08-01 22:42:23 -04:00
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
if(!config.json)
|
|
|
|
{
|
|
|
|
throw("⛔ Config is malformed.");
|
|
|
|
}
|
|
|
|
else if(!imports.json?.imports)
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
const resp = confirm(`🚨🔧 Configuration has no import map. Fix it now?`);
|
2023-08-02 06:07:34 -04:00
|
|
|
if(resp)
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
|
2023-08-01 22:42:23 -04:00
|
|
|
}
|
|
|
|
}
|
2023-08-10 22:37:53 -04:00
|
|
|
|
|
|
|
|
|
|
|
if(config.json && imports.json?.imports)
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
const importMap = imports.json.imports as Record<string, string>;
|
|
|
|
let changes = ``;
|
|
|
|
if(!importMap["react"])
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
const resp = confirm(`🚨🔧 Import map has no specifier for React ("react"). Fix it now? (Will use Preact compat)`);
|
|
|
|
if(resp)
|
|
|
|
{
|
|
|
|
importMap["react"] = "https://esm.sh/preact@10.16.0/compat";
|
|
|
|
changes += `"react": "https://esm.sh/preact@10.16.0/compat",\n`;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
throw(`⛔ A React import ("react") is required.`);
|
|
|
|
}
|
2023-08-01 22:42:23 -04:00
|
|
|
}
|
2023-08-10 22:37:53 -04:00
|
|
|
if(!importMap[">able/"])
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
const resp = confirm(`🚨🔧 Import map has no specifier for Able (">able/"). Fix it now?`);
|
|
|
|
if(resp)
|
|
|
|
{
|
|
|
|
importMap[">able/"] = RootHost;
|
|
|
|
changes += `">able": "${RootHost}",\n`;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
throw(`⛔ The Able import (">able/") is required.`);
|
|
|
|
}
|
2023-08-01 22:42:23 -04:00
|
|
|
}
|
2023-08-10 22:37:53 -04:00
|
|
|
|
|
|
|
const compOpts = imports.json.compilerOptions as Record<string, string>;
|
|
|
|
if(compOpts)
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
const compJSX = compOpts["jsx"];
|
|
|
|
const compJSXImportSource = compOpts["jsxImportSource"]
|
|
|
|
if(compJSX || compJSXImportSource)
|
2023-08-01 22:42:23 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
if(!importMap["react/"])
|
|
|
|
{
|
|
|
|
//const resp = await Prompt(` ! Import map has no specifier for React ("react"). Add it now? [y/n]`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-01 22:42:23 -04:00
|
|
|
|
|
|
|
|
2023-08-10 22:37:53 -04:00
|
|
|
}
|
2023-08-01 15:30:31 -04:00
|
|
|
}
|
2023-08-10 22:37:53 -04:00
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
console.log(e, "\n (Able Exiting...)");
|
|
|
|
Deno.exit();
|
|
|
|
}
|
|
|
|
console.log(`🚗 Good to go!`);
|
|
|
|
|
2023-08-01 17:23:19 -04:00
|
|
|
}
|
2023-08-10 22:37:53 -04:00
|
|
|
|
|
|
|
if(arg._.length)
|
2023-08-01 17:23:19 -04:00
|
|
|
{
|
2023-08-10 22:37:53 -04:00
|
|
|
|
|
|
|
const [config, imports] = await HuntConfig();
|
|
|
|
|
|
|
|
switch(arg._[0])
|
|
|
|
{
|
|
|
|
case "work" :
|
|
|
|
{
|
|
|
|
await SubProcess(["run", `--config=${config.path}`, RootHost+"run.tsx", "--dev", ...Deno.args]);
|
|
|
|
}
|
|
|
|
case "host" :
|
|
|
|
{
|
|
|
|
await SubProcess(["run", `--config=${config.path}`, RootHost+"run.tsx", ...Deno.args]);
|
|
|
|
}
|
|
|
|
case "push" :
|
|
|
|
{
|
|
|
|
let useToken = await collect("DENO_DEPLOY_TOKEN", arg, env);
|
|
|
|
let useProject = await collect("DENO_DEPLOY_PROJECT", arg, env);
|
|
|
|
|
|
|
|
let scanProd:string[]|string|null = prompt(`Do you want to deploy to *production*?`);
|
|
|
|
if(scanProd)
|
|
|
|
{
|
|
|
|
scanProd = prompt(`Are you sure? This will update the live project at "${useProject}"`);
|
|
|
|
scanProd = scanProd ? ["--prod"] : [];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
scanProd = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
await SubProcess([
|
|
|
|
"run",
|
|
|
|
"-A",
|
|
|
|
"--no-lock",
|
|
|
|
`--config=${config.path}`,
|
|
|
|
"https://deno.land/x/deploy/deployctl.ts",
|
|
|
|
"deploy",
|
|
|
|
`--project=${useProject}`,
|
|
|
|
`--token=${useToken}`,
|
|
|
|
`--import-map=${imports.path}`,
|
|
|
|
RootHost+"run.tsx",
|
|
|
|
...scanProd,
|
|
|
|
...Deno.args]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|