one-way-git-sync 6.0.26 → 6.0.28

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.
Files changed (2) hide show
  1. package/dist/index.js +2 -2
  2. package/package.json +33 -29
package/dist/index.js CHANGED
@@ -1,2 +1,2 @@
1
- import e from"node:fs";import t from"node:path";import{hideBin as o}from"yargs/helpers";import i from"yargs/yargs";import{pino as r}from"pino";import a from"node:child_process";import n from"node:fs/promises";import{copy as s}from"fs-extra";import c from"micromatch";import{simpleGit as d}from"simple-git";const m=r({transport:{target:"pino-pretty",options:{colorize:!0}}});const l=t.join("node_modules",".temp","sync-git-repo");const g={dest:{type:"string",alias:"d",describe:"A URL of a destination git repository.",demand:!0},prefix:{type:"string",alias:"p",describe:"A prefix of a commit hash used to generate a commit title, e.g. 'sync <prefix>/<hash>'.\n A typical value is like 'https://github.com/WillBooster/one-way-git-sync/commits'"},branch:{type:"string",alias:"b",describe:"Specify branch of destination repo."},tag:{type:"string",alias:"t",describe:"Specify tag to be created in destination repo."},"tag-hash":{type:"boolean",describe:"Create version+hash tag (e.g. v1.31.5-2-gcdde507). It should be a unique tag."},"tag-version":{type:"boolean",describe:"Create version tag (e.g. v1.31.5). It could be a non-unique tag."},"ignore-patterns":{type:"array",alias:"i",describe:"Exclude the files whose path matches one of the given patterns.\n The patterns are processed by micromatch (https://github.com/micromatch/micromatch).\n You may specify the option multiple times. Default value is \"-i .git -i .github -i node_modules -i '.renovaterc.*' -i '.renovate.*'\"",default:[".git",".github","node_modules",".renovaterc.*","renovate.*"]},dry:{type:"boolean",describe:"Enable dry-run mode."},force:{type:"boolean",describe:"Force to overwrite the destination git repository."},verbose:{type:"boolean",alias:"v",describe:"Show details logs."}},p=await i(o(process.argv)).scriptName("one-way-git-sync").options(g).middleware(e=>{m.level=e.verbose?"trace":"info"}).strict().version(function(){let o=t.dirname(new URL(import.meta.url).pathname);for(;!e.existsSync(t.join(o,"package.json"));)o=t.dirname(o);return JSON.parse(e.readFileSync(t.join(o,"package.json"),"utf8")).version}()).help().argv;await async function(e){await n.mkdir(l,{recursive:!0});const o=await n.mkdtemp(t.join(l,"repo-")),i=await async function(e,o,i=process.cwd()){const r={"--single-branch":null};o.force||(r["--depth"]=1);o.branch&&(r["--branch"]=o.branch);try{await d(i).clone(o.dest,e,r)}catch{delete r["--branch"],delete r["--single-branch"],await d(i).clone(o.dest,e,r),d(e).checkout(["-b",o.branch])}m.debug(`Cloned destination repo on ${e}`);const l=d(e);let g=[];try{g=function(e){if(0===e.all.length)return m.debug("No commit history"),[];for(const t of e.all){const[e,...o]=t.message.replaceAll(/[()]/g,"").split(/[\s/]/);if("sync"===e&&o.length>0)return[t.message,o.at(-1)]}return m.debug(`No sync commit: ${e.all[0].message}`),[]}(await l.log())}catch{}const[p,h]=g;if(h)m.debug(`Extracted a valid commit: ${h}`),m.debug(`(${p})`);else if(!o.force)return m.error("No valid commit in destination repo"),!1;const u=d(i);let f;try{f=await u.log(h?{from:h,to:"HEAD","--first-parent":void 0}:void 0)}catch(e){return m.error(`Failed to get source commit history: ${e.stack}`),!1}const b=f.latest?.hash;if(!b)return m.info("No synchronizable commit"),!0;const y=[...new Set([...o["ignore-patterns"].map(String),".git"])];let[w,v]=await Promise.all([n.readdir(e),n.readdir(i)]);w=c.not(w,y),v=c.not(v,y),m.debug("destFiles: %o",w),m.debug("srcFiles: %o",v);for(const o of w)await n.rm(t.join(e,o),{recursive:!0,force:!0});for(const o of v)await s(t.join(i,o),t.join(e,o));await l.add("-A");let $="";if(o["tag-hash"]||o["tag-version"]){const e="git describe --tags --always "+(o["tag-version"]?"--abbrev=0":"");$=a.execSync(e,{cwd:i}).toString().trim()}let j=o.prefix??await async function(e){const t=(await e.getRemotes(!0)).find(e=>"origin"===e.name),o=t?.refs?.fetch??t?.refs?.push;if("string"==typeof o&&o.includes("github.com")){const e=o.split("/"),t=e.at(-2),i=e.at(-1)?.replace(/.git$/,"");if(t&&i)return`https://github.com/${t}/${i}/commits`}}(u)??"";j&&!j.endsWith("/")&&(j+="/");const x=`${j}${b}`,S=$?`sync ${$} (${x})`:`sync ${x}`,k=h?f.all.map(e=>`* ${e.message}`).join("\n\n"):`Replace all the files with those of ${o.dest} due to missing sync commit.`;try{const e=await l.commit(`${S}\n\n${k}`);m.debug("Created a commit: %o",e),m.debug(S),m.debug(` with body: ${k}`)}catch(e){return m.error(`Failed to commit changes: ${e.stack}`),!1}const F=$||o.tag;if(F)try{await l.addTag(F),m.debug(`Created a tag: ${F}`)}catch{m.warn(`Failed to create a tag: ${F}`)}if(o.dry)return m.debug("Finished dry run"),!0;try{await(o.branch?l.push("origin",o.branch):l.push()),F&&await l.push({"--tags":null})}catch(e){return m.error(`Failed to push the commit: ${e.stack}`),!1}return m.debug("Pushed the commit"),!0}(o,e);process.exit(i?0:1)}(p);
2
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","sources":["../src/logger.ts","../src/sync.ts","../src/yargsOptions.ts","../src/index.ts","../src/gitHub.ts"],"sourcesContent":["import { pino } from 'pino';\n\nexport const logger = pino({\n  transport: {\n    target: 'pino-pretty',\n    options: {\n      colorize: true,\n    },\n  },\n});\n","import child_process from 'node:child_process';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\n\nimport { copy } from 'fs-extra';\nimport micromatch from 'micromatch';\nimport type { LogResult, Options, SimpleGit, TaskOptions } from 'simple-git';\nimport { simpleGit } from 'simple-git';\nimport type { InferredOptionTypes } from 'yargs';\n\nimport { getGitHubCommitsUrl } from './gitHub.js';\nimport { logger } from './logger.js';\nimport type { yargsOptions } from './yargsOptions.js';\n\nconst syncDirPath = path.join('node_modules', '.temp', 'sync-git-repo');\n\nexport type YargsOptions = InferredOptionTypes<typeof yargsOptions>;\n\nexport async function sync(opts: YargsOptions): Promise<void> {\n  await fs.mkdir(syncDirPath, { recursive: true });\n  const dirPath = await fs.mkdtemp(path.join(syncDirPath, 'repo-'));\n  const ret = await syncCore(dirPath, opts);\n  // await fs.rm(dirPath, { recursive: true, force: true });\n  process.exit(ret ? 0 : 1);\n}\n\nexport async function syncCore(\n  destRepoPath: string,\n  opts: YargsOptions,\n  srcRepoPath = process.cwd()\n): Promise<boolean> {\n  // eslint-disable-next-line unicorn/no-null\n  const cloneOpts: Options = { '--single-branch': null };\n  if (!opts.force) {\n    cloneOpts['--depth'] = 1;\n  }\n  if (opts.branch) {\n    cloneOpts['--branch'] = opts.branch;\n  }\n  try {\n    await simpleGit(srcRepoPath).clone(opts.dest, destRepoPath, cloneOpts);\n  } catch {\n    delete cloneOpts['--branch'];\n    delete cloneOpts['--single-branch'];\n    await simpleGit(srcRepoPath).clone(opts.dest, destRepoPath, cloneOpts);\n    simpleGit(destRepoPath).checkout(['-b', opts.branch] as TaskOptions);\n  }\n  logger.debug(`Cloned destination repo on ${destRepoPath}`);\n\n  const dstGit: SimpleGit = simpleGit(destRepoPath);\n  let commitHashResult: [string, string] | [] = [];\n  try {\n    const dstLog = await dstGit.log();\n    commitHashResult = extractCommitHash(dstLog);\n  } catch {\n    // do nothing\n  }\n  const [head, from] = commitHashResult;\n  if (from) {\n    logger.debug(`Extracted a valid commit: ${from}`);\n    logger.debug(`(${head})`);\n  } else if (!opts.force) {\n    logger.error('No valid commit in destination repo');\n    return false;\n  }\n\n  const srcGit: SimpleGit = simpleGit(srcRepoPath);\n  let srcLog: LogResult;\n  try {\n    // '--first-parent' hides children commits of merge commits\n    srcLog = await srcGit.log(from ? { from, to: 'HEAD', '--first-parent': undefined } : undefined);\n  } catch (error) {\n    logger.error(`Failed to get source commit history: ${(error as Error).stack}`);\n    return false;\n  }\n\n  const latestHash = srcLog.latest?.hash;\n  if (!latestHash) {\n    logger.info('No synchronizable commit');\n    return true;\n  }\n\n  // Force to ignore .git directory\n  const ignorePatterns = [...new Set([...opts['ignore-patterns'].map(String), '.git'])];\n  let [destFiles, srcFiles] = await Promise.all([fs.readdir(destRepoPath), fs.readdir(srcRepoPath)]);\n  destFiles = micromatch.not(destFiles, ignorePatterns);\n  srcFiles = micromatch.not(srcFiles, ignorePatterns);\n  logger.debug('destFiles: %o', destFiles);\n  logger.debug('srcFiles: %o', srcFiles);\n  for (const destFile of destFiles) {\n    await fs.rm(path.join(destRepoPath, destFile), { recursive: true, force: true });\n  }\n  for (const srcFile of srcFiles) {\n    await copy(path.join(srcRepoPath, srcFile), path.join(destRepoPath, srcFile));\n  }\n  await dstGit.add('-A');\n\n  let srcTag = '';\n  if (opts['tag-hash'] || opts['tag-version']) {\n    // e.g. `--abbrev=0` changes `v1.31.5-2-gcdde507` to `v1.31.5`\n    const describeCommand = `git describe --tags --always ${opts['tag-version'] ? '--abbrev=0' : ''}`;\n    srcTag = child_process.execSync(describeCommand, { cwd: srcRepoPath }).toString().trim();\n  }\n  let prefix = opts.prefix ?? (await getGitHubCommitsUrl(srcGit)) ?? '';\n  if (prefix && !prefix.endsWith('/')) {\n    prefix += '/';\n  }\n  const link = `${prefix}${latestHash}`;\n  const title = srcTag ? `sync ${srcTag} (${link})` : `sync ${link}`;\n  const body = from\n    ? srcLog.all.map((l) => `* ${l.message}`).join('\\n\\n')\n    : `Replace all the files with those of ${opts.dest} due to missing sync commit.`;\n  try {\n    const ret = await dstGit.commit(`${title}\\n\\n${body}`);\n    logger.debug(`Created a commit: %o`, ret);\n    logger.debug(title);\n    logger.debug(`  with body: ${body}`);\n  } catch (error) {\n    logger.error(`Failed to commit changes: ${(error as Error).stack}`);\n    return false;\n  }\n\n  const destTag = srcTag || opts.tag;\n  if (destTag) {\n    try {\n      await dstGit.addTag(destTag);\n      logger.debug(`Created a tag: ${destTag}`);\n    } catch {\n      // Ignore the error since `--abbrev=0` may yield a tag that already exists\n      logger.warn(`Failed to create a tag: ${destTag}`);\n    }\n  }\n\n  if (opts.dry) {\n    logger.debug('Finished dry run');\n    return true;\n  }\n\n  try {\n    await (opts.branch ? dstGit.push('origin', opts.branch) : dstGit.push());\n    if (destTag) {\n      // eslint-disable-next-line unicorn/no-null\n      await dstGit.push({ '--tags': null });\n    }\n  } catch (error) {\n    logger.error(`Failed to push the commit: ${(error as Error).stack}`);\n    return false;\n  }\n\n  logger.debug('Pushed the commit');\n  return true;\n}\n\nfunction extractCommitHash(logResult: LogResult): [string, string] | [] {\n  if (logResult.all.length === 0) {\n    logger.debug('No commit history');\n    return [];\n  }\n\n  for (const log of logResult.all) {\n    const [head, ...words] = log.message.replaceAll(/[()]/g, '').split(/[\\s/]/);\n    if (head === 'sync' && words.length > 0) {\n      return [log.message, words.at(-1) as string];\n    }\n  }\n  logger.debug(`No sync commit: ${logResult.all[0].message}`);\n  return [];\n}\n","export const yargsOptions = {\n  dest: {\n    type: 'string',\n    alias: 'd',\n    describe: 'A URL of a destination git repository.',\n    demand: true,\n  },\n  prefix: {\n    type: 'string',\n    alias: 'p',\n    describe: `A prefix of a commit hash used to generate a commit title, e.g. 'sync <prefix>/<hash>'.\n               A typical value is like 'https://github.com/WillBooster/one-way-git-sync/commits'`,\n  },\n  branch: {\n    type: 'string',\n    alias: 'b',\n    describe: 'Specify branch of destination repo.',\n  },\n  tag: {\n    type: 'string',\n    alias: 't',\n    describe: 'Specify tag to be created in destination repo.',\n  },\n  'tag-hash': {\n    type: 'boolean',\n    describe: 'Create version+hash tag (e.g. v1.31.5-2-gcdde507). It should be a unique tag.',\n  },\n  'tag-version': {\n    type: 'boolean',\n    describe: 'Create version tag (e.g. v1.31.5). It could be a non-unique tag.',\n  },\n  'ignore-patterns': {\n    type: 'array',\n    alias: 'i',\n    describe: `Exclude the files whose path matches one of the given patterns.\n               The patterns are processed by micromatch (https://github.com/micromatch/micromatch).\n               You may specify the option multiple times. Default value is \"-i .git -i .github -i node_modules -i '.renovaterc.*' -i '.renovate.*'\"`,\n    default: ['.git', '.github', 'node_modules', '.renovaterc.*', 'renovate.*'],\n  },\n  dry: {\n    type: 'boolean',\n    describe: 'Enable dry-run mode.',\n  },\n  force: {\n    type: 'boolean',\n    describe: 'Force to overwrite the destination git repository.',\n  },\n  verbose: {\n    type: 'boolean',\n    alias: 'v',\n    describe: 'Show details logs.',\n  },\n} as const;\n","import fs from 'node:fs';\nimport path from 'node:path';\n\nimport { hideBin } from 'yargs/helpers';\nimport yargs from 'yargs/yargs';\n\nimport { logger } from './logger.js';\nimport { sync } from './sync.js';\nimport { yargsOptions } from './yargsOptions.js';\n\nconst argv = await yargs(hideBin(process.argv))\n  .scriptName('one-way-git-sync')\n  .options(yargsOptions)\n  .middleware((argv) => {\n    logger.level = argv.verbose ? 'trace' : 'info';\n  })\n  .strict()\n  .version(getVersion())\n  .help().argv;\n\nfunction getVersion(): string {\n  let packageJsonDir = path.dirname(new URL(import.meta.url).pathname);\n  while (!fs.existsSync(path.join(packageJsonDir, 'package.json'))) {\n    packageJsonDir = path.dirname(packageJsonDir);\n  }\n  const packageJson = JSON.parse(fs.readFileSync(path.join(packageJsonDir, 'package.json'), 'utf8')) as {\n    version: string;\n  };\n  return packageJson.version;\n}\n\nawait sync(argv);\n","import type { SimpleGit } from 'simple-git';\n\nexport async function getGitHubCommitsUrl(git: SimpleGit): Promise<string | undefined> {\n  const remotes = await git.getRemotes(true);\n  const origin = remotes.find((r) => r.name === 'origin');\n  const remoteUrl = origin?.refs?.fetch ?? origin?.refs?.push;\n  if (typeof remoteUrl === 'string' && remoteUrl.includes('github.com')) {\n    const words = remoteUrl.split('/');\n    const org = words.at(-2);\n    const name = words.at(-1)?.replace(/.git$/, '');\n    if (org && name) return `https://github.com/${org}/${name}/commits`;\n  }\n}\n"],"names":["logger","pino","transport","target","options","colorize","syncDirPath","path","join","yargsOptions","dest","type","alias","describe","demand","prefix","branch","tag","default","dry","force","verbose","argv","yargs","hideBin","process","scriptName","middleware","level","strict","version","packageJsonDir","dirname","URL","url","pathname","fs","existsSync","JSON","parse","readFileSync","getVersion","help","async","opts","mkdir","recursive","dirPath","mkdtemp","ret","destRepoPath","srcRepoPath","cwd","cloneOpts","simpleGit","clone","checkout","debug","dstGit","commitHashResult","logResult","all","length","log","head","words","message","replaceAll","split","at","extractCommitHash","from","error","srcGit","srcLog","to","undefined","stack","latestHash","latest","hash","info","ignorePatterns","Set","map","String","destFiles","srcFiles","Promise","readdir","micromatch","not","destFile","rm","srcFile","copy","add","srcTag","describeCommand","child_process","execSync","toString","trim","git","origin","getRemotes","find","r","name","remoteUrl","refs","fetch","push","includes","org","replace","getGitHubCommitsUrl","endsWith","link","title","body","l","commit","destTag","addTag","warn","syncCore","exit","sync"],"mappings":"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"}
1
+ import e from"node:fs";import t from"node:path";import{hideBin as i}from"yargs/helpers";import o from"yargs/yargs";import{pino as r}from"pino";import a from"node:child_process";import n from"node:fs/promises";import{copy as s}from"fs-extra";import c from"micromatch";import{simpleGit as d}from"simple-git";const m=r({transport:{target:"pino-pretty",options:{colorize:!0}}});const l=t.join("node_modules",".temp","sync-git-repo");const g={dest:{type:"string",alias:"d",describe:"A URL of a destination git repository.",demand:!0},prefix:{type:"string",alias:"p",describe:"A prefix of a commit hash used to generate a commit title, e.g. 'sync <prefix>/<hash>'.\n A typical value is like 'https://github.com/WillBooster/one-way-git-sync/commits'"},branch:{type:"string",alias:"b",describe:"Specify branch of destination repo."},tag:{type:"string",alias:"t",describe:"Specify tag to be created in destination repo."},"tag-hash":{type:"boolean",describe:"Create version+hash tag (e.g. v1.31.5-2-gcdde507). It should be a unique tag."},"tag-version":{type:"boolean",describe:"Create version tag (e.g. v1.31.5). It could be a non-unique tag."},"ignore-patterns":{type:"array",alias:"i",describe:"Exclude the files whose path matches one of the given patterns.\n The patterns are processed by micromatch (https://github.com/micromatch/micromatch).\n You may specify the option multiple times. Default value is \"-i .git -i .github -i node_modules -i '.renovaterc.*' -i '.renovate.*'\"",default:[".git",".github","node_modules",".renovaterc.*","renovate.*"]},dry:{type:"boolean",describe:"Enable dry-run mode."},force:{type:"boolean",describe:"Force to overwrite the destination git repository."},verbose:{type:"boolean",alias:"v",describe:"Show details logs."}},p=await o(i(process.argv)).scriptName("one-way-git-sync").options(g).middleware(e=>{m.level=e.verbose?"trace":"info"}).strict().version(function(){let i=t.dirname(new URL(import.meta.url).pathname);for(;!e.existsSync(t.join(i,"package.json"));)i=t.dirname(i);return JSON.parse(e.readFileSync(t.join(i,"package.json"),"utf8")).version}()).help().argv;await async function(e){await n.mkdir(l,{recursive:!0});const i=await n.mkdtemp(t.join(l,"repo-")),o=await async function(e,i,o=process.cwd()){const r={"--single-branch":null};i.force||(r["--depth"]=1);i.branch&&(r["--branch"]=i.branch);try{await d(o).clone(i.dest,e,r)}catch{delete r["--branch"],delete r["--single-branch"],await d(o).clone(i.dest,e,r),await d(e).checkout(["-b",i.branch])}m.debug(`Cloned destination repo on ${e}`);const l=d(e);let g=[];try{g=function(e){if(0===e.all.length)return m.debug("No commit history"),[];for(const t of e.all){const[e,...i]=t.message.replaceAll(/[()]/g,"").split(/[\s/]/);if("sync"===e&&i.length>0)return[t.message,i.at(-1)]}const[t]=e.all;t&&m.debug(`No sync commit: ${t.message}`);return[]}(await l.log())}catch{}const[p,h]=g;if(h)m.debug(`Extracted a valid commit: ${h}`),m.debug(`(${p})`);else if(!i.force)return m.error("No valid commit in destination repo"),!1;const u=d(o);let f;try{f=await u.log(h?{from:h,to:"HEAD","--first-parent":void 0}:void 0)}catch(e){return m.error(`Failed to get source commit history: ${e.stack}`),!1}const b=f.latest?.hash;if(!b)return m.info("No synchronizable commit"),!0;const y=[...new Set([...i["ignore-patterns"].map(String),".git"])];let[w,v]=await Promise.all([n.readdir(e),n.readdir(o)]);w=c.not(w,y),v=c.not(v,y),m.debug("destFiles: %o",w),m.debug("srcFiles: %o",v);for(const i of w)await n.rm(t.join(e,i),{recursive:!0,force:!0});for(const i of v)await s(t.join(o,i),t.join(e,i));await l.add("-A");let $="";if(i["tag-hash"]||i["tag-version"]){const e="git describe --tags --always "+(i["tag-version"]?"--abbrev=0":"");$=a.execSync(e,{cwd:o}).toString().trim()}let j=i.prefix??await async function(e){const t=(await e.getRemotes(!0)).find(e=>"origin"===e.name);if(!t)return;let i=t.refs.fetch;if(i||(i=t.refs.push),"string"==typeof i&&i.includes("github.com")){const e=i.split("/"),t=e.at(-2),o=e.at(-1)?.replace(/.git$/,"");if(t&&o)return`https://github.com/${t}/${o}/commits`}}(u)??"";j&&!j.endsWith("/")&&(j+="/");const x=`${j}${b}`,S=$?`sync ${$} (${x})`:`sync ${x}`,k=h?f.all.map(e=>`* ${e.message}`).join("\n\n"):`Replace all the files with those of ${i.dest} due to missing sync commit.`;try{const e=await l.commit(`${S}\n\n${k}`);m.debug("Created a commit: %o",e),m.debug(S),m.debug(` with body: ${k}`)}catch(e){return m.error(`Failed to commit changes: ${e.stack}`),!1}const F=$||i.tag;if(F)try{await l.addTag(F),m.debug(`Created a tag: ${F}`)}catch{m.warn(`Failed to create a tag: ${F}`)}if(i.dry)return m.debug("Finished dry run"),!0;try{await(i.branch?l.push("origin",i.branch):l.push()),F&&await l.push({"--tags":null})}catch(e){return m.error(`Failed to push the commit: ${e.stack}`),!1}return m.debug("Pushed the commit"),!0}(i,e);process.exit(o?0:1)}(p);
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","sources":["../src/logger.ts","../src/sync.ts","../src/yargsOptions.ts","../src/index.ts","../src/gitHub.ts"],"sourcesContent":["import { pino } from 'pino';\n\nexport const logger = pino({\n  transport: {\n    target: 'pino-pretty',\n    options: {\n      colorize: true,\n    },\n  },\n});\n","import child_process from 'node:child_process';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\n\nimport { copy } from 'fs-extra';\nimport micromatch from 'micromatch';\nimport type { LogResult, Options, SimpleGit, TaskOptions } from 'simple-git';\nimport { simpleGit } from 'simple-git';\nimport type { InferredOptionTypes } from 'yargs';\n\nimport { getGitHubCommitsUrl } from './gitHub.js';\nimport { logger } from './logger.js';\nimport type { yargsOptions } from './yargsOptions.js';\n\nconst syncDirPath = path.join('node_modules', '.temp', 'sync-git-repo');\n\nexport type YargsOptions = InferredOptionTypes<typeof yargsOptions>;\n\nexport async function sync(opts: YargsOptions): Promise<void> {\n  await fs.mkdir(syncDirPath, { recursive: true });\n  const dirPath = await fs.mkdtemp(path.join(syncDirPath, 'repo-'));\n  const ret = await syncCore(dirPath, opts);\n  // await fs.rm(dirPath, { recursive: true, force: true });\n  process.exit(ret ? 0 : 1);\n}\n\nexport async function syncCore(\n  destRepoPath: string,\n  opts: YargsOptions,\n  srcRepoPath = process.cwd()\n): Promise<boolean> {\n  // eslint-disable-next-line unicorn/no-null\n  const cloneOpts: Options = { '--single-branch': null };\n  if (!opts.force) {\n    cloneOpts['--depth'] = 1;\n  }\n  if (opts.branch) {\n    cloneOpts['--branch'] = opts.branch;\n  }\n  try {\n    await simpleGit(srcRepoPath).clone(opts.dest, destRepoPath, cloneOpts);\n  } catch {\n    delete cloneOpts['--branch'];\n    delete cloneOpts['--single-branch'];\n    await simpleGit(srcRepoPath).clone(opts.dest, destRepoPath, cloneOpts);\n    await simpleGit(destRepoPath).checkout(['-b', opts.branch] as TaskOptions);\n  }\n  logger.debug(`Cloned destination repo on ${destRepoPath}`);\n\n  const dstGit: SimpleGit = simpleGit(destRepoPath);\n  let commitHashResult: [string, string] | [] = [];\n  try {\n    const dstLog = await dstGit.log();\n    commitHashResult = extractCommitHash(dstLog);\n  } catch {\n    // do nothing\n  }\n  const [head, from] = commitHashResult;\n  if (from) {\n    logger.debug(`Extracted a valid commit: ${from}`);\n    logger.debug(`(${head})`);\n  } else if (!opts.force) {\n    logger.error('No valid commit in destination repo');\n    return false;\n  }\n\n  const srcGit: SimpleGit = simpleGit(srcRepoPath);\n  let srcLog: LogResult;\n  try {\n    // '--first-parent' hides children commits of merge commits\n    srcLog = await srcGit.log(from ? { from, to: 'HEAD', '--first-parent': undefined } : undefined);\n  } catch (error) {\n    logger.error(`Failed to get source commit history: ${(error as Error).stack}`);\n    return false;\n  }\n\n  const latestHash = srcLog.latest?.hash;\n  if (!latestHash) {\n    logger.info('No synchronizable commit');\n    return true;\n  }\n\n  // Force to ignore .git directory\n  const ignorePatterns = [...new Set([...opts['ignore-patterns'].map(String), '.git'])];\n  let [destFiles, srcFiles] = await Promise.all([fs.readdir(destRepoPath), fs.readdir(srcRepoPath)]);\n  destFiles = micromatch.not(destFiles, ignorePatterns);\n  srcFiles = micromatch.not(srcFiles, ignorePatterns);\n  logger.debug('destFiles: %o', destFiles);\n  logger.debug('srcFiles: %o', srcFiles);\n  for (const destFile of destFiles) {\n    await fs.rm(path.join(destRepoPath, destFile), { recursive: true, force: true });\n  }\n  for (const srcFile of srcFiles) {\n    await copy(path.join(srcRepoPath, srcFile), path.join(destRepoPath, srcFile));\n  }\n  await dstGit.add('-A');\n\n  let srcTag = '';\n  if (opts['tag-hash'] || opts['tag-version']) {\n    // e.g. `--abbrev=0` changes `v1.31.5-2-gcdde507` to `v1.31.5`\n    const describeCommand = `git describe --tags --always ${opts['tag-version'] ? '--abbrev=0' : ''}`;\n    srcTag = child_process.execSync(describeCommand, { cwd: srcRepoPath }).toString().trim();\n  }\n  let prefix = opts.prefix ?? (await getGitHubCommitsUrl(srcGit)) ?? '';\n  if (prefix && !prefix.endsWith('/')) {\n    prefix += '/';\n  }\n  const link = `${prefix}${latestHash}`;\n  const title = srcTag ? `sync ${srcTag} (${link})` : `sync ${link}`;\n  const body = from\n    ? srcLog.all.map((l) => `* ${l.message}`).join('\\n\\n')\n    : `Replace all the files with those of ${opts.dest} due to missing sync commit.`;\n  try {\n    const ret = await dstGit.commit(`${title}\\n\\n${body}`);\n    logger.debug(`Created a commit: %o`, ret);\n    logger.debug(title);\n    logger.debug(`  with body: ${body}`);\n  } catch (error) {\n    logger.error(`Failed to commit changes: ${(error as Error).stack}`);\n    return false;\n  }\n\n  const destTag = srcTag || opts.tag;\n  if (destTag) {\n    try {\n      await dstGit.addTag(destTag);\n      logger.debug(`Created a tag: ${destTag}`);\n    } catch {\n      // Ignore the error since `--abbrev=0` may yield a tag that already exists\n      logger.warn(`Failed to create a tag: ${destTag}`);\n    }\n  }\n\n  if (opts.dry) {\n    logger.debug('Finished dry run');\n    return true;\n  }\n\n  try {\n    await (opts.branch ? dstGit.push('origin', opts.branch) : dstGit.push());\n    if (destTag) {\n      // eslint-disable-next-line unicorn/no-null\n      await dstGit.push({ '--tags': null });\n    }\n  } catch (error) {\n    logger.error(`Failed to push the commit: ${(error as Error).stack}`);\n    return false;\n  }\n\n  logger.debug('Pushed the commit');\n  return true;\n}\n\nfunction extractCommitHash(logResult: LogResult): [string, string] | [] {\n  if (logResult.all.length === 0) {\n    logger.debug('No commit history');\n    return [];\n  }\n\n  for (const log of logResult.all) {\n    const [head, ...words] = log.message.replaceAll(/[()]/g, '').split(/[\\s/]/);\n    if (head === 'sync' && words.length > 0) {\n      return [log.message, words.at(-1) as string];\n    }\n  }\n  const [firstLog] = logResult.all;\n  if (firstLog) {\n    logger.debug(`No sync commit: ${firstLog.message}`);\n  }\n  return [];\n}\n","export const yargsOptions = {\n  dest: {\n    type: 'string',\n    alias: 'd',\n    describe: 'A URL of a destination git repository.',\n    demand: true,\n  },\n  prefix: {\n    type: 'string',\n    alias: 'p',\n    describe: `A prefix of a commit hash used to generate a commit title, e.g. 'sync <prefix>/<hash>'.\n               A typical value is like 'https://github.com/WillBooster/one-way-git-sync/commits'`,\n  },\n  branch: {\n    type: 'string',\n    alias: 'b',\n    describe: 'Specify branch of destination repo.',\n  },\n  tag: {\n    type: 'string',\n    alias: 't',\n    describe: 'Specify tag to be created in destination repo.',\n  },\n  'tag-hash': {\n    type: 'boolean',\n    describe: 'Create version+hash tag (e.g. v1.31.5-2-gcdde507). It should be a unique tag.',\n  },\n  'tag-version': {\n    type: 'boolean',\n    describe: 'Create version tag (e.g. v1.31.5). It could be a non-unique tag.',\n  },\n  'ignore-patterns': {\n    type: 'array',\n    alias: 'i',\n    describe: `Exclude the files whose path matches one of the given patterns.\n               The patterns are processed by micromatch (https://github.com/micromatch/micromatch).\n               You may specify the option multiple times. Default value is \"-i .git -i .github -i node_modules -i '.renovaterc.*' -i '.renovate.*'\"`,\n    default: ['.git', '.github', 'node_modules', '.renovaterc.*', 'renovate.*'],\n  },\n  dry: {\n    type: 'boolean',\n    describe: 'Enable dry-run mode.',\n  },\n  force: {\n    type: 'boolean',\n    describe: 'Force to overwrite the destination git repository.',\n  },\n  verbose: {\n    type: 'boolean',\n    alias: 'v',\n    describe: 'Show details logs.',\n  },\n} as const;\n","import fs from 'node:fs';\nimport path from 'node:path';\n\nimport { hideBin } from 'yargs/helpers';\nimport yargs from 'yargs/yargs';\n\nimport { logger } from './logger.js';\nimport { sync } from './sync.js';\nimport { yargsOptions } from './yargsOptions.js';\n\nconst argv = await yargs(hideBin(process.argv))\n  .scriptName('one-way-git-sync')\n  .options(yargsOptions)\n  .middleware((argv) => {\n    logger.level = argv.verbose ? 'trace' : 'info';\n  })\n  .strict()\n  .version(getVersion())\n  .help().argv;\n\nfunction getVersion(): string {\n  let packageJsonDir = path.dirname(new URL(import.meta.url).pathname);\n  while (!fs.existsSync(path.join(packageJsonDir, 'package.json'))) {\n    packageJsonDir = path.dirname(packageJsonDir);\n  }\n  const packageJson = JSON.parse(fs.readFileSync(path.join(packageJsonDir, 'package.json'), 'utf8')) as {\n    version: string;\n  };\n  return packageJson.version;\n}\n\nawait sync(argv);\n","import type { SimpleGit } from 'simple-git';\n\nexport async function getGitHubCommitsUrl(git: SimpleGit): Promise<string | undefined> {\n  const remotes = await git.getRemotes(true);\n  const origin = remotes.find((r) => r.name === 'origin');\n  if (!origin) {\n    return;\n  }\n  let remoteUrl: string | undefined = origin.refs.fetch;\n  if (!remoteUrl) {\n    remoteUrl = origin.refs.push;\n  }\n  if (typeof remoteUrl === 'string' && remoteUrl.includes('github.com')) {\n    const words = remoteUrl.split('/');\n    const org = words.at(-2);\n    const name = words.at(-1)?.replace(/.git$/, '');\n    if (org && name) return `https://github.com/${org}/${name}/commits`;\n  }\n}\n"],"names":["logger","pino","transport","target","options","colorize","syncDirPath","path","join","yargsOptions","dest","type","alias","describe","demand","prefix","branch","tag","default","dry","force","verbose","argv","yargs","hideBin","process","scriptName","middleware","level","strict","version","packageJsonDir","dirname","URL","url","pathname","fs","existsSync","JSON","parse","readFileSync","getVersion","help","async","opts","mkdir","recursive","dirPath","mkdtemp","ret","destRepoPath","srcRepoPath","cwd","cloneOpts","simpleGit","clone","checkout","debug","dstGit","commitHashResult","logResult","all","length","log","head","words","message","replaceAll","split","at","firstLog","extractCommitHash","from","error","srcGit","srcLog","to","undefined","stack","latestHash","latest","hash","info","ignorePatterns","Set","map","String","destFiles","srcFiles","Promise","readdir","micromatch","not","destFile","rm","srcFile","copy","add","srcTag","describeCommand","child_process","execSync","toString","trim","git","origin","getRemotes","find","r","name","remoteUrl","refs","fetch","push","includes","org","replace","getGitHubCommitsUrl","endsWith","link","title","body","l","commit","destTag","addTag","warn","syncCore","exit","sync"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "one-way-git-sync",
3
- "version": "6.0.26",
3
+ "version": "6.0.28",
4
4
  "description": "A tool for synchronizing a destination git repository with a source git repository SAFELY.",
5
5
  "keywords": [
6
6
  "git",
@@ -17,15 +17,17 @@
17
17
  ],
18
18
  "scripts": {
19
19
  "build": "build-ts app",
20
+ "check-all-for-ai": "yarn check-for-ai && yarn test",
21
+ "check-for-ai": "yarn install > /dev/null && yarn format > /dev/null 2> /dev/null || true && yarn lint-fix --quiet && yarn typecheck",
20
22
  "cleanup": "yarn format && yarn lint-fix",
21
23
  "format": "sort-package-json && yarn prettify",
22
24
  "_postinstall": "husky || true",
23
- "lint": "eslint --color \"./{scripts,src,tests}/**/*.{cjs,cts,js,jsx,mjs,mts,ts,tsx}\"",
25
+ "lint": "eslint --color",
24
26
  "lint-fix": "yarn lint --fix",
25
27
  "prepack": "pinst --disable",
26
28
  "postpack": "pinst --enable",
27
29
  "prepare": "husky || true",
28
- "prettify": "prettier --cache --color --write \"**/{.*/,}*.{cjs,css,cts,htm,html,js,json,json5,jsonc,jsx,md,mjs,mts,scss,ts,tsx,vue,yaml,yml}\" \"!**/test-fixtures/**\"",
30
+ "prettify": "prettier --cache --color --write \"**/{.*/,}*.{cjs,css,cts,htm,html,java,js,json,json5,jsonc,jsx,md,mjs,mts,scss,ts,tsx,vue,yaml,yml}\" \"!**/test{-,/}fixtures/**\" || true",
29
31
  "release": "yarn build && semantic-release",
30
32
  "start": "build-ts run src/index.ts --",
31
33
  "start-prod": "yarn build && yarn one-way-git-sync",
@@ -34,44 +36,46 @@
34
36
  },
35
37
  "prettier": "@willbooster/prettier-config",
36
38
  "dependencies": {
37
- "fs-extra": "11.3.1",
39
+ "fs-extra": "11.3.2",
38
40
  "micromatch": "4.0.8",
39
- "pino": "9.9.0",
40
- "pino-pretty": "13.1.1",
41
- "simple-git": "3.28.0",
42
- "yargs": "17.7.2"
41
+ "pino": "10.1.0",
42
+ "pino-pretty": "13.1.2",
43
+ "simple-git": "3.30.0",
44
+ "yargs": "18.0.0"
43
45
  },
44
46
  "devDependencies": {
45
- "@types/eslint": "8.56.11",
47
+ "@types/eslint": "9.6.1",
46
48
  "@types/fs-extra": "11.0.4",
47
- "@types/micromatch": "4.0.9",
48
- "@types/semantic-release": "20.0.6",
49
- "@types/yargs": "17.0.33",
50
- "@typescript-eslint/eslint-plugin": "8.39.1",
51
- "@typescript-eslint/parser": "8.39.1",
52
- "@willbooster/eslint-config-ts": "10.6.1",
53
- "@willbooster/prettier-config": "9.1.3",
54
- "build-ts": "13.2.3",
55
- "conventional-changelog-conventionalcommits": "8.0.0",
56
- "eslint": "8.57.0",
49
+ "@types/micromatch": "4.0.10",
50
+ "@types/yargs": "17.0.35",
51
+ "@willbooster/eslint-config-ts": "11.4.10",
52
+ "@willbooster/prettier-config": "10.2.1",
53
+ "build-ts": "17.0.2",
54
+ "conventional-changelog-conventionalcommits": "9.1.0",
55
+ "eslint": "9.39.1",
56
+ "eslint-config-flat-gitignore": "2.1.0",
57
57
  "eslint-config-prettier": "10.1.8",
58
- "eslint-import-resolver-typescript": "3.10.1",
59
- "eslint-plugin-import": "2.32.0",
58
+ "eslint-import-resolver-typescript": "4.4.4",
59
+ "eslint-plugin-import-x": "4.16.1",
60
60
  "eslint-plugin-sort-class-members": "1.21.0",
61
61
  "eslint-plugin-sort-destructure-keys": "2.0.0",
62
- "eslint-plugin-unicorn": "56.0.1",
62
+ "eslint-plugin-unicorn": "62.0.0",
63
+ "eslint-plugin-unused-imports": "4.3.0",
64
+ "globals": "16.5.0",
63
65
  "husky": "9.1.7",
64
- "lint-staged": "15.5.2",
66
+ "lint-staged": "16.2.6",
65
67
  "pinst": "3.0.0",
66
68
  "prettier": "3.6.2",
67
- "semantic-release": "24.2.7",
68
- "sort-package-json": "2.15.1",
69
- "typescript": "5.9.2",
70
- "vitest": "3.2.4"
69
+ "prettier-plugin-java": "2.7.7",
70
+ "semantic-release": "25.0.2",
71
+ "sort-package-json": "3.4.0",
72
+ "typescript": "5.9.3",
73
+ "typescript-eslint": "8.47.0",
74
+ "vitest": "4.0.10"
71
75
  },
72
- "packageManager": "yarn@4.9.2",
76
+ "packageManager": "yarn@4.11.0",
73
77
  "engines": {
74
- "node": ">=18"
78
+ "node": ">=24"
75
79
  },
76
80
  "publishConfig": {
77
81
  "access": "public"