"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.isValidFile = exports.isValidFileSync = exports.loadSnippet = exports.loadSnippetSync = exports.evalHelper = exports.evalHelperSync = exports.evalSnippet = exports.evalSnippetSync = exports.evaluateContent = exports.glob = exports.globSync = exports.rewriteSnippetToSyncVersion = exports.rewriteSnippetToAsyncVersion = exports.arrayBody = void 0;
const typescript_1 = __importStar(require("typescript"));
const fs_1 = __importStar(require("fs"));
const child_process_1 = require("child_process");
const path_1 = __importDefault(require("path"));
const fast_glob_1 = __importDefault(require("fast-glob"));
const sync_fetch_1 = __importDefault(require("sync-fetch"));
const url_1 = require("url");
const node_query_1 = __importDefault(require("@synvert-hq/node-query"));
const node_mutation_1 = __importDefault(require("@synvert-hq/node-mutation"));
const errors_1 = require("./errors");
const configuration_1 = __importDefault(require("./configuration"));
const Synvert = require("./synvert");
const REWRITER_METHODS = "addFile removeFile renameFile withinFiles withinFile addSnippet";
const SCOPE_METHODS = "withinNode withNode findNode gotoNode";
const CONDITION_METHODS = "ifExistNode unlessExistNode ifOnlyExistNode ifAllNodes";
// delete is a reserved word, we define another expression in GLOBAL_DSL_QUERY
const ACTION_METHODS = "group append prepend insert insertAfter insertBefore remove replace replaceWith noop";
const ALL_METHODS = `configure description ifNode ifNpm ${REWRITER_METHODS} ${SCOPE_METHODS} ${CONDITION_METHODS} ${ACTION_METHODS} callHelper wrapWithQuotes appendSemicolon addLeadingSpaces indent`;
const arrayBody = (node, nodeQueryAdapter) => {
switch (nodeQueryAdapter.getNodeType(node)) {
case "SourceFile":
return node["statements"];
case "ClassDeclaration":
return node["members"];
case "ClassDefinition":
return node["body"]["body"];
case "MethodDefinition":
return node["value"]["body"]["body"];
default:
return node["body"];
}
};
exports.arrayBody = arrayBody;
const NEW_REWRITER_WITH_FUNCTION_QUERY = new node_query_1.default(`.NewExpression[expression=Synvert.Rewriter][arguments.length=3][arguments.2=.FunctionExpression[modifiers=undefined]]`, { adapter: "typescript" });
const NEW_HELPER_WITH_FUNCTION_QUERY = new node_query_1.default(`.NewExpression[expression=Synvert.Helper][arguments.length=2][arguments.1=.FunctionExpression[modifiers=undefined]]`, { adapter: "typescript" });
const NEW_INSTANCE_WITH_FUNCTION_QUERY = new node_query_1.default(`.CallExpression[expression=.PropertyAccessExpression[expression=.ThisKeyword]
[name IN (withinFiles withinFile)]][arguments.length=2][arguments.1=.FunctionExpression[modifiers=undefined]]`, { adapter: "typescript" });
const SCOPES_AND_CONDITIONS_QUERY = new node_query_1.default(`.CallExpression[expression=.PropertyAccessExpression[expression=.ThisKeyword]
[name IN (${SCOPE_METHODS} ${CONDITION_METHODS})]]
[arguments.-1.nodeType IN (FunctionExpression ArrowFunction)][arguments.-1.modifiers=undefined]`, { adapter: "typescript" });
const GROUPS_QUERY = new node_query_1.default(`.CallExpression[expression=.PropertyAccessExpression[expression=.ThisKeyword]
[name=group]]
[arguments.-1.nodeType IN (FunctionExpression ArrowFunction)][arguments.-1.modifiers=undefined]`, { adapter: "typescript" });
const ASYNC_METHODS_QUERY = new node_query_1.default(`.CallExpression[expression=.PropertyAccessExpression[expression=.ThisKeyword]
[name IN (${REWRITER_METHODS} callHelper group ${SCOPE_METHODS} ${CONDITION_METHODS})]]`, { adapter: "typescript" });
/**
* Rewrite javascript snippet to async version.
*/
const rewriteSnippetToAsyncVersion = (snippet) => {
let newSnippet = addProperScopeToSnippet(snippet);
const node = parseCode(newSnippet);
const mutation = new node_mutation_1.default(newSnippet, {
adapter: "typescript",
});
NEW_REWRITER_WITH_FUNCTION_QUERY.queryNodes(node).forEach((node) => mutation.insert(node, "async ", { at: "beginning", to: "arguments.-1" }));
NEW_HELPER_WITH_FUNCTION_QUERY.queryNodes(node).forEach((node) => mutation.insert(node, "async ", { at: "beginning", to: "arguments.-1" }));
NEW_INSTANCE_WITH_FUNCTION_QUERY.queryNodes(node).forEach((node) => mutation.insert(node, "async ", { at: "beginning", to: "arguments.-1" }));
SCOPES_AND_CONDITIONS_QUERY.queryNodes(node).forEach((node) => mutation.insert(node, "async ", { at: "beginning", to: "arguments.-1" }));
GROUPS_QUERY.queryNodes(node).forEach((node) => mutation.insert(node, "async ", { at: "beginning", to: "arguments.-1" }));
ASYNC_METHODS_QUERY.queryNodes(node).forEach((node) => {
if (node.parent.kind != typescript_1.SyntaxKind.AwaitExpression) {
mutation.insert(node, "await ", { at: "beginning" });
}
});
const { affected, newSource } = mutation.process();
return affected ? newSource : newSnippet;
};
exports.rewriteSnippetToAsyncVersion = rewriteSnippetToAsyncVersion;
const SYNC_METHODS_QUERY = new node_query_1.default(`.CallExpression[expression=.PropertyAccessExpression[expression=.ThisKeyword]
[name IN (${REWRITER_METHODS} callHelper ${SCOPE_METHODS} ${CONDITION_METHODS})]]`, { adapter: "typescript" });
/**
* Rewrite javascript snippet to sync version.
*/
const rewriteSnippetToSyncVersion = (snippet) => {
let newSnippet = addProperScopeToSnippet(snippet);
const node = parseCode(newSnippet);
const mutation = new node_mutation_1.default(newSnippet, {
adapter: "typescript",
});
SYNC_METHODS_QUERY.queryNodes(node).forEach((node) => mutation.insert(node, "Sync", { at: "end", to: "expression" }));
const { affected, newSource } = mutation.process();
return affected ? newSource : newSnippet;
};
exports.rewriteSnippetToSyncVersion = rewriteSnippetToSyncVersion;
const NEW_REWRITER_WITH_ARROW_FUNCTION_QUERY = new node_query_1.default(`.NewExpression[expression=Synvert.Rewriter][arguments.length=3][arguments.2=.ArrowFunction]`, { adapter: "typescript" });
const NEW_HELPER_WITH_ARROW_FUNCTION_QUERY = new node_query_1.default(`.NewExpression[expression=Synvert.Helper][arguments.length=2][arguments.1=.ArrowFunction]`, { adapter: "typescript" });
const NEW_INSTANCE_WITH_ARROW_FUNCTION_QUERY = new node_query_1.default(`.CallExpression[expression IN (withinFiles withinFile)][arguments.length=2][arguments.1=.ArrowFunction]`, { adapter: "typescript" });
const GLOBAL_DSL_QUERY = new node_query_1.default(`.CallExpression[expression IN (${ALL_METHODS})],
.DeleteExpression[expression=.ParenthesizedExpression[expression.nodeType IN (StringLiteral ArrayLiteralExpression)]],
.DeleteExpression[expression=.ParenthesizedExpression[expression=.BinaryExpression[left.nodeType IN (StringLiteral ArrayLiteralExpression)][right.nodeType=ObjectLiteralExpression]]]`, { adapter: "typescript" });
const addProperScopeToSnippet = (snippet) => {
const node = parseCode(snippet);
const mutation = new node_mutation_1.default(snippet, {
adapter: "typescript",
});
NEW_REWRITER_WITH_ARROW_FUNCTION_QUERY.queryNodes(node).forEach((node) => {
mutation.delete(node, "arguments.2.equalsGreaterThanToken");
mutation.insert(node, "function ", { at: "beginning", to: "arguments.2" });
});
NEW_HELPER_WITH_ARROW_FUNCTION_QUERY.queryNodes(node).forEach((node) => {
mutation.delete(node, "arguments.1.equalsGreaterThanToken");
mutation.insert(node, "function ", { at: "beginning", to: "arguments.1" });
});
NEW_INSTANCE_WITH_ARROW_FUNCTION_QUERY.queryNodes(node).forEach((node) => {
mutation.delete(node, "arguments.1.equalsGreaterThanToken");
mutation.insert(node, "function ", { at: "beginning", to: "arguments.1" });
});
GLOBAL_DSL_QUERY.queryNodes(node).forEach((node) => {
mutation.insert(node, "this.", { at: "beginning" });
});
const { affected, newSource } = mutation.process();
return affected ? newSource : snippet;
};
const parseCode = (snippet) => {
return typescript_1.default.createSourceFile("test.js", snippet, typescript_1.default.ScriptTarget.Latest, true, typescript_1.default.ScriptKind.JS);
};
function runShellCommandSync(command, args, input) {
const child = (0, child_process_1.spawnSync)(command, args, {
cwd: configuration_1.default.rootPath,
input: input,
});
let output = child.stdout ? child.stdout.toString() : "";
let error = child.stderr ? child.stderr.toString() : "";
return { stdout: output, stderr: error };
}
/**
* Sync to glob matching files.
* @param {string} filePattern file pattern
* @returns {string[]} matching files
*/
const globSync = (filePattern) => {
const onlyPaths = configuration_1.default.onlyPaths.length > 0 ? configuration_1.default.onlyPaths : [""];
const fsStats = fast_glob_1.default.sync(onlyPaths.map((onlyPath) => path_1.default.join(onlyPath, filePattern)), {
ignore: configuration_1.default.skipPaths,
cwd: configuration_1.default.rootPath,
onlyFiles: true,
unique: true,
stats: true,
});
const allPaths = fsStats
.filter((fsStat) => fsStat.stats.size < configuration_1.default.maxFileSize)
.map((fsStat) => fsStat.path);
if (configuration_1.default.respectGitignore) {
const { stdout } = runShellCommandSync("git", ["check-ignore", "--stdin"], allPaths.join("\n"));
const ignoredPaths = new Set(stdout.split("\n").filter(Boolean));
return allPaths.filter((path) => !ignoredPaths.has(path));
}
return allPaths;
};
exports.globSync = globSync;
function runShellCommand(command, args, input) {
return new Promise((resolve) => {
const child = (0, child_process_1.spawn)(command, args, { cwd: configuration_1.default.rootPath });
if (child.stdin && input) {
child.stdin.write(input);
child.stdin.end();
}
let output = "";
if (child.stdout) {
child.stdout.on("data", (data) => {
output += data;
});
}
let error = "";
if (child.stderr) {
child.stderr.on("data", (data) => {
error += data;
});
}
child.on("error", (e) => {
return resolve({ stdout: "", stderr: e.message });
});
child.on("exit", () => {
return resolve({ stdout: output, stderr: error });
});
});
}
/**
* Async to glob matching files.
* @async
* @param {string} filePattern file pattern
* @returns {Promise<string[]>} matching files
*/
const glob = (filePattern) => __awaiter(void 0, void 0, void 0, function* () {
const onlyPaths = configuration_1.default.onlyPaths.length > 0 ? configuration_1.default.onlyPaths : [""];
const fsStats = yield (0, fast_glob_1.default)(onlyPaths.map((onlyPath) => path_1.default.join(onlyPath, filePattern)), {
ignore: configuration_1.default.skipPaths,
cwd: configuration_1.default.rootPath,
onlyFiles: true,
unique: true,
stats: true,
});
const allPaths = fsStats
.filter((fsStat) => fsStat.stats.size < configuration_1.default.maxFileSize)
.map((fsStat) => fsStat.path);
if (configuration_1.default.respectGitignore) {
const { stdout } = yield runShellCommand("git", ["check-ignore", "--stdin"], allPaths.join("\n"));
const ignoredPaths = new Set(stdout.split("\n").filter(Boolean));
return allPaths.filter((path) => !ignoredPaths.has(path));
}
return allPaths;
});
exports.glob = glob;
/**
* Helper function to safely evaluate snippet/helper content
* @param content The snippet content to evaluate
* @param type The type to replace ('Rewriter' or 'Helper')
* @returns The evaluated instance
*/
const evaluateContent = (content, type) => {
const fn = new Function('Synvert', `
let instance;
${content.replace(`new Synvert.${type}`, `return new Synvert.${type}`)}
return instance;
`);
return fn(Synvert);
};
exports.evaluateContent = evaluateContent;
/**
* Sync to eval the snippet by name.
* @param {string} snippetName - snippet name, it can be a http url, file path or short snippet name.
* @returns {Rewriter} a Rewriter object
*/
const evalSnippetSync = (snippetName) => {
const snippetContent = (0, exports.loadSnippetSync)(snippetName);
return (0, exports.evaluateContent)(snippetContent, 'Rewriter');
};
exports.evalSnippetSync = evalSnippetSync;
/**
* Async to eval the snippet by name.
* @async
* @param {string} snippetName - snippet name, it can be a http url, file path or short snippet name.
* @returns {Promise<Rewriter>} a Rewriter object
*/
const evalSnippet = (snippetName) => __awaiter(void 0, void 0, void 0, function* () {
const snippetContent = yield (0, exports.loadSnippet)(snippetName);
return (0, exports.evaluateContent)(snippetContent, 'Rewriter');
});
exports.evalSnippet = evalSnippet;
/**
* Sync to eval the helper by name.
* @param {string} helperName - helper name, it can be a http url, file path or helper name.
* @returns {Helper} a Helper object
*/
const evalHelperSync = (helperName) => {
const helperContent = (0, exports.loadSnippetSync)(helperName);
return (0, exports.evaluateContent)(helperContent, 'Helper');
};
exports.evalHelperSync = evalHelperSync;
/**
* Async to eval the helper by name.
* @async
* @param {string} helperName - helper name, it can be a http url, file path or helper name.
* @returns {Promise<Helper>} a Helper object
*/
const evalHelper = (helperName) => __awaiter(void 0, void 0, void 0, function* () {
const helperContent = yield (0, exports.loadSnippet)(helperName);
return (0, exports.evaluateContent)(helperContent, 'Helper');
});
exports.evalHelper = evalHelper;
/**
* Sync to load snippet by snippet name.
* @param {string} snippetName - snippet name, it can be a http url, file path or short snippet name.
* @returns {string} snippet helper content
* @throws {SnippetNotFoundError} snippet not found
*/
const loadSnippetSync = (snippetName) => {
const snippetContent = loadSnippetContentSync(snippetName);
return (0, exports.rewriteSnippetToSyncVersion)(snippetContent);
};
exports.loadSnippetSync = loadSnippetSync;
/**
* Sync to load snippet by snippet name.
* @async
* @param {string} snippetName - snippet name, it can be a http url, file path or short snippet name.
* @returns {Promise<string>} snippet helper content
* @throws {SnippetNotFoundError} snippet not found
*/
const loadSnippet = (snippetName) => __awaiter(void 0, void 0, void 0, function* () {
const snippetContent = yield loadSnippetContent(snippetName);
return (0, exports.rewriteSnippetToAsyncVersion)(snippetContent);
});
exports.loadSnippet = loadSnippet;
/**
* Sync to check if it is a valid file path.
* @param {string} path - file path
* @returns {boolean} gets true it is a valid file
*/
const isValidFileSync = (path) => {
try {
const stats = fs_1.default.statSync(path);
return stats.isFile();
}
catch (_a) {
return false;
}
};
exports.isValidFileSync = isValidFileSync;
/**
* Async to check if it is a valid file path.
* @async
* @param {string} path - file path
* @returns {Promise<boolean>} gets true it is a valid file
*/
const isValidFile = (path) => __awaiter(void 0, void 0, void 0, function* () {
try {
const stats = yield fs_1.promises.stat(path);
return stats.isFile();
}
catch (_a) {
return false;
}
});
exports.isValidFile = isValidFile;
const isValidUrl = (urlString) => {
try {
const url = new url_1.URL(urlString);
return url.protocol === "http:" || url.protocol === "https:";
}
catch (_a) {
return false;
}
};
const formatUrl = (url) => convertToGithubRawUrl(url);
const snippetExpandPath = (snippetName) => path_1.default.join(snippetsHome(), "lib", `${snippetName}.js`);
const remoteSnippetExistsSync = (snippetPath) => (0, sync_fetch_1.default)(snippetPath).status === 200;
const remoteSnippetExists = (snippetPath) => __awaiter(void 0, void 0, void 0, function* () {
const response = yield fetch(snippetPath);
return response.status === 200;
});
const remoteSnippetUrl = (snippetName) => `https://github.com/synvert-hq/synvert-snippets-javascript/blob/main/lib/${snippetName}.js`;
const snippetsHome = () => {
return (process.env.SYNVERT_SNIPPETS_HOME ||
path_1.default.join(process.env.HOME, ".synvert-javascript"));
};
const convertToGithubRawUrl = (url) => {
if (url.startsWith("https://github.com/")) {
return url
.replace("//github.com/", "//raw.githubusercontent.com/")
.replace("/blob/", "/");
}
if (url.startsWith("https://gist.github.com")) {
return (url.replace("gist.github.com/", "gist.githubusercontent.com/") + "/raw");
}
return url;
};
const loadSnippetContent = (snippetName) => __awaiter(void 0, void 0, void 0, function* () {
if (isValidUrl(snippetName)) {
const snippetUrl = formatUrl(snippetName);
if (yield remoteSnippetExists(snippetUrl)) {
const response = yield fetch(snippetUrl);
return yield response.text();
}
throw new errors_1.SnippetNotFoundError(`${snippetName} not found`);
}
else if (yield (0, exports.isValidFile)(snippetName)) {
return yield fs_1.promises.readFile(snippetName, "utf-8");
}
else {
const snippetPath = snippetExpandPath(snippetName);
if (yield (0, exports.isValidFile)(snippetPath)) {
return yield fs_1.promises.readFile(snippetPath, "utf-8");
}
const snippetUrl = formatUrl(remoteSnippetUrl(snippetName));
if (yield remoteSnippetExists(snippetUrl)) {
const response = yield fetch(snippetUrl);
return yield response.text();
}
throw new errors_1.SnippetNotFoundError(`${snippetName} not found`);
}
});
const loadSnippetContentSync = (snippetName) => {
if (isValidUrl(snippetName)) {
const snippetUrl = formatUrl(snippetName);
if (remoteSnippetExistsSync(snippetUrl)) {
return (0, sync_fetch_1.default)(snippetUrl).text();
}
throw new errors_1.SnippetNotFoundError(`${snippetName} not found`);
}
else if ((0, exports.isValidFileSync)(snippetName)) {
return fs_1.default.readFileSync(snippetName, "utf-8");
}
else {
const snippetPath = snippetExpandPath(snippetName);
if ((0, exports.isValidFileSync)(snippetPath)) {
return fs_1.default.readFileSync(snippetPath, "utf-8");
}
const snippetUrl = formatUrl(remoteSnippetUrl(snippetName));
if (remoteSnippetExistsSync(snippetUrl)) {
return (0, sync_fetch_1.default)(snippetUrl).text();
}
throw new errors_1.SnippetNotFoundError(`${snippetName} not found`);
}
};