UNPKG

@nlabs/lex

Version:
302 lines (301 loc) 38 kB
/** * Copyright (c) 2018-Present, Nitrogen Labs, Inc. * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms. */ import boxen from 'boxen'; import chalk from 'chalk'; import { copyFile, existsSync, lstatSync, mkdirSync, readdirSync, readFileSync, writeFileSync } from 'fs'; import { sync as globSync } from 'glob'; import isEmpty from 'lodash/isEmpty.js'; import ora from 'ora'; import { basename as pathBasename, join as pathJoin, relative as pathRelative, resolve as pathResolve } from 'path'; import { rimrafSync } from 'rimraf'; import { log } from './log.js'; export const cwd = process.cwd(); export const getFilenames = (props)=>{ const { callback, cliName, name, quiet, type, useTypescript } = props; let nameCaps; const itemTypes = [ 'stores', 'views' ]; if (!name) { if (itemTypes.includes(type)) { log(`\n${cliName} Error: ${type} name is required. Please use 'lex -h' for options.`, 'error', quiet); callback?.(1); return undefined; } } else { nameCaps = `${name.charAt(0).toUpperCase()}${name.substr(1)}`; } log(`${cliName} adding ${name} ${type}...`, 'info', quiet); let templatePath; let templateExt; let templateReact; if (useTypescript) { templatePath = '../../templates/typescript'; templateExt = '.ts'; templateReact = '.tsx'; } else { templatePath = '../../templates/flow'; templateExt = '.js'; templateReact = '.js'; } return { nameCaps, templateExt, templatePath, templateReact }; }; export const createSpinner = (quiet = false)=>{ if (quiet) { return { fail: ()=>{}, start: ()=>{}, succeed: ()=>{} }; } return ora({ color: 'yellow' }); }; export const createProgressBar = (percentage)=>{ const width = 20; const filled = Math.round(percentage / 100 * width); const empty = width - filled; const filledBar = chalk.cyan('█').repeat(filled); const emptyBar = chalk.gray('░').repeat(empty); return filledBar + emptyBar; }; export const handleWebpackProgress = (output, spinner, quiet, emoji, action)=>{ if (quiet) { return; } const progressMatch = output.match(/\[webpack\.Progress\] (\d+)%/); if (progressMatch) { const progress = parseInt(progressMatch[1]); const progressBar = createProgressBar(progress); spinner.text = `${emoji} ${action}: ${progressBar} ${progress}%`; } else if (output.includes('[webpack.Progress]')) { const generalProgressMatch = output.match(/(\d+)%/); if (generalProgressMatch) { const progress = parseInt(generalProgressMatch[1]); const progressBar = createProgressBar(progress); spinner.text = `${emoji} ${action}: ${progressBar} ${progress}%`; } } }; export const copyFiles = async (files, typeName, spinner, config)=>{ const { outputFullPath, sourceFullPath } = config; const items = files.map((fileName)=>({ from: fileName, to: pathResolve(outputFullPath, pathRelative(sourceFullPath, fileName)) })); try { spinner.start(`Copying ${typeName} files...`); await Promise.all(items.map(({ from, to })=>new Promise((resolve, reject)=>{ mkdirSync(pathResolve(to, '..'), { recursive: true }); return copyFile(from, to, (copyError)=>{ if (copyError) { reject(); } else { resolve(true); } }); }))); spinner.succeed(`Successfully copied ${files.length} ${typeName} files!`); } catch (error) { spinner.fail(`Copying of ${typeName} files failed.`); log(`Error: ${error.message}`, 'error'); log(error, 'error'); } }; export const copyConfiguredFiles = async (spinner, config, quiet)=>{ const { copyFiles: copyFilesConfig, outputFullPath, sourceFullPath, sourcePath } = config; if (!copyFilesConfig || copyFilesConfig.length === 0) { return; } try { spinner.start('Copying configured files...'); let totalCopied = 0; const baseDir = sourceFullPath || (sourcePath ? pathResolve(cwd, sourcePath) : cwd); const allCopyPromises = []; for (const pattern of copyFilesConfig){ const resolvedPattern = pathResolve(baseDir, pattern); const matchingFiles = globSync(resolvedPattern, { absolute: true, nodir: true }); if (matchingFiles.length === 0) { if (!quiet) { log(`Warning: No files found matching pattern: ${pattern}`, 'warn', quiet); } continue; } const copyPromises = matchingFiles.map((sourceFile)=>{ const relativePath = pathRelative(baseDir, sourceFile); const destPath = pathResolve(outputFullPath, relativePath); const destDir = pathResolve(destPath, '..'); mkdirSync(destDir, { recursive: true }); return new Promise((resolve, reject)=>{ copyFile(sourceFile, destPath, (copyError)=>{ if (copyError) { reject(copyError); } else { resolve(true); } }); }); }); allCopyPromises.push(...copyPromises); totalCopied += matchingFiles.length; } await Promise.all(allCopyPromises); if (totalCopied > 0) { spinner.succeed(`Successfully copied ${totalCopied} configured files!`); } else { spinner.succeed('No configured files to copy'); } } catch (error) { spinner.fail('Failed to copy configured files'); log(`Error copying configured files: ${error.message}`, 'error', quiet); throw error; } }; export const copyFileSync = (source, target)=>{ let targetFile = target; if (existsSync(target)) { if (lstatSync(target).isDirectory()) { targetFile = pathJoin(target, pathBasename(source)); } } writeFileSync(targetFile, readFileSync(source)); }; export const copyFolderRecursiveSync = (source, target)=>{ let files = []; const targetFolder = pathJoin(target, pathBasename(source)); if (!existsSync(targetFolder)) { mkdirSync(targetFolder); } if (lstatSync(source).isDirectory()) { files = readdirSync(source); files.forEach((file)=>{ const curSource = pathJoin(source, file); if (lstatSync(curSource).isDirectory()) { copyFolderRecursiveSync(curSource, targetFolder); } else { copyFileSync(curSource, targetFolder); } }); } }; export const getPackageJson = (packagePath)=>{ const formatPath = packagePath || `${process.cwd()}/package.json`; const packageData = readFileSync(formatPath).toString(); return JSON.parse(packageData); }; export const getFilesByExt = (ext, config)=>{ const { sourceFullPath } = config; return globSync(`**/**${ext}`, { absolute: true, cwd: sourceFullPath, nodir: true }); }; export const removeConflictModules = (moduleList)=>{ const updatedList = { ...moduleList }; Object.keys(updatedList).forEach((moduleName)=>{ const regex = new RegExp('^(?!@types/).*?(jest|webpack).*$', 'gi'); if (regex.test(moduleName)) { delete updatedList[moduleName]; } }); return updatedList; }; export const removeFiles = (fileName, isRelative = false)=>new Promise((resolve, reject)=>{ const filePath = isRelative ? pathResolve(cwd, fileName) : fileName; try { rimrafSync(filePath); return resolve(null); } catch (error) { return reject(error); } }); export const removeModules = ()=>new Promise(async (resolve, reject)=>{ try { await removeFiles('./node_modules', true); await removeFiles('./yarn.lock', true); await removeFiles('./package-lock.json', true); resolve(null); } catch (error) { reject(error); } }); export const setPackageJson = (json, packagePath)=>{ if (!json) { return; } const formatPath = packagePath || `${process.cwd()}/package.json`; writeFileSync(formatPath, JSON.stringify(json, null, 2)); }; export const linkedModules = (startPath)=>{ const workingPath = startPath || process.cwd(); let modulePath; let prefix; if (workingPath.includes('@')) { prefix = `@${workingPath.split('@').pop()}`; modulePath = workingPath; } else { modulePath = pathJoin(workingPath, 'node_modules'); } const foundPaths = globSync('*', { absolute: true, cwd: modulePath, nodir: false }); return foundPaths.reduce((list, foundPath)=>{ try { const stats = lstatSync(foundPath); if (stats.isDirectory()) { const deepList = linkedModules(foundPath); list.push(...deepList); } else if (stats.isSymbolicLink()) { const moduleNames = [ prefix, pathBasename(foundPath) ].filter((item)=>!isEmpty(item)); list.push({ name: `${moduleNames.join('/')}`, path: foundPath }); } return list; } catch { return list; } }, []); }; export const checkLinkedModules = ()=>{ const linked = linkedModules(); if (linked.length) { const msgModule = linked.length > 1 ? 'Modules' : 'Module'; const linkedMsg = linked.reduce((msg, linkedModule)=>`${msg}\n * ${linkedModule.name}`, `Linked ${msgModule}:`); log(boxen(linkedMsg, { dimBorder: true, padding: 1 }), 'warn'); } }; export const updateTemplateName = (filePath, replace, replaceCaps)=>{ let data = readFileSync(filePath, 'utf8'); data = data.replace(/sample/g, replace); data = data.replace(/Sample/g, replaceCaps); writeFileSync(filePath, data, 'utf8'); }; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/utils/app.ts"],"sourcesContent":["/**\n * Copyright (c) 2018-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\n\nimport boxen from 'boxen';\nimport chalk from 'chalk';\nimport {copyFile, existsSync, lstatSync, mkdirSync, readdirSync, readFileSync, writeFileSync} from 'fs';\nimport {sync as globSync} from 'glob';\nimport isEmpty from 'lodash/isEmpty.js';\nimport ora from 'ora';\nimport {basename as pathBasename, join as pathJoin, relative as pathRelative, resolve as pathResolve} from 'path';\nimport {rimrafSync} from 'rimraf';\n\n\nimport {log} from './log.js';\n\nimport type {LexConfigType} from '../LexConfig.js';\n\n\nexport const cwd: string = process.cwd();\n\nexport interface GetFilenamesProps {\n  readonly callback?: (status: number)=> void;\n  readonly cliName?: string;\n  readonly name?: string;\n  readonly quiet?: boolean;\n  readonly type?: string;\n  readonly useTypescript?: boolean;\n}\n\ninterface FilenamesResult {\n  nameCaps: string;\n  templateExt: string;\n  templatePath: string;\n  templateReact: string;\n}\n\nexport const getFilenames = (props: GetFilenamesProps): FilenamesResult | undefined => {\n  const {callback, cliName, name, quiet, type, useTypescript} = props;\n\n  let nameCaps: string;\n  const itemTypes: string[] = ['stores', 'views'];\n\n  if(!name) {\n    if(itemTypes.includes(type)) {\n      log(`\\n${cliName} Error: ${type} name is required. Please use 'lex -h' for options.`, 'error', quiet);\n      callback?.(1);\n      return undefined;\n    }\n  } else {\n    nameCaps = `${name.charAt(0).toUpperCase()}${name.substr(1)}`;\n  }\n\n  log(`${cliName} adding ${name} ${type}...`, 'info', quiet);\n\n  let templatePath: string;\n  let templateExt: string;\n  let templateReact: string;\n\n  if(useTypescript) {\n    templatePath = '../../templates/typescript';\n    templateExt = '.ts';\n    templateReact = '.tsx';\n  } else {\n    templatePath = '../../templates/flow';\n    templateExt = '.js';\n    templateReact = '.js';\n  }\n\n  return {\n    nameCaps,\n    templateExt,\n    templatePath,\n    templateReact\n  };\n};\n\nexport interface Spinner {\n  fail: (text?: string)=> void;\n  start: (text?: string)=> void;\n  succeed: (text?: string)=> void;\n  text?: string;\n}\n\nexport const createSpinner = (quiet = false): Spinner => {\n  if(quiet) {\n    return {\n      fail: () => {},\n      start: () => {},\n      succeed: () => {}\n    };\n  }\n\n  return ora({color: 'yellow'});\n};\n\nexport const createProgressBar = (percentage: number): string => {\n  const width = 20;\n  const filled = Math.round((percentage / 100) * width);\n  const empty = width - filled;\n\n  const filledBar = chalk.cyan('█').repeat(filled);\n  const emptyBar = chalk.gray('░').repeat(empty);\n\n  return filledBar + emptyBar;\n};\n\nexport const handleWebpackProgress = (\n  output: string,\n  spinner: Spinner,\n  quiet: boolean,\n  emoji: string,\n  action: string\n): void => {\n  if(quiet) {\n    return;\n  }\n\n  const progressMatch = output.match(/\\[webpack\\.Progress\\] (\\d+)%/);\n  if(progressMatch) {\n    const progress = parseInt(progressMatch[1]);\n    const progressBar = createProgressBar(progress);\n    spinner.text = `${emoji} ${action}: ${progressBar} ${progress}%`;\n  } else if(output.includes('[webpack.Progress]')) {\n    const generalProgressMatch = output.match(/(\\d+)%/);\n    if(generalProgressMatch) {\n      const progress = parseInt(generalProgressMatch[1]);\n      const progressBar = createProgressBar(progress);\n      spinner.text = `${emoji} ${action}: ${progressBar} ${progress}%`;\n    }\n  }\n};\n\nexport const copyFiles = async (files: string[], typeName: string, spinner, config: LexConfigType) => {\n  const {outputFullPath, sourceFullPath} = config;\n  const items = files.map((fileName: string) => ({\n    from: fileName,\n    to: pathResolve(outputFullPath, pathRelative(sourceFullPath, fileName))\n  }));\n\n  try {\n    spinner.start(`Copying ${typeName} files...`);\n    await Promise.all(items.map(({from, to}) => new Promise(\n      (resolve, reject) => {\n        mkdirSync(pathResolve(to, '..'), {recursive: true});\n        return copyFile(from, to, (copyError) => {\n          if(copyError) {\n            reject();\n          } else {\n            resolve(true);\n          }\n        });\n      }\n    )));\n    spinner.succeed(`Successfully copied ${files.length} ${typeName} files!`);\n  } catch(error) {\n    spinner.fail(`Copying of ${typeName} files failed.`);\n    log(`Error: ${error.message}`, 'error');\n    log(error, 'error');\n  }\n};\n\nexport const copyConfiguredFiles = async (spinner, config: LexConfigType, quiet: boolean) => {\n  const {copyFiles: copyFilesConfig, outputFullPath, sourceFullPath, sourcePath} = config;\n  if(!copyFilesConfig || copyFilesConfig.length === 0) {\n    return;\n  }\n\n  try {\n    spinner.start('Copying configured files...');\n    let totalCopied = 0;\n\n    const baseDir = sourceFullPath || (sourcePath ? pathResolve(cwd, sourcePath) : cwd);\n    const allCopyPromises: Promise<unknown>[] = [];\n\n    for(const pattern of copyFilesConfig) {\n      const resolvedPattern = pathResolve(baseDir, pattern);\n      const matchingFiles = globSync(resolvedPattern, {\n        absolute: true,\n        nodir: true\n      });\n\n      if(matchingFiles.length === 0) {\n        if(!quiet) {\n          log(`Warning: No files found matching pattern: ${pattern}`, 'warn', quiet);\n        }\n        continue;\n      }\n\n      const copyPromises = matchingFiles.map((sourceFile) => {\n        const relativePath = pathRelative(baseDir, sourceFile);\n        const destPath = pathResolve(outputFullPath, relativePath);\n        const destDir = pathResolve(destPath, '..');\n        mkdirSync(destDir, {recursive: true});\n\n        return new Promise((resolve, reject) => {\n          copyFile(sourceFile, destPath, (copyError) => {\n            if(copyError) {\n              reject(copyError);\n            } else {\n              resolve(true);\n            }\n          });\n        });\n      });\n\n      allCopyPromises.push(...copyPromises);\n      totalCopied += matchingFiles.length;\n    }\n\n    await Promise.all(allCopyPromises);\n\n    if(totalCopied > 0) {\n      spinner.succeed(`Successfully copied ${totalCopied} configured files!`);\n    } else {\n      spinner.succeed('No configured files to copy');\n    }\n  } catch(error) {\n    spinner.fail('Failed to copy configured files');\n    log(`Error copying configured files: ${error.message}`, 'error', quiet);\n    throw error;\n  }\n};\n\nexport const copyFileSync = (source: string, target: string) => {\n  let targetFile: string = target;\n\n  if(existsSync(target)) {\n    if(lstatSync(target).isDirectory()) {\n      targetFile = pathJoin(target, pathBasename(source));\n    }\n  }\n\n  writeFileSync(targetFile, readFileSync(source));\n};\n\nexport const copyFolderRecursiveSync = (source: string, target: string): void => {\n  let files: string[] = [];\n\n  const targetFolder: string = pathJoin(target, pathBasename(source));\n\n  if(!existsSync(targetFolder)) {\n    mkdirSync(targetFolder);\n  }\n\n  if(lstatSync(source).isDirectory()) {\n    files = readdirSync(source);\n    files.forEach((file: string) => {\n      const curSource: string = pathJoin(source, file);\n\n      if(lstatSync(curSource).isDirectory()) {\n        copyFolderRecursiveSync(curSource, targetFolder);\n      } else {\n        copyFileSync(curSource, targetFolder);\n      }\n    });\n  }\n};\n\nexport const getPackageJson = (packagePath?: string) => {\n  const formatPath: string = packagePath || `${process.cwd()}/package.json`;\n  const packageData: string = readFileSync(formatPath).toString();\n\n  return JSON.parse(packageData);\n};\n\nexport const getFilesByExt = (ext: string, config: LexConfigType): string[] => {\n  const {sourceFullPath} = config;\n  return globSync(`**/**${ext}`, {\n    absolute: true,\n    cwd: sourceFullPath,\n    nodir: true\n  });\n};\n\nexport const removeConflictModules = (moduleList: object) => {\n  const updatedList: object = {...moduleList};\n\n  Object.keys(updatedList).forEach((moduleName: string) => {\n    const regex: RegExp = new RegExp('^(?!@types/).*?(jest|webpack).*$', 'gi');\n    if(regex.test(moduleName)) {\n      delete updatedList[moduleName];\n    }\n  });\n\n  return updatedList;\n};\n\nexport const removeFiles = (fileName: string, isRelative: boolean = false) => new Promise((resolve, reject) => {\n  const filePath: string = isRelative ? pathResolve(cwd, fileName) : fileName;\n\n  try {\n    rimrafSync(filePath);\n    return resolve(null);\n  } catch(error) {\n    return reject(error);\n  }\n});\n\nexport const removeModules = () => new Promise(async (resolve, reject) => {\n  try {\n    await removeFiles('./node_modules', true);\n    await removeFiles('./yarn.lock', true);\n    await removeFiles('./package-lock.json', true);\n\n    resolve(null);\n  } catch(error) {\n    reject(error);\n  }\n});\n\nexport const setPackageJson = (json, packagePath?: string) => {\n  if(!json) {\n    return;\n  }\n\n  const formatPath: string = packagePath || `${process.cwd()}/package.json`;\n\n  writeFileSync(formatPath, JSON.stringify(json, null, 2));\n};\n\nexport interface LinkedModuleType {\n  readonly name: string;\n  readonly path: string;\n}\n\nexport const linkedModules = (startPath?: string): LinkedModuleType[] => {\n  const workingPath: string = startPath || process.cwd();\n  let modulePath: string;\n  let prefix: string;\n\n  if(workingPath.includes('@')) {\n    prefix = `@${workingPath.split('@').pop()}`;\n    modulePath = workingPath;\n  } else {\n    modulePath = pathJoin(workingPath, 'node_modules');\n  }\n\n  const foundPaths: string[] = globSync('*', {\n    absolute: true,\n    cwd: modulePath,\n    nodir: false\n  });\n\n  return foundPaths.reduce((list: LinkedModuleType[], foundPath: string) => {\n    try {\n      const stats = lstatSync(foundPath);\n\n      if(stats.isDirectory()) {\n        const deepList: LinkedModuleType[] = linkedModules(foundPath);\n        list.push(...deepList);\n      } else if(stats.isSymbolicLink()) {\n        const moduleNames: string[] = ([prefix, pathBasename(foundPath)]).filter((item: string) => !isEmpty(item));\n        list.push({name: `${moduleNames.join('/')}`, path: foundPath});\n      }\n\n      return list;\n    } catch{\n      return list;\n    }\n  }, []);\n};\n\nexport const checkLinkedModules = () => {\n  const linked = linkedModules();\n\n  if(linked.length) {\n    const msgModule: string = linked.length > 1 ? 'Modules' : 'Module';\n    const linkedMsg: string = linked.reduce(\n      (msg: string, linkedModule: LinkedModuleType) =>\n        `${msg}\\n * ${linkedModule.name}`,\n      `Linked ${msgModule}:`\n    );\n\n    log(boxen(linkedMsg, {dimBorder: true, padding: 1}), 'warn');\n  }\n};\n\nexport const updateTemplateName = (filePath: string, replace: string, replaceCaps: string) => {\n  let data: string = readFileSync(filePath, 'utf8');\n  data = data.replace(/sample/g, replace);\n  data = data.replace(/Sample/g, replaceCaps);\n  writeFileSync(filePath, data, 'utf8');\n};"],"names":["boxen","chalk","copyFile","existsSync","lstatSync","mkdirSync","readdirSync","readFileSync","writeFileSync","sync","globSync","isEmpty","ora","basename","pathBasename","join","pathJoin","relative","pathRelative","resolve","pathResolve","rimrafSync","log","cwd","process","getFilenames","props","callback","cliName","name","quiet","type","useTypescript","nameCaps","itemTypes","includes","undefined","charAt","toUpperCase","substr","templatePath","templateExt","templateReact","createSpinner","fail","start","succeed","color","createProgressBar","percentage","width","filled","Math","round","empty","filledBar","cyan","repeat","emptyBar","gray","handleWebpackProgress","output","spinner","emoji","action","progressMatch","match","progress","parseInt","progressBar","text","generalProgressMatch","copyFiles","files","typeName","config","outputFullPath","sourceFullPath","items","map","fileName","from","to","Promise","all","reject","recursive","copyError","length","error","message","copyConfiguredFiles","copyFilesConfig","sourcePath","totalCopied","baseDir","allCopyPromises","pattern","resolvedPattern","matchingFiles","absolute","nodir","copyPromises","sourceFile","relativePath","destPath","destDir","push","copyFileSync","source","target","targetFile","isDirectory","copyFolderRecursiveSync","targetFolder","forEach","file","curSource","getPackageJson","packagePath","formatPath","packageData","toString","JSON","parse","getFilesByExt","ext","removeConflictModules","moduleList","updatedList","Object","keys","moduleName","regex","RegExp","test","removeFiles","isRelative","filePath","removeModules","setPackageJson","json","stringify","linkedModules","startPath","workingPath","modulePath","prefix","split","pop","foundPaths","reduce","list","foundPath","stats","deepList","isSymbolicLink","moduleNames","filter","item","path","checkLinkedModules","linked","msgModule","linkedMsg","msg","linkedModule","dimBorder","padding","updateTemplateName","replace","replaceCaps","data"],"mappings":"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"}