Files
safeImport/src/index.mjs

120 lines
4.4 KiB
JavaScript
Raw Normal View History

2025-07-09 13:26:51 +01:00
import { readFileSync ,mkdirSync} from 'node:fs';
2025-07-21 17:01:43 +01:00
import { writeFile } from 'node:fs/promises';
import { Project} from 'ts-morph';
2025-07-21 17:01:43 +01:00
import {getSliceAndInfoSync} from 'slice-js/src/slice-code/test/helpers/utils.js';
import path from 'node:path';
2025-08-01 20:19:24 +01:00
import { getImportCallsAndArgumentTypes } from './tsCalls.mjs';
import { wpCompress } from './bundle/index.mjs';
import { LibraryTypesRecorder } from './libcalls.mjs';
2025-07-09 13:26:51 +01:00
/**
2025-08-01 20:19:24 +01:00
*
* @param {ReturnType<LibraryTypesRecorder['generateAllArgumentsForRecordedCalls']>} calls
2025-08-01 20:19:24 +01:00
* @param {string} FILE_PATH
2025-07-09 13:26:51 +01:00
*/
2025-08-01 20:19:24 +01:00
export async function sliceAndWriteCalls(calls, FILE_PATH) {
2025-07-21 17:01:43 +01:00
const writePromises = [];
2025-08-01 20:19:24 +01:00
2025-07-09 13:26:51 +01:00
for (const [moduleName, callBox] of calls) {
2025-08-01 20:19:24 +01:00
if (isRelativeModule(moduleName) || isNodeModule(moduleName)) { // not relative module
console.warn(`Skipping module ${moduleName} - relative or inbuilt Node.js module`);
2025-07-09 13:26:51 +01:00
continue;
}
2025-08-01 20:19:24 +01:00
console.log(`Slicing module ${moduleName} - ${callBox.size} calls`);
// const relatedModuleNamePath = import.meta.resolve(moduleName);
// console.log(`Related module path`, relatedModuleNamePath);
const relatedModuleNamePath = await wpCompress(moduleName)
2025-07-09 13:26:51 +01:00
const fileSource = readFileSync(relatedModuleNamePath).toString('utf-8');
2025-08-01 20:19:24 +01:00
// continue; // TODO - handle relative modules
const { slicedCode } = getSliceAndInfoSync(fileSource, (moduleExports) => {
return [...callBox.entries()].flatMap(([methodName, methodArgsList]) => {
2025-07-09 13:26:51 +01:00
const methodNameNormed = methodName.substring(1);
2025-08-01 20:19:24 +01:00
console.log("Calls for ", methodNameNormed, methodArgsList);
return methodArgsList.map(methodArgsList => {
const methodObj = (methodNameNormed === '') ? moduleExports : moduleExports[methodNameNormed];
methodObj.apply(moduleExports[methodNameNormed], methodArgsList);
2025-07-21 17:01:43 +01:00
});
2025-08-01 20:19:24 +01:00
});
}, relatedModuleNamePath);
2025-08-01 20:51:30 +01:00
// console.log(`Sliced code ${moduleName}\n`,slicedCode);
// continue;
const writePath = path.resolve('./dist', moduleName,'index.cjs');
2025-08-01 20:19:24 +01:00
if (writePath === moduleName) {
throw Error("Unexpected Directory rewrite. Not allowed.");
2025-07-21 17:01:43 +01:00
}
2025-08-01 20:19:24 +01:00
mkdirSync(path.dirname(writePath), { recursive: true });
2025-08-01 20:51:30 +01:00
console.log(`Writing module '${moduleName}' to '${writePath}'`);
2025-07-09 13:26:51 +01:00
2025-08-01 20:19:24 +01:00
writePromises.push(writeFile(writePath, slicedCode));
2025-07-26 13:44:32 +01:00
2025-07-21 17:01:43 +01:00
}
2025-07-09 13:26:51 +01:00
2025-08-01 20:19:24 +01:00
Promise.all(writePromises).then(p => {
2025-08-01 20:51:30 +01:00
// console.log("write finished");
2025-07-21 17:01:43 +01:00
}).catch(console.log);
2025-07-09 13:26:51 +01:00
}
2025-08-01 20:19:24 +01:00
function main() {
// const FILE_PATH = './test_src/index.cjs';
const FILE_PATH = './test_src/index.mjs';
2025-07-26 13:44:32 +01:00
const project = new Project({compilerOptions:{allowJs: true, checkJs: false,}});
project.addSourceFileAtPathIfExists(FILE_PATH);
// const project = tsc.createProgram([FILE_PATH],);
const checker = project.getTypeChecker();
const sourceFile = project.getSourceFile(FILE_PATH)
2025-08-01 20:19:24 +01:00
2025-07-26 13:44:32 +01:00
const importDecls = sourceFile.getImportStringLiterals()
2025-08-01 20:19:24 +01:00
// foreach library, get a list of import calls
2025-07-26 13:44:32 +01:00
2025-08-01 20:19:24 +01:00
const calls = getImportCallsAndArgumentTypes(importDecls,checker,FILE_PATH);
const callMap = calls.generateAllArgumentsForRecordedCalls();
logCallList(callMap);
sliceAndWriteCalls(callMap, FILE_PATH).then(()=>{
console.log("Slicing and writing calls done");
});
2025-07-26 13:44:32 +01:00
}
2025-07-09 13:26:51 +01:00
if (process.argv[1] === import.meta.filename) {
2025-07-26 13:44:32 +01:00
console.log("[SafeImport] started");
2025-08-01 20:19:24 +01:00
main();
2025-08-01 20:51:30 +01:00
// console.log("done");
2025-07-09 13:26:51 +01:00
}
2025-08-01 20:19:24 +01:00
export function logCallList(calls) {
console.log(`[Call Log] Call List for ${calls.size} modules`);
2025-07-21 17:01:43 +01:00
for (const [moduleName, callBoxes] of calls.entries()) {
if (isRelativeModule(moduleName)) {
console.log('Importing', moduleName, callBoxes);
} else {
console.log(`Module "${moduleName}" - System module. FIXME skipping`);
}
}
console.log(`Call List`, calls);
2025-08-01 20:19:24 +01:00
console.log(`[Call Log] End List for ${calls.size} modules`);
2025-07-21 17:01:43 +01:00
}
function isRelativeModule(moduleName) {
return moduleName.startsWith('.');
}
2025-08-01 20:19:24 +01:00
/**
* True if an inbuilt Node.js module.
* @param {string} moduleName
* @returns
*/
function isNodeModule(moduleName) {
if(moduleName.startsWith('node:')) return true;
const nodeModules = ['fs', 'fs/promises', 'path', 'http', 'https', 'os', 'crypto']
return nodeModules.includes(moduleName);
}