kschema-fs-api-gen-get-actions 1.3.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +323 -0
- package/bin/cli.js +12 -0
- package/bin/core/getLatestVersion.js +13 -0
- package/bin/core/loadRunner.js +9 -0
- package/bin/v2/core/createFolder.js +34 -0
- package/bin/v2/core/parseInput.js +9 -0
- package/bin/v2/core/resolveCommand.js +11 -0
- package/bin/v2/core/showUsage.js +49 -0
- package/bin/v2/start.js +22 -0
- package/bin/v2/tasks/actions/ShowAll/actions.json +26 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/common/readFile.js +8 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/common/writeFile.js +10 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/buildImport.js +9 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/checkDuplicate.js +8 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/findInsertIndex.js +10 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/index.js +47 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/buildUseLine.js +5 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/checkDuplicate.js +9 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/findInsertIndex.js +10 -0
- package/bin/v2/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/index.js +56 -0
- package/bin/v2/tasks/actions/ShowAll/steps/announce.js +9 -0
- package/bin/v2/tasks/actions/ShowAll/steps/createFolder.js +7 -0
- package/bin/v2/tasks/actions/ShowAll/steps/decideTemplate.js +3 -0
- package/bin/v2/tasks/actions/ShowAll/steps/locateDestination.js +7 -0
- package/bin/v2/tasks/actions/ShowAll/steps/locateSource.js +18 -0
- package/bin/v2/tasks/actions/ShowAll/steps/resolveFolderName.js +20 -0
- package/bin/v2/tasks/actions/ShowAll/steps/updateAppJs.js +7 -0
- package/bin/v2/tasks/actions/ShowAll/steps/updateEndPointsFile.js +30 -0
- package/bin/v2/tasks/actions/ShowAll/template/v2/controller.js +24 -0
- package/bin/v2/tasks/actions/ShowAll/template/v2/errors.js +11 -0
- package/bin/v2/tasks/actions/ShowAll/template/v2/getData.js +11 -0
- package/bin/v2/tasks/actions/ShowAll/template/v2/service.js +9 -0
- package/bin/v2/tasks/actions/showAll.js +63 -0
- package/bin/v2/tasks/core/createFolder.js +34 -0
- package/bin/v2/tasks/core/parseInput.js +12 -0
- package/bin/v2/tasks/core/resolveCommand.js +24 -0
- package/bin/v2/tasks/core/showUsage.js +43 -0
- package/bin/v3/core/createFolder.js +34 -0
- package/bin/v3/core/parseInput.js +9 -0
- package/bin/v3/core/resolveCommand.js +11 -0
- package/bin/v3/core/showUsage.js +49 -0
- package/bin/v3/start.js +22 -0
- package/bin/v3/tasks/actions/ShowAll/actions.json +26 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/common/readFile.js +8 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/common/writeFile.js +10 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/buildImport.js +9 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/checkDuplicate.js +8 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/findInsertIndex.js +10 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateImports/index.js +47 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/buildUseLine.js +5 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/checkDuplicate.js +9 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/findInsertIndex.js +10 -0
- package/bin/v3/tasks/actions/ShowAll/steps/UpdateRoutesJs/updateUse/index.js +56 -0
- package/bin/v3/tasks/actions/ShowAll/steps/announce.js +9 -0
- package/bin/v3/tasks/actions/ShowAll/steps/createFolder.js +7 -0
- package/bin/v3/tasks/actions/ShowAll/steps/decideTemplate.js +3 -0
- package/bin/v3/tasks/actions/ShowAll/steps/locateDestination.js +7 -0
- package/bin/v3/tasks/actions/ShowAll/steps/locateSource.js +18 -0
- package/bin/v3/tasks/actions/ShowAll/steps/resolveFolderName.js +20 -0
- package/bin/v3/tasks/actions/ShowAll/steps/updateAppJs.js +7 -0
- package/bin/v3/tasks/actions/ShowAll/steps/updateEndPointsFile.js +30 -0
- package/bin/v3/tasks/actions/ShowAll/template/v2/controller.js +24 -0
- package/bin/v3/tasks/actions/ShowAll/template/v2/errors.js +11 -0
- package/bin/v3/tasks/actions/ShowAll/template/v2/getData.js +11 -0
- package/bin/v3/tasks/actions/ShowAll/template/v2/service.js +9 -0
- package/bin/v3/tasks/actions/showAll.js +63 -0
- package/bin/v3/tasks/core/createFolder.js +34 -0
- package/bin/v3/tasks/core/parseInput.js +12 -0
- package/bin/v3/tasks/core/resolveCommand.js +24 -0
- package/bin/v3/tasks/core/showUsage.js +43 -0
- package/index.js +22 -0
- package/package.json +37 -0
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import readFile from "../common/readFile.js";
|
|
2
|
+
import writeFile from "../common/writeFile.js";
|
|
3
|
+
|
|
4
|
+
import buildUseLine from "./buildUseLine.js";
|
|
5
|
+
import checkDuplicate from "./checkDuplicate.js";
|
|
6
|
+
import findUseInsertIndex from "./findInsertIndex.js";
|
|
7
|
+
|
|
8
|
+
const appOrRouter = "router";
|
|
9
|
+
|
|
10
|
+
const updateAppUse = ({ appJsPath, endpoint, showLog, inFuncName }) => {
|
|
11
|
+
const summary = {
|
|
12
|
+
use: { added: false, skipped: false, line: null }
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
const content = readFile(appJsPath);
|
|
16
|
+
|
|
17
|
+
const useLine = buildUseLine({
|
|
18
|
+
inAppOrRouter: appOrRouter,
|
|
19
|
+
inEndpoint: endpoint,
|
|
20
|
+
inFuncName
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
const fromCheckDuplicate = checkDuplicate({
|
|
24
|
+
inContent: content,
|
|
25
|
+
inEndpoint: endpoint,
|
|
26
|
+
inAppOrRouter: appOrRouter
|
|
27
|
+
});
|
|
28
|
+
console.log("fromCheckDuplicate : ", fromCheckDuplicate);
|
|
29
|
+
|
|
30
|
+
if (fromCheckDuplicate) {
|
|
31
|
+
summary.use.skipped = true;
|
|
32
|
+
|
|
33
|
+
if (showLog) console.log(summary);
|
|
34
|
+
|
|
35
|
+
return summary;
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
const index = findUseInsertIndex(content);
|
|
39
|
+
|
|
40
|
+
const before = content.slice(0, index);
|
|
41
|
+
const lineNumber = before.split("\n").length + 1;
|
|
42
|
+
|
|
43
|
+
const updated =
|
|
44
|
+
before + "\n" + useLine + content.slice(index);
|
|
45
|
+
|
|
46
|
+
writeFile(appJsPath, updated);
|
|
47
|
+
|
|
48
|
+
summary.use.added = true;
|
|
49
|
+
summary.use.line = lineNumber;
|
|
50
|
+
|
|
51
|
+
if (showLog) console.log(summary);
|
|
52
|
+
|
|
53
|
+
return summary;
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
export default updateAppUse;
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
export const announce = ({ inResolvedFolderName }) => {
|
|
2
|
+
console.log(`[keshavsoft] Folder created: ${inResolvedFolderName}`);
|
|
3
|
+
|
|
4
|
+
console.log("");
|
|
5
|
+
console.log("Next:");
|
|
6
|
+
console.log("");
|
|
7
|
+
console.log(`cd ${inResolvedFolderName}`);
|
|
8
|
+
console.log(`npx kschema-api-gen AddSubRoute`);
|
|
9
|
+
};
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import fs from "fs";
|
|
2
|
+
import path from "path";
|
|
3
|
+
import { fileURLToPath } from "url";
|
|
4
|
+
|
|
5
|
+
const __filename = fileURLToPath(import.meta.url);
|
|
6
|
+
const __dirname = path.dirname(__filename);
|
|
7
|
+
|
|
8
|
+
export const locateSource = () => {
|
|
9
|
+
const templatePath = path.join(__dirname, "..", "template");
|
|
10
|
+
|
|
11
|
+
const versions = fs.readdirSync(templatePath);
|
|
12
|
+
|
|
13
|
+
const maxVersion = Math.max(
|
|
14
|
+
...versions.map(v => Number(v.replace("v", "")))
|
|
15
|
+
);
|
|
16
|
+
|
|
17
|
+
return path.join(templatePath, `v${maxVersion}`);
|
|
18
|
+
};
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import fs from "fs";
|
|
2
|
+
|
|
3
|
+
export default function resolveFolderName({ name, inDefaultFolderName = "InsertGenPk" }) {
|
|
4
|
+
const defaultFolderName = inDefaultFolderName;
|
|
5
|
+
|
|
6
|
+
// case 1: force new → timestamp
|
|
7
|
+
if (name === null) {
|
|
8
|
+
name = defaultFolderName;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
// case 2: user provided → strict
|
|
12
|
+
if (fs.existsSync(name)) {
|
|
13
|
+
return {
|
|
14
|
+
KTF: false,
|
|
15
|
+
KReason: `Folder already exists : ${name}`
|
|
16
|
+
};
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
return name;
|
|
20
|
+
};
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import updateImports from "./UpdateRoutesJs/updateImports/index.js";
|
|
2
|
+
import updateAppUse from "./UpdateRoutesJs/updateUse/index.js";
|
|
3
|
+
|
|
4
|
+
export default ({ appJsPath, endpoint }) => {
|
|
5
|
+
updateImports({ appJsPath, endpoint });
|
|
6
|
+
updateAppUse({ appJsPath, endpoint });
|
|
7
|
+
};
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import fs from 'fs';
|
|
2
|
+
|
|
3
|
+
function updateEndPointsFile({ filePath, inTableName }) {
|
|
4
|
+
try {
|
|
5
|
+
let content = readFile(filePath);
|
|
6
|
+
|
|
7
|
+
content = content.replace("<TABLE_NAME>", inTableName);
|
|
8
|
+
|
|
9
|
+
writeFile(filePath, content);
|
|
10
|
+
} catch (e) {
|
|
11
|
+
handleError(e);
|
|
12
|
+
};
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
// read
|
|
16
|
+
function readFile(filePath) {
|
|
17
|
+
return fs.readFileSync(filePath, "utf-8");
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
// write
|
|
21
|
+
function writeFile(filePath, content) {
|
|
22
|
+
fs.writeFileSync(filePath, content);
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
// error handler
|
|
26
|
+
function handleError(e) {
|
|
27
|
+
console.error("ROUTE USE ERROR:", e.message);
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
export { updateEndPointsFile };
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { startFunc as Service } from "./service.js";
|
|
2
|
+
import { ConflictError, StorageError } from "./errors.js";
|
|
3
|
+
|
|
4
|
+
const getFunc = async ({ req, res, inTablePath }) => {
|
|
5
|
+
try {
|
|
6
|
+
const fromService = await Service({
|
|
7
|
+
inTablePath
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
res.type("application/json").send(fromService);
|
|
11
|
+
} catch (err) {
|
|
12
|
+
|
|
13
|
+
if (err instanceof ConflictError)
|
|
14
|
+
return res.status(409).send(err.message);
|
|
15
|
+
|
|
16
|
+
if (err instanceof StorageError)
|
|
17
|
+
return res.status(500).send("Failed to persist data");
|
|
18
|
+
|
|
19
|
+
console.error(err);
|
|
20
|
+
res.status(500).send("Unexpected error");
|
|
21
|
+
}
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export default getFunc;
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
class AppError extends Error {
|
|
2
|
+
constructor(message) {
|
|
3
|
+
super(message);
|
|
4
|
+
this.name = this.constructor.name;
|
|
5
|
+
}
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
class ConflictError extends AppError { }
|
|
9
|
+
class StorageError extends AppError { }
|
|
10
|
+
|
|
11
|
+
export { AppError, ConflictError, StorageError };
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import path from "path";
|
|
2
|
+
|
|
3
|
+
import generateRest from "kschema-fs-api-gen-rest";
|
|
4
|
+
import fixEndpointsJs from "express-fix-endpoints-js";
|
|
5
|
+
|
|
6
|
+
import { locateSource } from "./ShowAll/steps/locateSource.js";
|
|
7
|
+
import { locateDestination } from "./ShowAll/steps/locateDestination.js";
|
|
8
|
+
import { createFolder } from "../../core/createFolder.js";
|
|
9
|
+
|
|
10
|
+
import { announce } from "./ShowAll/steps/announce.js";
|
|
11
|
+
|
|
12
|
+
import resolveFolderName from "./ShowAll/steps/resolveFolderName.js";
|
|
13
|
+
import actions from "./ShowAll/actions.json" with { type: "json" };
|
|
14
|
+
|
|
15
|
+
const startFunc = async ({ cmd = "", toPath, isAnnounce = true, checkBeforeCreate = true,
|
|
16
|
+
toConfigPath, inTargetPath, inGenerateRest = false
|
|
17
|
+
}) => {
|
|
18
|
+
|
|
19
|
+
const matched = actions;
|
|
20
|
+
|
|
21
|
+
const localToPath = toPath;
|
|
22
|
+
|
|
23
|
+
const resolvedFolderName = resolveFolderName({
|
|
24
|
+
name: cmd
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
if (resolvedFolderName.KTF === false) {
|
|
28
|
+
console.log(resolvedFolderName.KReason);
|
|
29
|
+
|
|
30
|
+
return;
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
const source = locateSource();
|
|
34
|
+
const destination = locateDestination({
|
|
35
|
+
inResolvedFolderName: resolvedFolderName,
|
|
36
|
+
toPath: localToPath
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
const createFolderResponse = createFolder({
|
|
40
|
+
source, destination,
|
|
41
|
+
isAnnounce, checkBeforeCreate
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
if (createFolderResponse.KTF) {
|
|
45
|
+
const fromEndPointsJs = await fixEndpointsJs({
|
|
46
|
+
endPointsJsPath: path.join(localToPath, "end-points.js"),
|
|
47
|
+
inActionName: cmd
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
if (inGenerateRest) {
|
|
51
|
+
generateRest({
|
|
52
|
+
toConfigPath, inTargetPath,
|
|
53
|
+
toPath: path.join(localToPath, resolvedFolderName),
|
|
54
|
+
});
|
|
55
|
+
};
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
if (isAnnounce) announce({ inResolvedFolderName: resolvedFolderName });
|
|
59
|
+
|
|
60
|
+
return resolvedFolderName;
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
export default startFunc;
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import fs from "fs";
|
|
2
|
+
|
|
3
|
+
export const createFolder = ({ source, destination, checkBeforeCreate = false, isAnnounce = true }) => {
|
|
4
|
+
if (checkBeforeCreate) {
|
|
5
|
+
return createFolderWithCheck({ source, destination, isAnnounce });
|
|
6
|
+
} else {
|
|
7
|
+
return createOnly({ source, destination });
|
|
8
|
+
};
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
const createOnly = ({ source, destination }) => {
|
|
12
|
+
fs.mkdirSync(destination, { recursive: true });
|
|
13
|
+
|
|
14
|
+
fs.cpSync(source, destination, { recursive: true });
|
|
15
|
+
|
|
16
|
+
return {
|
|
17
|
+
KTF: true
|
|
18
|
+
};
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
const createFolderWithCheck = ({ source, destination, isAnnounce }) => {
|
|
22
|
+
if (fs.existsSync(destination)) {
|
|
23
|
+
if (isAnnounce) console.log("Folder already exists :", destination);
|
|
24
|
+
|
|
25
|
+
return {
|
|
26
|
+
KTF: false,
|
|
27
|
+
KReason: "Folder already exists"
|
|
28
|
+
};
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
if (isAnnounce) console.log("Folder created :", destination);
|
|
32
|
+
|
|
33
|
+
return createOnly({ source, destination });
|
|
34
|
+
};
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import StartEndPoint from "../commands/startEndPoint.js";
|
|
2
|
+
import AddSubRoute from "../commands/addSubRoute.js";
|
|
3
|
+
import AddTableName from "../tasks/tables/addTableName.js";
|
|
4
|
+
// import ShowAll from "../commands/showAll.js";
|
|
5
|
+
import Insert from "../tasks/actions/insert.js";
|
|
6
|
+
import ShowAll from "../tasks/actions/showAll.js";
|
|
7
|
+
|
|
8
|
+
import CreateApi from "../Orchestration/CreateApi/createApi.js";
|
|
9
|
+
import InsertApi from "../Orchestration/InsertApi/insertApi.js";
|
|
10
|
+
|
|
11
|
+
// resolveCommand.js
|
|
12
|
+
const map = {
|
|
13
|
+
StartEndPoint,
|
|
14
|
+
AddSubRoute,
|
|
15
|
+
AddTableName,
|
|
16
|
+
ShowAll,
|
|
17
|
+
CreateApi,
|
|
18
|
+
InsertApi,
|
|
19
|
+
Insert
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
export default function resolveCommand(cmd) {
|
|
23
|
+
return map[cmd] || null;
|
|
24
|
+
};
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
/*
|
|
2
|
+
KSchema CLI – Entry Flow
|
|
3
|
+
|
|
4
|
+
1. Read user input from terminal (parseInput)
|
|
5
|
+
2. If no command → show usage (first-time user safety)
|
|
6
|
+
3. If help flags → show usage (quick guidance)
|
|
7
|
+
4. Resolve command dynamically (no hardcoding logic)
|
|
8
|
+
5. If command not found → inform + guide back to usage
|
|
9
|
+
6. Execute command with parsed input
|
|
10
|
+
|
|
11
|
+
Goal:
|
|
12
|
+
- Zero confusion for user
|
|
13
|
+
- Single source of truth (showUsage)
|
|
14
|
+
- Easy to extend (just add commands, no core changes)
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
export default function showUsage(version) {
|
|
18
|
+
const g = "\x1b[32m";
|
|
19
|
+
const y = "\x1b[33m";
|
|
20
|
+
const c = "\x1b[36m";
|
|
21
|
+
const gray = "\x1b[90m";
|
|
22
|
+
const r = "\x1b[0m";
|
|
23
|
+
|
|
24
|
+
console.log(`
|
|
25
|
+
${c}🚀 KSchema Api Generator v${version}${r}
|
|
26
|
+
|
|
27
|
+
${y}Usage:${r}
|
|
28
|
+
${g}npx @keshavsoft/kschema-api-gen${r} <command> [options]
|
|
29
|
+
|
|
30
|
+
${y}Commands:${r}
|
|
31
|
+
${g}StartEndPoint${r} Initialize a new folder and files
|
|
32
|
+
${g}CreateApi${r} Creates new end point and hooks to app.js
|
|
33
|
+
${g}InsertApi${r} Creates new InsertApi end point and hooks to app.js
|
|
34
|
+
|
|
35
|
+
${y}Examples:${r}
|
|
36
|
+
${gray}npx @keshavsoft/kschema-api-gen StartEndPoint${r}
|
|
37
|
+
${gray}npx @keshavsoft/kschema-api-gen CreateApi Api/V1/journals/ShowAll${r}
|
|
38
|
+
${gray}npx @keshavsoft/kschema-api-gen InsertApi Api/V1/journals/Insert${r}
|
|
39
|
+
|
|
40
|
+
${y}Tip:${r}
|
|
41
|
+
${gray}npm i -g @keshavsoft/kschema-api-gen${r}
|
|
42
|
+
`);
|
|
43
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import fs from "fs";
|
|
2
|
+
|
|
3
|
+
export const createFolder = ({ source, destination, checkBeforeCreate = false, isAnnounce = true }) => {
|
|
4
|
+
if (checkBeforeCreate) {
|
|
5
|
+
return createFolderWithCheck({ source, destination, isAnnounce });
|
|
6
|
+
} else {
|
|
7
|
+
return createOnly({ source, destination });
|
|
8
|
+
};
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
const createOnly = ({ source, destination }) => {
|
|
12
|
+
fs.mkdirSync(destination, { recursive: true });
|
|
13
|
+
|
|
14
|
+
fs.cpSync(source, destination, { recursive: true });
|
|
15
|
+
|
|
16
|
+
return {
|
|
17
|
+
KTF: true
|
|
18
|
+
};
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
const createFolderWithCheck = ({ source, destination, isAnnounce }) => {
|
|
22
|
+
if (fs.existsSync(destination)) {
|
|
23
|
+
if (isAnnounce) console.log("Folder already exists :", destination);
|
|
24
|
+
|
|
25
|
+
return {
|
|
26
|
+
KTF: false,
|
|
27
|
+
KReason: "Folder already exists"
|
|
28
|
+
};
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
if (isAnnounce) console.log("Folder created :", destination);
|
|
32
|
+
|
|
33
|
+
return createOnly({ source, destination });
|
|
34
|
+
};
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import actions from "../config/actions.json" with { type: "json" };
|
|
2
|
+
|
|
3
|
+
export default async function resolveCommand(cmd) {
|
|
4
|
+
const matched = actions.find(x => x.cmd === cmd);
|
|
5
|
+
|
|
6
|
+
if (!matched) return null;
|
|
7
|
+
|
|
8
|
+
const module = await import(`../tasks/actions/${matched.file}.js`);
|
|
9
|
+
|
|
10
|
+
return module.default;
|
|
11
|
+
};
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
/*
|
|
2
|
+
KSchema CLI – Execution Flow
|
|
3
|
+
|
|
4
|
+
1. Parse terminal input
|
|
5
|
+
2. Show usage if command/help missing
|
|
6
|
+
3. Resolve command dynamically
|
|
7
|
+
4. Load command dynamically
|
|
8
|
+
5. Execute action
|
|
9
|
+
|
|
10
|
+
Architecture Goals:
|
|
11
|
+
- JSON driven command system
|
|
12
|
+
- Zero hardcoded command maps
|
|
13
|
+
- Single source of truth
|
|
14
|
+
- Easy scalability
|
|
15
|
+
- Beginner friendly structure
|
|
16
|
+
*/
|
|
17
|
+
import actions from "../config/actions.json" with { type: "json" };
|
|
18
|
+
|
|
19
|
+
export default function showUsage(version) {
|
|
20
|
+
const g = "\x1b[32m";
|
|
21
|
+
const y = "\x1b[33m";
|
|
22
|
+
const c = "\x1b[36m";
|
|
23
|
+
const gray = "\x1b[90m";
|
|
24
|
+
const r = "\x1b[0m";
|
|
25
|
+
|
|
26
|
+
const commandsText = actions.map((item) => {
|
|
27
|
+
return ` ${g}${item.exportFile}${r} ${item.description}`;
|
|
28
|
+
}).join("\n");
|
|
29
|
+
|
|
30
|
+
const examplesText = actions.map((item) => {
|
|
31
|
+
return ` ${gray}npx @keshavsoft/kschema-api-gen-actions ${item.exportFile}${r}`;
|
|
32
|
+
}).join("\n");
|
|
33
|
+
|
|
34
|
+
console.log(`
|
|
35
|
+
${c}🚀 KSchema Api Generator v${version}${r}
|
|
36
|
+
|
|
37
|
+
${y}Usage:${r}
|
|
38
|
+
${g}npx @keshavsoft/kschema-api-gen-actions${r} <command> [options]
|
|
39
|
+
|
|
40
|
+
${y}Commands:${r}
|
|
41
|
+
${commandsText}
|
|
42
|
+
|
|
43
|
+
${y}Examples:${r}
|
|
44
|
+
${examplesText}
|
|
45
|
+
|
|
46
|
+
${y}Tip:${r}
|
|
47
|
+
${gray}npm i -g @keshavsoft/kschema-api-gen-actions${r}
|
|
48
|
+
`);
|
|
49
|
+
}
|
package/bin/v3/start.js
ADDED
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import parseInput from "./core/parseInput.js";
|
|
2
|
+
import resolveCommand from "./core/resolveCommand.js";
|
|
3
|
+
import showUsage from './core/showUsage.js';
|
|
4
|
+
import pkg from '../../package.json' with { type: 'json' };
|
|
5
|
+
|
|
6
|
+
const version = pkg.version;
|
|
7
|
+
|
|
8
|
+
const run = async () => {
|
|
9
|
+
const input = parseInput();
|
|
10
|
+
|
|
11
|
+
if (!input.cmd) return showUsage(version);
|
|
12
|
+
|
|
13
|
+
if (input.cmd === "--help" || input.cmd === "-h" || input.cmd === "help") return showUsage(version);
|
|
14
|
+
|
|
15
|
+
const command = await resolveCommand(input.cmd);
|
|
16
|
+
|
|
17
|
+
if (!command) return (console.log(`Unknown command: ${input.cmd}\n`), showUsage(version));
|
|
18
|
+
|
|
19
|
+
await command(input);
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
export default run;
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
{
|
|
2
|
+
"cmd": "WithMail",
|
|
3
|
+
"file": "withMail",
|
|
4
|
+
"exportFile": "WithMail",
|
|
5
|
+
"group": "PostMethods",
|
|
6
|
+
"description": "Generate WithMail POST action",
|
|
7
|
+
"endPointsJs": {
|
|
8
|
+
"importLines": {
|
|
9
|
+
"toInsertLine": "import funcFromWithMail from './WithMail/controller.js';",
|
|
10
|
+
"duplicationCheck": "from './WithMail/controller.js'",
|
|
11
|
+
"insertAfter": [
|
|
12
|
+
"import funcFrom",
|
|
13
|
+
"import express"
|
|
14
|
+
]
|
|
15
|
+
},
|
|
16
|
+
"useLines": {
|
|
17
|
+
"toInsertLine1": "router.get('/WithMail', (req, res) => funcFromWithMail({ res, inTablePath: tablePath }));",
|
|
18
|
+
"toInsertLine": "router.post('/WithMail', express.json(), (req, res) => funcFromWithMail({ req, res, inTablePath: tablePath, inConfigPath: configPath }));",
|
|
19
|
+
"duplicationCheck": "router.use('/WithMail'",
|
|
20
|
+
"insertAfter": [
|
|
21
|
+
"router.",
|
|
22
|
+
"const router = "
|
|
23
|
+
]
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
const buildImport = ({ inFuncName, inEndpoint }) => {
|
|
2
|
+
const localEndpoint = inEndpoint;
|
|
3
|
+
|
|
4
|
+
const safeName = localEndpoint.replace(/[^a-zA-Z0-9]/g, "_");
|
|
5
|
+
|
|
6
|
+
return `import { ${inFuncName} } from "./${localEndpoint}/controller.js";`;
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
export default buildImport;
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
const findInsertIndex = (inContent) => {
|
|
2
|
+
const localContent = inContent;
|
|
3
|
+
|
|
4
|
+
const matches = [...localContent.matchAll(/^\s*import.*$/gm)];
|
|
5
|
+
const last = matches.at(-1);
|
|
6
|
+
|
|
7
|
+
return last ? last.index + last[0].length : 0;
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
export default findInsertIndex;
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import readFile from "../common/readFile.js";
|
|
2
|
+
import buildImport from "./buildImport.js";
|
|
3
|
+
import checkDuplicate from "./checkDuplicate.js";
|
|
4
|
+
import findInsertIndex from "./findInsertIndex.js";
|
|
5
|
+
import writeFile from "../common/writeFile.js";
|
|
6
|
+
|
|
7
|
+
const updateImports = ({ appJsPath, endpoint, showLog, inFuncName }) => {
|
|
8
|
+
const summary = {
|
|
9
|
+
import: { added: false, line: null },
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
const content = readFile(appJsPath);
|
|
13
|
+
|
|
14
|
+
const importLine = buildImport({
|
|
15
|
+
inEndpoint: endpoint,
|
|
16
|
+
inFuncName
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
const fromCheckDuplicate = checkDuplicate({ inContent: content, inFuncName });
|
|
20
|
+
|
|
21
|
+
if (fromCheckDuplicate) {
|
|
22
|
+
summary.import.skipped = true;
|
|
23
|
+
|
|
24
|
+
if (showLog) console.log(summary);
|
|
25
|
+
|
|
26
|
+
return summary;
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
const index = findInsertIndex(content);
|
|
30
|
+
|
|
31
|
+
const before = content.slice(0, index);
|
|
32
|
+
const lineNumber = before.split("\n").length + 1;
|
|
33
|
+
|
|
34
|
+
const updated =
|
|
35
|
+
before + "\n" + importLine + content.slice(index);
|
|
36
|
+
|
|
37
|
+
writeFile(appJsPath, updated);
|
|
38
|
+
|
|
39
|
+
summary.import.added = true;
|
|
40
|
+
summary.import.line = lineNumber;
|
|
41
|
+
|
|
42
|
+
if (showLog) console.log(summary);
|
|
43
|
+
|
|
44
|
+
return summary;
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
export default updateImports;
|