@mcesystems/usbmuxd-instance-manager 1.0.75 → 1.0.76

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/dist/cli.js CHANGED
@@ -430,7 +430,6 @@ var import_dotenv = __toESM(require_main());
430
430
  // src/UsbmuxdService.ts
431
431
  var import_node_events2 = require("node:events");
432
432
  var import_tool_debug_g44 = require("@mcesystems/tool-debug-g4");
433
- var import_usb_device_listener = __toESM(require("@mcesystems/usb-device-listener"));
434
433
 
435
434
  // src/InstanceManager.ts
436
435
  var import_node_events = require("node:events");
@@ -1151,12 +1150,10 @@ var InstanceManager = class extends import_node_events.EventEmitter {
1151
1150
  var { logInfo: logInfo3, logError, logDataObject: logDataObject2 } = (0, import_tool_debug_g44.createLoggers)("usbmuxd-service");
1152
1151
  var UsbmuxdService = class extends import_node_events2.EventEmitter {
1153
1152
  manager;
1154
- usbListener;
1155
1153
  isListening = false;
1156
1154
  constructor(config2 = {}) {
1157
1155
  super();
1158
1156
  this.manager = new InstanceManager(config2);
1159
- this.usbListener = new import_usb_device_listener.default();
1160
1157
  this.setupEventHandlers();
1161
1158
  }
1162
1159
  /**
@@ -1209,39 +1206,18 @@ var UsbmuxdService = class extends import_node_events2.EventEmitter {
1209
1206
  throw new Error("Service is already running");
1210
1207
  }
1211
1208
  const config2 = this.manager.getConfig();
1212
- const appleVid = Number.parseInt(config2.appleVendorId, 16);
1213
1209
  logDataObject2("Starting service...", { config: config2 });
1214
- this.usbListener.onDeviceAdd(async (device) => {
1215
- if (device.vid !== appleVid) {
1216
- return;
1217
- }
1218
- logInfo3(`Apple device connected: ${device.deviceId} (${device.deviceName || "Unknown"})`);
1219
- try {
1220
- await this.manager.onDeviceConnected(device);
1221
- } catch (error) {
1222
- logError("Error handling device connection:", error);
1223
- }
1224
- });
1225
- this.usbListener.onDeviceRemove(async (device) => {
1226
- if (device.vid !== appleVid) {
1227
- return;
1228
- }
1229
- logInfo3(`Apple device disconnected: ${device.deviceId}`);
1230
- try {
1231
- await this.manager.onDeviceDisconnected(device);
1232
- } catch (error) {
1233
- logError("Error handling device disconnection:", error);
1234
- }
1235
- });
1236
1210
  try {
1237
- this.usbListener.startListening({
1238
- targetDevices: []
1239
- // Monitor all devices, we'll filter by VID
1240
- });
1241
1211
  this.isListening = true;
1242
1212
  this.manager.start();
1243
1213
  logInfo3("Service started successfully");
1244
1214
  logInfo3("Waiting for iOS devices...");
1215
+ const onAdd = this.manager.onDeviceConnected.bind(this.manager);
1216
+ const onRemove = this.manager.onDeviceDisconnected.bind(this.manager);
1217
+ return {
1218
+ onAdd,
1219
+ onRemove
1220
+ };
1245
1221
  } catch (error) {
1246
1222
  logError("Failed to start USB listener:", error);
1247
1223
  throw error;
@@ -1256,7 +1232,6 @@ var UsbmuxdService = class extends import_node_events2.EventEmitter {
1256
1232
  return;
1257
1233
  }
1258
1234
  logInfo3("Stopping service...");
1259
- this.usbListener.stopListening();
1260
1235
  this.isListening = false;
1261
1236
  await this.manager.stop();
1262
1237
  logInfo3("Service stopped");
package/dist/cli.js.map CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../node_modules/.pnpm/dotenv@17.2.3/node_modules/dotenv/package.json", "../../../node_modules/.pnpm/dotenv@17.2.3/node_modules/dotenv/lib/main.js", "../src/cli.ts", "../src/UsbmuxdService.ts", "../src/InstanceManager.ts", "../src/usbipd.ts", "../src/wsl.ts"],
4
- "sourcesContent": ["{\n \"name\": \"dotenv\",\n \"version\": \"17.2.3\",\n \"description\": \"Loads environment variables from .env file\",\n \"main\": \"lib/main.js\",\n \"types\": \"lib/main.d.ts\",\n \"exports\": {\n \".\": {\n \"types\": \"./lib/main.d.ts\",\n \"require\": \"./lib/main.js\",\n \"default\": \"./lib/main.js\"\n },\n \"./config\": \"./config.js\",\n \"./config.js\": \"./config.js\",\n \"./lib/env-options\": \"./lib/env-options.js\",\n \"./lib/env-options.js\": \"./lib/env-options.js\",\n \"./lib/cli-options\": \"./lib/cli-options.js\",\n \"./lib/cli-options.js\": \"./lib/cli-options.js\",\n \"./package.json\": \"./package.json\"\n },\n \"scripts\": {\n \"dts-check\": \"tsc --project tests/types/tsconfig.json\",\n \"lint\": \"standard\",\n \"pretest\": \"npm run lint && npm run dts-check\",\n \"test\": \"tap run tests/**/*.js --allow-empty-coverage --disable-coverage --timeout=60000\",\n \"test:coverage\": \"tap run tests/**/*.js --show-full-coverage --timeout=60000 --coverage-report=text --coverage-report=lcov\",\n \"prerelease\": \"npm test\",\n \"release\": \"standard-version\"\n },\n \"repository\": {\n \"type\": \"git\",\n \"url\": \"git://github.com/motdotla/dotenv.git\"\n },\n \"homepage\": \"https://github.com/motdotla/dotenv#readme\",\n \"funding\": \"https://dotenvx.com\",\n \"keywords\": [\n \"dotenv\",\n \"env\",\n \".env\",\n \"environment\",\n \"variables\",\n \"config\",\n \"settings\"\n ],\n \"readmeFilename\": \"README.md\",\n \"license\": \"BSD-2-Clause\",\n \"devDependencies\": {\n \"@types/node\": \"^18.11.3\",\n \"decache\": \"^4.6.2\",\n \"sinon\": \"^14.0.1\",\n \"standard\": \"^17.0.0\",\n \"standard-version\": \"^9.5.0\",\n \"tap\": \"^19.2.0\",\n \"typescript\": \"^4.8.4\"\n },\n \"engines\": {\n \"node\": \">=12\"\n },\n \"browser\": {\n \"fs\": false\n }\n}\n", "const fs = require('fs')\nconst path = require('path')\nconst os = require('os')\nconst crypto = require('crypto')\nconst packageJson = require('../package.json')\n\nconst version = packageJson.version\n\n// Array of tips to display randomly\nconst TIPS = [\n '\uD83D\uDD10 encrypt with Dotenvx: https://dotenvx.com',\n '\uD83D\uDD10 prevent committing .env to code: https://dotenvx.com/precommit',\n '\uD83D\uDD10 prevent building .env in docker: https://dotenvx.com/prebuild',\n '\uD83D\uDCE1 add observability to secrets: https://dotenvx.com/ops',\n '\uD83D\uDC65 sync secrets across teammates & machines: https://dotenvx.com/ops',\n '\uD83D\uDDC2\uFE0F backup and recover secrets: https://dotenvx.com/ops',\n '\u2705 audit secrets and track compliance: https://dotenvx.com/ops',\n '\uD83D\uDD04 add secrets lifecycle management: https://dotenvx.com/ops',\n '\uD83D\uDD11 add access controls to secrets: https://dotenvx.com/ops',\n '\uD83D\uDEE0\uFE0F run anywhere with `dotenvx run -- yourcommand`',\n '\u2699\uFE0F specify custom .env file path with { path: \\'/custom/path/.env\\' }',\n '\u2699\uFE0F enable debug logging with { debug: true }',\n '\u2699\uFE0F override existing env vars with { override: true }',\n '\u2699\uFE0F suppress all logs with { quiet: true }',\n '\u2699\uFE0F write to custom object with { processEnv: myObject }',\n '\u2699\uFE0F load multiple .env files with { path: [\\'.env.local\\', \\'.env\\'] }'\n]\n\n// Get a random tip from the tips array\nfunction _getRandomTip () {\n return TIPS[Math.floor(Math.random() * TIPS.length)]\n}\n\nfunction parseBoolean (value) {\n if (typeof value === 'string') {\n return !['false', '0', 'no', 'off', ''].includes(value.toLowerCase())\n }\n return Boolean(value)\n}\n\nfunction supportsAnsi () {\n return process.stdout.isTTY // && process.env.TERM !== 'dumb'\n}\n\nfunction dim (text) {\n return supportsAnsi() ? `\\x1b[2m${text}\\x1b[0m` : text\n}\n\nconst LINE = /(?:^|^)\\s*(?:export\\s+)?([\\w.-]+)(?:\\s*=\\s*?|:\\s+?)(\\s*'(?:\\\\'|[^'])*'|\\s*\"(?:\\\\\"|[^\"])*\"|\\s*`(?:\\\\`|[^`])*`|[^#\\r\\n]+)?\\s*(?:#.*)?(?:$|$)/mg\n\n// Parse src into an Object\nfunction parse (src) {\n const obj = {}\n\n // Convert buffer to string\n let lines = src.toString()\n\n // Convert line breaks to same format\n lines = lines.replace(/\\r\\n?/mg, '\\n')\n\n let match\n while ((match = LINE.exec(lines)) != null) {\n const key = match[1]\n\n // Default undefined or null to empty string\n let value = (match[2] || '')\n\n // Remove whitespace\n value = value.trim()\n\n // Check if double quoted\n const maybeQuote = value[0]\n\n // Remove surrounding quotes\n value = value.replace(/^(['\"`])([\\s\\S]*)\\1$/mg, '$2')\n\n // Expand newlines if double quoted\n if (maybeQuote === '\"') {\n value = value.replace(/\\\\n/g, '\\n')\n value = value.replace(/\\\\r/g, '\\r')\n }\n\n // Add to object\n obj[key] = value\n }\n\n return obj\n}\n\nfunction _parseVault (options) {\n options = options || {}\n\n const vaultPath = _vaultPath(options)\n options.path = vaultPath // parse .env.vault\n const result = DotenvModule.configDotenv(options)\n if (!result.parsed) {\n const err = new Error(`MISSING_DATA: Cannot parse ${vaultPath} for an unknown reason`)\n err.code = 'MISSING_DATA'\n throw err\n }\n\n // handle scenario for comma separated keys - for use with key rotation\n // example: DOTENV_KEY=\"dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=prod,dotenv://:key_7890@dotenvx.com/vault/.env.vault?environment=prod\"\n const keys = _dotenvKey(options).split(',')\n const length = keys.length\n\n let decrypted\n for (let i = 0; i < length; i++) {\n try {\n // Get full key\n const key = keys[i].trim()\n\n // Get instructions for decrypt\n const attrs = _instructions(result, key)\n\n // Decrypt\n decrypted = DotenvModule.decrypt(attrs.ciphertext, attrs.key)\n\n break\n } catch (error) {\n // last key\n if (i + 1 >= length) {\n throw error\n }\n // try next key\n }\n }\n\n // Parse decrypted .env string\n return DotenvModule.parse(decrypted)\n}\n\nfunction _warn (message) {\n console.error(`[dotenv@${version}][WARN] ${message}`)\n}\n\nfunction _debug (message) {\n console.log(`[dotenv@${version}][DEBUG] ${message}`)\n}\n\nfunction _log (message) {\n console.log(`[dotenv@${version}] ${message}`)\n}\n\nfunction _dotenvKey (options) {\n // prioritize developer directly setting options.DOTENV_KEY\n if (options && options.DOTENV_KEY && options.DOTENV_KEY.length > 0) {\n return options.DOTENV_KEY\n }\n\n // secondary infra already contains a DOTENV_KEY environment variable\n if (process.env.DOTENV_KEY && process.env.DOTENV_KEY.length > 0) {\n return process.env.DOTENV_KEY\n }\n\n // fallback to empty string\n return ''\n}\n\nfunction _instructions (result, dotenvKey) {\n // Parse DOTENV_KEY. Format is a URI\n let uri\n try {\n uri = new URL(dotenvKey)\n } catch (error) {\n if (error.code === 'ERR_INVALID_URL') {\n const err = new Error('INVALID_DOTENV_KEY: Wrong format. Must be in valid uri format like dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=development')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n throw error\n }\n\n // Get decrypt key\n const key = uri.password\n if (!key) {\n const err = new Error('INVALID_DOTENV_KEY: Missing key part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get environment\n const environment = uri.searchParams.get('environment')\n if (!environment) {\n const err = new Error('INVALID_DOTENV_KEY: Missing environment part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get ciphertext payload\n const environmentKey = `DOTENV_VAULT_${environment.toUpperCase()}`\n const ciphertext = result.parsed[environmentKey] // DOTENV_VAULT_PRODUCTION\n if (!ciphertext) {\n const err = new Error(`NOT_FOUND_DOTENV_ENVIRONMENT: Cannot locate environment ${environmentKey} in your .env.vault file.`)\n err.code = 'NOT_FOUND_DOTENV_ENVIRONMENT'\n throw err\n }\n\n return { ciphertext, key }\n}\n\nfunction _vaultPath (options) {\n let possibleVaultPath = null\n\n if (options && options.path && options.path.length > 0) {\n if (Array.isArray(options.path)) {\n for (const filepath of options.path) {\n if (fs.existsSync(filepath)) {\n possibleVaultPath = filepath.endsWith('.vault') ? filepath : `${filepath}.vault`\n }\n }\n } else {\n possibleVaultPath = options.path.endsWith('.vault') ? options.path : `${options.path}.vault`\n }\n } else {\n possibleVaultPath = path.resolve(process.cwd(), '.env.vault')\n }\n\n if (fs.existsSync(possibleVaultPath)) {\n return possibleVaultPath\n }\n\n return null\n}\n\nfunction _resolveHome (envPath) {\n return envPath[0] === '~' ? path.join(os.homedir(), envPath.slice(1)) : envPath\n}\n\nfunction _configVault (options) {\n const debug = parseBoolean(process.env.DOTENV_CONFIG_DEBUG || (options && options.debug))\n const quiet = parseBoolean(process.env.DOTENV_CONFIG_QUIET || (options && options.quiet))\n\n if (debug || !quiet) {\n _log('Loading env from encrypted .env.vault')\n }\n\n const parsed = DotenvModule._parseVault(options)\n\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n\n DotenvModule.populate(processEnv, parsed, options)\n\n return { parsed }\n}\n\nfunction configDotenv (options) {\n const dotenvPath = path.resolve(process.cwd(), '.env')\n let encoding = 'utf8'\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n let debug = parseBoolean(processEnv.DOTENV_CONFIG_DEBUG || (options && options.debug))\n let quiet = parseBoolean(processEnv.DOTENV_CONFIG_QUIET || (options && options.quiet))\n\n if (options && options.encoding) {\n encoding = options.encoding\n } else {\n if (debug) {\n _debug('No encoding is specified. UTF-8 is used by default')\n }\n }\n\n let optionPaths = [dotenvPath] // default, look for .env\n if (options && options.path) {\n if (!Array.isArray(options.path)) {\n optionPaths = [_resolveHome(options.path)]\n } else {\n optionPaths = [] // reset default\n for (const filepath of options.path) {\n optionPaths.push(_resolveHome(filepath))\n }\n }\n }\n\n // Build the parsed data in a temporary object (because we need to return it). Once we have the final\n // parsed data, we will combine it with process.env (or options.processEnv if provided).\n let lastError\n const parsedAll = {}\n for (const path of optionPaths) {\n try {\n // Specifying an encoding returns a string instead of a buffer\n const parsed = DotenvModule.parse(fs.readFileSync(path, { encoding }))\n\n DotenvModule.populate(parsedAll, parsed, options)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${path} ${e.message}`)\n }\n lastError = e\n }\n }\n\n const populated = DotenvModule.populate(processEnv, parsedAll, options)\n\n // handle user settings DOTENV_CONFIG_ options inside .env file(s)\n debug = parseBoolean(processEnv.DOTENV_CONFIG_DEBUG || debug)\n quiet = parseBoolean(processEnv.DOTENV_CONFIG_QUIET || quiet)\n\n if (debug || !quiet) {\n const keysCount = Object.keys(populated).length\n const shortPaths = []\n for (const filePath of optionPaths) {\n try {\n const relative = path.relative(process.cwd(), filePath)\n shortPaths.push(relative)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${filePath} ${e.message}`)\n }\n lastError = e\n }\n }\n\n _log(`injecting env (${keysCount}) from ${shortPaths.join(',')} ${dim(`-- tip: ${_getRandomTip()}`)}`)\n }\n\n if (lastError) {\n return { parsed: parsedAll, error: lastError }\n } else {\n return { parsed: parsedAll }\n }\n}\n\n// Populates process.env from .env file\nfunction config (options) {\n // fallback to original dotenv if DOTENV_KEY is not set\n if (_dotenvKey(options).length === 0) {\n return DotenvModule.configDotenv(options)\n }\n\n const vaultPath = _vaultPath(options)\n\n // dotenvKey exists but .env.vault file does not exist\n if (!vaultPath) {\n _warn(`You set DOTENV_KEY but you are missing a .env.vault file at ${vaultPath}. Did you forget to build it?`)\n\n return DotenvModule.configDotenv(options)\n }\n\n return DotenvModule._configVault(options)\n}\n\nfunction decrypt (encrypted, keyStr) {\n const key = Buffer.from(keyStr.slice(-64), 'hex')\n let ciphertext = Buffer.from(encrypted, 'base64')\n\n const nonce = ciphertext.subarray(0, 12)\n const authTag = ciphertext.subarray(-16)\n ciphertext = ciphertext.subarray(12, -16)\n\n try {\n const aesgcm = crypto.createDecipheriv('aes-256-gcm', key, nonce)\n aesgcm.setAuthTag(authTag)\n return `${aesgcm.update(ciphertext)}${aesgcm.final()}`\n } catch (error) {\n const isRange = error instanceof RangeError\n const invalidKeyLength = error.message === 'Invalid key length'\n const decryptionFailed = error.message === 'Unsupported state or unable to authenticate data'\n\n if (isRange || invalidKeyLength) {\n const err = new Error('INVALID_DOTENV_KEY: It must be 64 characters long (or more)')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n } else if (decryptionFailed) {\n const err = new Error('DECRYPTION_FAILED: Please check your DOTENV_KEY')\n err.code = 'DECRYPTION_FAILED'\n throw err\n } else {\n throw error\n }\n }\n}\n\n// Populate process.env with parsed values\nfunction populate (processEnv, parsed, options = {}) {\n const debug = Boolean(options && options.debug)\n const override = Boolean(options && options.override)\n const populated = {}\n\n if (typeof parsed !== 'object') {\n const err = new Error('OBJECT_REQUIRED: Please check the processEnv argument being passed to populate')\n err.code = 'OBJECT_REQUIRED'\n throw err\n }\n\n // Set process.env\n for (const key of Object.keys(parsed)) {\n if (Object.prototype.hasOwnProperty.call(processEnv, key)) {\n if (override === true) {\n processEnv[key] = parsed[key]\n populated[key] = parsed[key]\n }\n\n if (debug) {\n if (override === true) {\n _debug(`\"${key}\" is already defined and WAS overwritten`)\n } else {\n _debug(`\"${key}\" is already defined and was NOT overwritten`)\n }\n }\n } else {\n processEnv[key] = parsed[key]\n populated[key] = parsed[key]\n }\n }\n\n return populated\n}\n\nconst DotenvModule = {\n configDotenv,\n _configVault,\n _parseVault,\n config,\n decrypt,\n parse,\n populate\n}\n\nmodule.exports.configDotenv = DotenvModule.configDotenv\nmodule.exports._configVault = DotenvModule._configVault\nmodule.exports._parseVault = DotenvModule._parseVault\nmodule.exports.config = DotenvModule.config\nmodule.exports.decrypt = DotenvModule.decrypt\nmodule.exports.parse = DotenvModule.parse\nmodule.exports.populate = DotenvModule.populate\n\nmodule.exports = DotenvModule\n", "#!/usr/bin/env node\n\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport dotenv from \"dotenv\";\nimport { UsbmuxdService } from \"./UsbmuxdService.js\";\ndotenv.config();\n\nconst { logInfo, logError, logHeader, logDataObject } = createLoggers(\"usbmuxd-cli\");\n\n// Parse command line arguments\nconst args = process.argv.slice(2);\nconst options: Record<string, string> = {};\n\nfor (let i = 0; i < args.length; i++) {\n\tif (args[i].startsWith(\"--\")) {\n\t\tconst key = args[i].substring(2);\n\t\tconst value = args[i + 1];\n\t\tif (value && !value.startsWith(\"--\")) {\n\t\t\toptions[key] = value;\n\t\t\ti++;\n\t\t}\n\t}\n}\n\n// Build configuration from command line\n// usbmuxdPath is the binary run inside WSL (Alpine) \u2014 must be the Linux usbmuxd, not the Windows exe\nconst config = {\n\tbatchSize: options.batchSize ? Number.parseInt(options.batchSize, 10) : 4,\n\tbasePort: options.basePort ? Number.parseInt(options.basePort, 10) : 27015,\n\tmaxInstances: options.maxInstances ? Number.parseInt(options.maxInstances, 10) : 20,\n\tusbmuxdPath: options.usbmuxdPath || \"usbmuxd\",\n\tverboseLogging: options.verbose !== \"false\",\n\tappleVendorId: options.appleVid || \"05AC\",\n};\n\nlogHeader(\"usbmuxd Instance Manager\");\nlogDataObject(\"Multi-instance manager for iOS device connections\", { config });\n\n// Create and start service\nconst service = new UsbmuxdService(config);\n\nasync function shutdown(eventName?: string) {\n\tlogInfo(\"\");\n\tlogInfo(`Received ${eventName}, shutting down gracefully...`);\n\ttry {\n\t\tawait service.stop();\n\t\tlogInfo(\"Shutdown complete\");\n\t\tprocess.exit(0);\n\t} catch (error) {\n\t\tlogError(`Error during ${eventName}:`, error);\n\t\tprocess.exit(1);\n\t}\n}\n\n// Handle graceful shutdown\nprocess.on(\"SIGINT\", async () => {\n\tvoid shutdown(\"SIGINT\");\n});\n\nprocess.on(\"SIGTERM\", async () => {\n\tvoid shutdown(\"SIGTERM\");\n});\n\n// Print stats every 30 seconds\nsetInterval(() => {\n\tconst stats = service.getStats();\n\tconst instances = service.getInstances();\n\n\tlogHeader(\n\t\t`Uptime: ${stats.uptimeSeconds}s | Instances: ${stats.instanceCount} | Devices: ${stats.deviceCount}`\n\t);\n\n\tif (instances.length > 0) {\n\t\tlogDataObject(\"Active instances:\", { instances });\n\t}\n}, 30000);\n\n// Print help if --help flag (before starting service)\nif (args.includes(\"--help\") || args.includes(\"-h\")) {\n\tlogInfo(\"Usage: usbmuxd-instance-manager [OPTIONS]\");\n\tlogInfo(\"\");\n\tlogInfo(\"Options:\");\n\tlogInfo(\" --batchSize <n> Number of devices per instance (default: 4)\");\n\tlogInfo(\" --basePort <port> Base TCP port for first instance (default: 27015)\");\n\tlogInfo(\" --maxInstances <n> Maximum number of instances (default: 20)\");\n\tlogInfo(\" --usbmuxdPath <path> Path to usbmuxd executable\");\n\tlogInfo(\" --verbose <true|false> Enable verbose logging (default: true)\");\n\tlogInfo(\" --appleVid <vid> Apple Vendor ID in hex (default: 05AC)\");\n\tlogInfo(\" -h, --help Show this help message\");\n\tlogInfo(\"\");\n\tlogInfo(\"Examples:\");\n\tlogInfo(\" usbmuxd-instance-manager\");\n\tlogInfo(\" usbmuxd-instance-manager --batchSize 6 --basePort 27020\");\n\tlogInfo(\"\");\n\tprocess.exit(0);\n}\n\n// Start the service\ntry {\n\tservice.start();\n} catch (error) {\n\tlogError(\"Failed to start service\", error);\n\tshutdown(\"Failed to start service\");\n}\n", "import { EventEmitter } from \"node:events\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport UsbDeviceListener from \"@mcesystems/usb-device-listener\";\nimport type { DeviceInfo } from \"@mcesystems/usb-device-listener\";\nimport { InstanceManager } from \"./InstanceManager.js\";\nimport type { DeviceMapping, InstanceManagerConfig, UsbmuxdInstance } from \"./types/index.js\";\n\nconst { logInfo, logError, logDataObject } = createLoggers(\"usbmuxd-service\");\n\n/**\n * Main service that integrates usb-device-listener with InstanceManager\n * Monitors iOS devices and manages usbmuxd instances automatically.\n * Extends EventEmitter: forwards instance-started, instance-stopped, device-assigned,\n * device-removed, instance-log, instance-exited from the underlying manager.\n */\nexport class UsbmuxdService extends EventEmitter {\n\tprivate manager: InstanceManager;\n\tprivate usbListener: UsbDeviceListener;\n\tprivate isListening = false;\n\n\tconstructor(config: Partial<InstanceManagerConfig> = {}) {\n\t\tsuper();\n\t\tthis.manager = new InstanceManager(config);\n\t\tthis.usbListener = new UsbDeviceListener();\n\t\tthis.setupEventHandlers();\n\t}\n\n\t/**\n\t * Forward manager events to service subscribers (no logging; CLI/examples handle logging).\n\t */\n\tprivate setupEventHandlers(): void {\n\t\tthis.manager.on(\"instance-started\", (instance: UsbmuxdInstance) => {\n\t\t\tthis.emit(\"instance-started\", instance);\n\t\t});\n\n\t\tthis.manager.on(\"instance-stopped\", (instance: UsbmuxdInstance) => {\n\t\t\tthis.emit(\"instance-stopped\", instance);\n\t\t});\n\n\t\tthis.manager.on(\n\t\t\t\"device-assigned\",\n\t\t\t(payload: { device: DeviceInfo; instance: UsbmuxdInstance; mapping: DeviceMapping }) => {\n\t\t\t\tlogInfo(\n\t\t\t\t\t`Device ${payload.device.deviceId} connected to usbmuxd instance at ${payload.mapping.host}:${payload.mapping.port}`\n\t\t\t\t);\n\t\t\t\tthis.emit(\"device-assigned\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\n\t\t\t\"device-removed\",\n\t\t\t(payload: { device: DeviceInfo; instance: UsbmuxdInstance }) => {\n\t\t\t\tthis.emit(\"device-removed\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\n\t\t\t\"instance-log\",\n\t\t\t(payload: { instanceId: number; level: string; message: string }) => {\n\t\t\t\tthis.emit(\"instance-log\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\n\t\t\t\"instance-exited\",\n\t\t\t(payload: { instanceId: number; code: number | null; signal: string | null }) => {\n\t\t\t\tthis.emit(\"instance-exited\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\"device-paired\", (payload: { udid: string; mapping: DeviceMapping }) => {\n\t\t\tthis.emit(\"device-paired\", payload);\n\t\t});\n\t}\n\n\t/**\n\t * Start the service\n\t * Begins monitoring for iOS devices\n\t */\n\tpublic start(): void {\n\t\tif (this.isListening) {\n\t\t\tthrow new Error(\"Service is already running\");\n\t\t}\n\n\t\tconst config = this.manager.getConfig();\n\n\t\t// Convert Apple VID from hex to decimal for comparison\n\t\tconst appleVid = Number.parseInt(config.appleVendorId, 16);\n\n\t\tlogDataObject(\"Starting service...\", { config });\n\n\t\t// Set up device listener callbacks\n\t\tthis.usbListener.onDeviceAdd(async (device: DeviceInfo) => {\n\t\t\t// Filter for Apple devices only\n\t\t\tif (device.vid !== appleVid) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlogInfo(`Apple device connected: ${device.deviceId} (${device.deviceName || \"Unknown\"})`);\n\n\t\t\ttry {\n\t\t\t\tawait this.manager.onDeviceConnected(device);\n\t\t\t} catch (error) {\n\t\t\t\tlogError(\"Error handling device connection:\", error);\n\t\t\t}\n\t\t});\n\n\t\tthis.usbListener.onDeviceRemove(async (device: DeviceInfo) => {\n\t\t\t// Filter for Apple devices only\n\t\t\tif (device.vid !== appleVid) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlogInfo(`Apple device disconnected: ${device.deviceId}`);\n\n\t\t\ttry {\n\t\t\t\tawait this.manager.onDeviceDisconnected(device);\n\t\t\t} catch (error) {\n\t\t\t\tlogError(\"Error handling device disconnection:\", error);\n\t\t\t}\n\t\t});\n\n\t\t// Start listening for USB events\n\t\ttry {\n\t\t\tthis.usbListener.startListening({\n\t\t\t\ttargetDevices: [], // Monitor all devices, we'll filter by VID\n\t\t\t});\n\n\t\t\tthis.isListening = true;\n\t\t\tthis.manager.start();\n\n\t\t\tlogInfo(\"Service started successfully\");\n\t\t\tlogInfo(\"Waiting for iOS devices...\");\n\t\t} catch (error) {\n\t\t\tlogError(\"Failed to start USB listener:\", error);\n\t\t\tthrow error;\n\t\t}\n\t}\n\n\t/**\n\t * Stop the service\n\t * Stops monitoring and terminates all instances\n\t */\n\tpublic async stop(): Promise<void> {\n\t\tif (!this.isListening) {\n\t\t\treturn;\n\t\t}\n\n\t\tlogInfo(\"Stopping service...\");\n\n\t\t// Stop USB listener\n\t\tthis.usbListener.stopListening();\n\t\tthis.isListening = false;\n\n\t\t// Stop all instances\n\t\tawait this.manager.stop();\n\n\t\tlogInfo(\"Service stopped\");\n\t}\n\n\t/**\n\t * Get device port mapping\n\t */\n\tpublic getDevicePort(udid: string): number | null {\n\t\treturn this.manager.getDevicePort(udid);\n\t}\n\n\t/**\n\t * Get all device mappings\n\t */\n\tpublic getDeviceMappings() {\n\t\treturn this.manager.getDeviceMappings();\n\t}\n\n\t/**\n\t * Get all running instances\n\t */\n\tpublic getInstances() {\n\t\treturn this.manager.getInstances();\n\t}\n\n\t/**\n\t * Get service statistics\n\t */\n\tpublic getStats() {\n\t\treturn this.manager.getStats();\n\t}\n\n\t/**\n\t * Get current configuration\n\t */\n\tpublic getConfig() {\n\t\treturn this.manager.getConfig();\n\t}\n}\n", "import type { ChildProcess } from \"node:child_process\";\nimport { EventEmitter } from \"node:events\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type {\n\tDeviceInfo,\n\tDeviceMapping,\n\tInstanceManagerConfig,\n\tManagerStats,\n\tUsbmuxdInstance,\n} from \"./types/index.js\";\nimport { Usbipd } from \"./usbipd.js\";\nimport { Wsl2 } from \"./wsl.js\";\n\nconst { logWarning, logDetail, logDataObject, logTask } = createLoggers(\"instance-manager\");\n\n/**\n * Default configuration for the instance manager\n */\nconst DEFAULT_CONFIG: InstanceManagerConfig = {\n\tbatchSize: 4,\n\tbasePort: 27015,\n\tmaxInstances: 5,\n\tusbmuxdPath: \"usbmuxd\", // Path inside WSL2\n\twslDistribution: \"alpine-usbmuxd-build\", // Alpine WSL2 distribution name\n\tverboseLogging: true,\n\tappleVendorId: \"05AC\",\n};\n\n/**\n * Manages multiple usbmuxd instances with batch device allocation\n *\n * @example\n * const manager = new InstanceManager({\n * batchSize: 4,\n * basePort: 27015,\n * });\n *\n * manager.on('instance-started', (instance) => {\n * logInfo(`Instance ${instance.id} started on port ${instance.port}`);\n * });\n *\n * manager.start();\n */\nexport class InstanceManager extends EventEmitter {\n\t/** Map of usbmuxd instance IDs to instances */\n\tprivate instances: Map<number, UsbmuxdInstance> = new Map();\n\t/** Map of device IDs to device mappings host:port */\n\tprivate deviceMappings: Map<string, DeviceMapping> = new Map();\n\t/** Map of usbmuxd instance IDs to child processes running it on wsl2*/\n\tprivate processes: Map<number, ChildProcess> = new Map();\n\t/** Tracks which devices have been attached to WSL */\n\tprivate attachedDevices: Set<string> = new Set();\n\t/** Device IDs currently in the attach flow\n\t * (ignore disconnect until attach completes) */\n\tprivate pendingAttachDevices: Set<string> = new Set();\n\n\tprivate config: InstanceManagerConfig;\n\tprivate nextInstanceId = 1;\n\tprivate startedAt: Date | null = null;\n\tprivate isRunning = false;\n\n\tprivate wsl: Wsl2;\n\tprivate usbipd: Usbipd;\n\tconstructor(config: Partial<InstanceManagerConfig> = {}) {\n\t\tsuper();\n\t\tthis.config = { ...DEFAULT_CONFIG, ...config };\n\t\tthis.wsl = new Wsl2(this.config.wslDistribution);\n\t\tthis.usbipd = new Usbipd(process.env.USBIPD_PATH ?? \"usbipd\");\n\t}\n\n\t/**\n\t * Start the instance manager\n\t */\n\tpublic start(): void {\n\t\tif (this.isRunning) {\n\t\t\tthrow new Error(\"Instance manager is already running\");\n\t\t}\n\n\t\tthis.isRunning = true;\n\t\tthis.startedAt = new Date();\n\t}\n\n\t/**\n\t * Stop the instance manager and all instances\n\t */\n\tpublic async stop(): Promise<void> {\n\t\tif (!this.isRunning) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis.isRunning = false;\n\n\t\t// Sync lockdown files from Alpine back to native Apple lockdown directory (before clearing mappings)\n\t\tconst udids = Array.from(this.deviceMappings.keys());\n\t\tawait this.wsl.syncFromAlpine(udids);\n\n\t\t// Detach all devices from WSL.\n\t\t// If you only called unbind without detach,\n\t\t// the device might still be attached to the WSL session\n\t\tawait this.usbipd.detachAllDevicesFromWsl();\n\t\tawait this.usbipd.unbindAllDevicesFromWsl();\n\t\tthis.attachedDevices.clear();\n\n\t\t// Stop all instances\n\t\tconst stopPromises = Array.from(this.instances.keys()).map((id) => this.stopInstance(id));\n\n\t\tawait Promise.all(stopPromises);\n\n\t\tthis.instances.clear();\n\t\tthis.deviceMappings.clear();\n\t\tthis.processes.clear();\n\t}\n\n\tprivate async attachToWsl({\n\t\tbusId,\n\t\tdeviceId,\n\t}: { busId: string; deviceId: string }): Promise<void> {\n\t\tif (!this.attachedDevices.has(busId)) {\n\t\t\tthis.pendingAttachDevices.add(deviceId);\n\t\t\tif (!this.config.wslDistribution) {\n\t\t\t\tthrow new Error(\"WSL distribution not configured\");\n\t\t\t}\n\t\t\tconst attached = await this.usbipd.attachDeviceToWsl(\n\t\t\t\tbusId,\n\t\t\t\tthis.wsl,\n\t\t\t\tthis.config.wslDistribution\n\t\t\t);\n\t\t\tif (attached) {\n\t\t\t\tthis.attachedDevices.add(busId);\n\t\t\t}\n\t\t}\n\t}\n\n\tprivate async removeDevice(mapping: DeviceMapping): Promise<UsbmuxdInstance> {\n\t\tconst instance = await this.detachFromWsl({ deviceId: mapping.udid });\n\t\tif (!instance) {\n\t\t\tthrow new Error(`Instance ${mapping.instanceId} not found`);\n\t\t}\n\t\tif (mapping.busId) {\n\t\t\tawait this.usbipd.detachDeviceFromWsl(mapping.busId);\n\t\t\tthis.attachedDevices.delete(mapping.busId);\n\t\t}\n\t\tconst deviceIndex = instance.deviceUdids.indexOf(mapping.udid);\n\t\tif (deviceIndex > -1) {\n\t\t\tinstance.deviceUdids.splice(deviceIndex, 1);\n\t\t}\n\n\t\tthis.deviceMappings.delete(mapping.udid);\n\t\treturn instance;\n\t}\n\n\tprivate async onUsbmuxdInstanceEnd({\n\t\tinstanceId,\n\t\tcode,\n\t\tsignal,\n\t}: { instanceId: number; code: number; signal: string }): Promise<void> {\n\t\tthis.emit(\"instance-exited\", {\n\t\t\tinstanceId,\n\t\t\tcode,\n\t\t\tsignal,\n\t\t});\n\t\tfor (const [_, mapping] of this.deviceMappings.entries()) {\n\t\t\tif (mapping.instanceId === instanceId) {\n\t\t\t\tthis.removeDevice(mapping);\n\t\t\t}\n\t\t}\n\t\tthis.instances.delete(instanceId);\n\t\tthis.processes.get(instanceId)?.kill();\n\t\tthis.processes.delete(instanceId);\n\t}\n\n\tprivate async onUsbmuxdInstanceStart({\n\t\tinstance,\n\t\tprocess,\n\t}: { instance: UsbmuxdInstance; process: ChildProcess }): Promise<void> {\n\t\tthis.instances.set(instance.id, instance);\n\t\tthis.processes.set(instance.id, process);\n\t\tthis.emit(\"instance-started\", {\n\t\t\tinstanceId: instance.id,\n\t\t});\n\t}\n\n\tprivate async getInstance() {\n\t\tlet targetInstance = this.findInstanceWithCapacity();\n\n\t\t// If no instance has capacity, create a new one\n\t\tif (!targetInstance) {\n\t\t\tif (this.instances.size >= this.config.maxInstances) {\n\t\t\t\tthrow new Error(`Maximum number of instances (${this.config.maxInstances}) reached`);\n\t\t\t}\n\n\t\t\tconst newInstanceId = this.nextInstanceId++;\n\n\t\t\tlet onExit = (code: number, signal: string) => {\n\t\t\t\tthis.onUsbmuxdInstanceEnd({ instanceId: newInstanceId, code, signal });\n\t\t\t};\n\n\t\t\tonExit = onExit.bind(this);\n\n\t\t\tconst { instance, process } = await this.wsl.createInstance({\n\t\t\t\tid: newInstanceId,\n\t\t\t\tbasePort: this.config.basePort,\n\t\t\t\tverboseLogging: this.config.verboseLogging,\n\t\t\t\tusbmuxdPath: this.config.usbmuxdPath,\n\t\t\t\tonLog: (message) =>\n\t\t\t\t\tthis.emit(\"instance-log\", {\n\t\t\t\t\t\tinstanceId: newInstanceId,\n\t\t\t\t\t\tlevel: \"info\",\n\t\t\t\t\t\tmessage,\n\t\t\t\t\t}),\n\t\t\t\tonExit,\n\t\t\t});\n\t\t\tthis.onUsbmuxdInstanceStart({ instance, process });\n\t\t\ttargetInstance = instance;\n\t\t}\n\t\treturn targetInstance;\n\t}\n\n\t/**\n\t * Handle device connection only for not attached devices\n\t * Attaches device to WSL, then assigns to an existing instance or creates a new one\n\t */\n\tpublic async onDeviceConnected(device: DeviceInfo): Promise<void> {\n\t\tconst busId = await this.usbipd.findBusIdForDevice(device);\n\n\t\tif (this.attachedDevices.has(busId)) {\n\t\t\tlogTask(`Device ${device.deviceId} is already attached. skipping onDeviceConnected...`);\n\t\t\treturn;\n\t\t}\n\n\t\tlogDataObject(\"Device connected\", { device });\n\t\tawait this.wsl.syncToAlpine(device.deviceId);\n\n\t\tawait this.attachToWsl({ busId, deviceId: device.deviceId });\n\n\t\tconst targetInstance = await this.getInstance();\n\n\t\t// Create mapping (include host and busId for cleanup on disconnect)\n\t\tconst mapping: DeviceMapping = {\n\t\t\tudid: device.deviceId,\n\t\t\tinstanceId: targetInstance.id,\n\t\t\thost: targetInstance.host,\n\t\t\tport: targetInstance.port,\n\t\t\taddedAt: new Date(),\n\t\t\tbusId: busId ?? undefined,\n\t\t};\n\n\t\tthis.deviceMappings.set(device.deviceId, mapping);\n\n\t\t// Add device to instance\n\t\ttargetInstance.deviceUdids.push(device.deviceId);\n\n\t\tthis.emit(\"device-assigned\", {\n\t\t\tdevice,\n\t\t\tinstance: targetInstance,\n\t\t\tmapping,\n\t\t});\n\n\t\tthis.pendingAttachDevices.delete(device.deviceId);\n\t}\n\n\tprivate async detachFromWsl({ deviceId }: { deviceId: string }): Promise<UsbmuxdInstance> {\n\t\tconst mapping = this.deviceMappings.get(deviceId);\n\t\tif (!mapping) {\n\t\t\tthrow new Error(`Device ${deviceId} was not tracked`);\n\t\t}\n\n\t\t// Detach device from WSL if it was attached\n\t\tif (mapping.busId) {\n\t\t\tawait this.usbipd.detachDeviceFromWsl(mapping.busId);\n\t\t\tthis.attachedDevices.delete(mapping.busId);\n\t\t}\n\n\t\tconst instance = this.instances.get(mapping.instanceId);\n\t\tif (!instance) {\n\t\t\tlogWarning(`Instance ${mapping.instanceId} not found`);\n\t\t\tthis.deviceMappings.delete(deviceId);\n\t\t\tthrow new Error(`Instance ${mapping.instanceId} not found`);\n\t\t}\n\n\t\treturn instance;\n\t}\n\n\t/**\n\t * Handle device disconnection only for attached devices\n\t * Detaches from WSL, removes device from instance, and stops instance if empty\n\t */\n\tpublic async onDeviceDisconnected(device: DeviceInfo): Promise<void> {\n\t\t// Ignore disconnect while we're still attaching this device (Windows sees disconnect when device moves to WSL)\n\t\tif (!this.attachedDevices.has(device.deviceId)) {\n\t\t\tlogTask(`Device ${device.deviceId} is not attached. skipping onDeviceDisconnected...`);\n\t\t\treturn;\n\t\t}\n\t\tlogDataObject(\"Device disconnected\", { device });\n\n\t\tconst mapping = this.deviceMappings.get(device.deviceId);\n\t\tif (!mapping) {\n\t\t\tthrow new Error(`Device ${device.deviceId} was not tracked`);\n\t\t}\n\t\t// Remove device from instance\n\t\tconst instance = await this.removeDevice(mapping);\n\n\t\tthis.emit(\"device-removed\", {\n\t\t\tdevice,\n\t\t\tinstance,\n\t\t});\n\n\t\t// If instance is now empty, stop it\n\t\tif (instance.deviceUdids.length === 0) {\n\t\t\tlogDetail(`Instance ${instance.id} is empty, stopping...`);\n\t\t\tawait this.stopInstance(instance.id);\n\t\t}\n\t}\n\n\t/**\n\t * Find an instance with available capacity\n\t */\n\tprivate findInstanceWithCapacity(): UsbmuxdInstance | null {\n\t\tfor (const instance of this.instances.values()) {\n\t\t\tif (instance.deviceUdids.length < this.config.batchSize) {\n\t\t\t\treturn instance;\n\t\t\t}\n\t\t}\n\t\treturn null;\n\t}\n\n\t/**\n\t * Stop a specific instance\n\t */\n\tprivate async stopInstance(instanceId: number): Promise<void> {\n\t\tconst instance = this.instances.get(instanceId);\n\t\tconst childProcess = this.processes.get(instanceId);\n\n\t\tif (!instance || !childProcess) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor (const process of this.processes.values()) {\n\t\t\tprocess.kill(\"SIGKILL\");\n\t\t}\n\n\t\t// Clean up\n\t\tthis.instances.delete(instanceId);\n\t\tthis.processes.delete(instanceId);\n\n\t\t// Remove device mappings\n\t\tfor (const [udid, mapping] of this.deviceMappings.entries()) {\n\t\t\tif (mapping.instanceId === instanceId) {\n\t\t\t\tthis.deviceMappings.delete(udid);\n\t\t\t}\n\t\t}\n\n\t\tthis.emit(\"instance-stopped\", instance);\n\t}\n\n\t/**\n\t * Get device-to-port mapping for a specific UDID\n\t */\n\tpublic getDevicePort(udid: string): number | null {\n\t\tconst mapping = this.deviceMappings.get(udid);\n\t\treturn mapping ? mapping.port : null;\n\t}\n\n\t/**\n\t * Get all device mappings\n\t */\n\tpublic getDeviceMappings(): DeviceMapping[] {\n\t\treturn Array.from(this.deviceMappings.values());\n\t}\n\n\t/**\n\t * Get all running instances\n\t */\n\tpublic getInstances(): UsbmuxdInstance[] {\n\t\treturn Array.from(this.instances.values());\n\t}\n\n\t/**\n\t * Get manager statistics\n\t */\n\tpublic getStats(): ManagerStats {\n\t\tconst now = Date.now();\n\t\tconst startTime = this.startedAt?.getTime() || now;\n\t\tconst uptimeSeconds = Math.floor((now - startTime) / 1000);\n\n\t\treturn {\n\t\t\tinstanceCount: this.instances.size,\n\t\t\tdeviceCount: this.deviceMappings.size,\n\t\t\tuptimeSeconds,\n\t\t\tstartedAt: this.startedAt || new Date(),\n\t\t};\n\t}\n\n\t/**\n\t * Get current configuration\n\t */\n\tpublic getConfig(): InstanceManagerConfig {\n\t\treturn { ...this.config };\n\t}\n}\n", "import { exec } from \"node:child_process\";\nimport { promisify } from \"node:util\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type { DeviceInfo } from \"./types\";\nimport type { UsbipdDevice } from \"./types/usbipd.js\";\nimport type { Wsl2 } from \"./wsl.js\";\n\nconst { logInfo, logWarning } = createLoggers(\"usbipd\");\n\nconst execAsync = promisify(exec);\n\nexport class Usbipd {\n\tconstructor(private readonly usbipdPath: string) {}\n\n\tpublic async unbindAllDevicesFromWsl(): Promise<void> {\n\t\ttry {\n\t\t\tawait execAsync(`\"${this.usbipdPath}\" unbind -a`);\n\t\t\tlogInfo(\"All devices unbound from WSL\");\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to unbind all devices from WSL: ${error}`);\n\t\t}\n\t}\n\n\tpublic async detachAllDevicesFromWsl(): Promise<void> {\n\t\ttry {\n\t\t\tawait execAsync(`\"${this.usbipdPath}\" detach -a`);\n\t\t\tlogInfo(\"All devices detached from WSL\");\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to detach all devices from WSL: ${error}`);\n\t\t}\n\t}\n\n\t/**\n\t * Detach a device from WSL via usbipd\n\t */\n\tpublic async detachDeviceFromWsl(busId: string): Promise<void> {\n\t\ttry {\n\t\t\tawait execAsync(`\"${this.usbipdPath}\" detach --busid=${busId}`);\n\t\t\tlogInfo(`Device ${busId} detached from WSL`);\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\tif (message.includes(\"no device with busid\") || message.includes(\"There is no device\")) {\n\t\t\t\tlogInfo(`Device ${busId} already detached`);\n\t\t\t} else {\n\t\t\t\tlogWarning(`Failed to detach device ${busId}: ${error}`);\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Attach a device to WSL via usbipd\n\t * Note: This requires administrator privileges\n\t */\n\tpublic async attachDeviceToWsl(busId: string, wsl: Wsl2, distro: string): Promise<boolean> {\n\t\ttry {\n\t\t\t// Ensure WSL is running first\n\t\t\tawait wsl.ensureWslRunning();\n\n\t\t\t// Bind the device (make it shareable)\n\t\t\tlogInfo(`Binding device ${busId}...`);\n\t\t\tawait execAsync(`\"${this.usbipdPath}\" bind --busid ${busId} --force`);\n\n\t\t\t// Attach to WSL (use specific distro if configured, otherwise let usbipd pick)\n\t\t\tif (distro) {\n\t\t\t\tlogInfo(`Attaching device ${busId} to WSL distribution ${distro}...`);\n\t\t\t\tawait execAsync(`\"${this.usbipdPath}\" attach --wsl=${distro} --busid=${busId}`);\n\t\t\t} else {\n\t\t\t\tlogInfo(`Attaching device ${busId} to default WSL...`);\n\t\t\t\tawait execAsync(`\"${this.usbipdPath}\" attach --wsl --busid=${busId}`);\n\t\t\t}\n\n\t\t\tlogInfo(`Device ${busId} attached to WSL successfully`);\n\t\t\treturn true;\n\t\t} catch (error) {\n\t\t\tconst message = error instanceof Error ? error.message : String(error);\n\t\t\t// Device already attached (e.g. from a previous run) \u2014 treat as success so we still start usbmuxd\n\t\t\tif (message.includes(\"already attached to a client\")) {\n\t\t\t\tlogInfo(`Device ${busId} is already attached to WSL, continuing`);\n\t\t\t\treturn true;\n\t\t\t}\n\t\t\tlogWarning(`Failed to attach device ${busId} to WSL: ${error}`);\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Parse usbipd list output to extract device information\n\t */\n\tprivate parseUsbipdList(output: string): UsbipdDevice[] {\n\t\tconst devices: UsbipdDevice[] = [];\n\t\t// Handle both \\r\\n and \\n line endings\n\t\tconst lines = output.split(/\\r?\\n/);\n\n\t\tfor (const line of lines) {\n\t\t\t// Format: \"4-5 05ac:12a8 Apple Mobile Device USB Composite Device Shared (forced)\"\n\t\t\t// More flexible regex: busid, vid:pid, then anything else\n\t\t\tconst match = line.match(/^(\\d+-\\d+)\\s+([0-9a-f]{4}):([0-9a-f]{4})\\s+(.+)$/i);\n\t\t\tif (match) {\n\t\t\t\t// Split the rest into description and state (state is the last word(s) after multiple spaces)\n\t\t\t\tconst rest = match[4].trim();\n\t\t\t\tconst stateMatch = rest.match(/^(.+?)\\s{2,}(\\S.*)$/);\n\t\t\t\tconst description = stateMatch ? stateMatch[1].trim() : rest;\n\t\t\t\tconst state = stateMatch ? stateMatch[2].trim() : \"Unknown\";\n\n\t\t\t\tdevices.push({\n\t\t\t\t\tbusId: match[1],\n\t\t\t\t\tvid: match[2].toUpperCase(),\n\t\t\t\t\tpid: match[3].toUpperCase(),\n\t\t\t\t\tdescription,\n\t\t\t\t\tstate,\n\t\t\t\t});\n\t\t\t}\n\t\t}\n\n\t\treturn devices;\n\t}\n\t/**\n\t * Find the usbipd bus ID for a device by matching VID/PID\n\t */\n\tpublic async findBusIdForDevice(device: DeviceInfo): Promise<string> {\n\t\ttry {\n\t\t\tconst { stdout } = await execAsync(`\"${this.usbipdPath}\" list`);\n\t\t\tconst usbipdDevices = this.parseUsbipdList(stdout);\n\n\t\t\t// Convert device VID/PID to hex string for comparison\n\t\t\tconst deviceVid = device.vid.toString(16).toUpperCase().padStart(4, \"0\");\n\t\t\tconst devicePid = device.pid.toString(16).toUpperCase().padStart(4, \"0\");\n\n\t\t\tlogInfo(`Looking for device with VID:PID ${deviceVid}:${devicePid}`);\n\t\t\tlogInfo(`Found ${usbipdDevices.length} devices from usbipd list`);\n\n\t\t\t// Find matching device by VID/PID\n\t\t\tconst match = usbipdDevices.find((d) => d.vid === deviceVid && d.pid === devicePid);\n\n\t\t\tif (match) {\n\t\t\t\tlogInfo(\n\t\t\t\t\t`Found usbipd bus ID ${match.busId} for device ${device.deviceId} (${deviceVid}:${devicePid})`\n\t\t\t\t);\n\t\t\t\treturn match.busId;\n\t\t\t}\n\n\t\t\tthrow new Error(\n\t\t\t\t`Could not find usbipd bus ID for device ${device.deviceId} (${deviceVid}:${devicePid})`\n\t\t\t);\n\t\t} catch (error) {\n\t\t\tthrow new Error(`Failed to run usbipd list: ${error}`);\n\t\t}\n\t}\n}\n", "import { type ChildProcess, exec, spawn } from \"node:child_process\";\nimport { existsSync, mkdirSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { promisify } from \"node:util\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type { UsbmuxdInstance } from \"./types\";\n\nconst { logInfo, logWarning, logDetail } = createLoggers(\"wsl\");\n\nconst execAsync = promisify(exec);\n\nexport class Wsl2 {\n\tprivate readonly ALPINE_LOCKDOWN_DIR = \"/var/lib/lockdown\";\n\tprivate readonly SYSTEM_CONFIG_PLIST = \"SystemConfiguration.plist\";\n\n\tprivate wslIpAddress: string | null = null;\n\n\tconstructor(private wslDistribution: string | undefined) {}\n\n\t/**\n\t * Detect the WSL2 IP address for the configured distribution\n\t * This IP is needed to connect from Windows to services inside WSL\n\t */\n\tpublic async detectWslIpAddress(): Promise<string> {\n\t\tif (this.wslIpAddress) {\n\t\t\treturn this.wslIpAddress;\n\t\t}\n\n\t\tconst distro = this.wslDistribution || \"alpine-usbmuxd-build\";\n\n\t\ttry {\n\t\t\t// Try to get IP from eth0 interface\n\t\t\tconst { stdout } = await execAsync(`wsl -d ${distro} -- ip -4 addr show eth0`);\n\t\t\tconst match = stdout.match(/inet\\s+(\\d+\\.\\d+\\.\\d+\\.\\d+)/);\n\t\t\tif (match) {\n\t\t\t\tthis.wslIpAddress = match[1];\n\t\t\t\tlogInfo(`Detected WSL IP address: ${this.wslIpAddress}`);\n\t\t\t\treturn this.wslIpAddress;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to detect WSL IP via ip addr: ${error}`);\n\t\t}\n\n\t\t// Fallback: try hostname -I\n\t\ttry {\n\t\t\tconst { stdout } = await execAsync(`wsl -d ${distro} -- hostname -I`);\n\t\t\tconst ip = stdout.trim().split(/\\s+/)[0];\n\t\t\tif (ip && /^\\d+\\.\\d+\\.\\d+\\.\\d+$/.test(ip)) {\n\t\t\t\tthis.wslIpAddress = ip;\n\t\t\t\tlogInfo(`Detected WSL IP address (hostname): ${this.wslIpAddress}`);\n\t\t\t\treturn this.wslIpAddress;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to detect WSL IP via hostname: ${error}`);\n\t\t}\n\n\t\t// Last resort fallback - localhost (may work with WSL2 mirrored networking)\n\t\tlogWarning(\"Could not detect WSL IP, falling back to localhost\");\n\t\tthis.wslIpAddress = \"127.0.0.1\";\n\t\treturn this.wslIpAddress;\n\t}\n\n\t/**\n\t * Ensure the WSL distribution is running\n\t */\n\tpublic async ensureWslRunning(): Promise<boolean> {\n\t\tconst distro = this.wslDistribution;\n\n\t\ttry {\n\t\t\tif (distro) {\n\t\t\t\t// Start specific distribution\n\t\t\t\tlogInfo(`Starting WSL distribution: ${distro}...`);\n\t\t\t\tawait execAsync(`wsl -d ${distro} -- echo \"WSL started\"`);\n\t\t\t} else {\n\t\t\t\t// Start default WSL\n\t\t\t\tlogInfo(\"Starting default WSL distribution...\");\n\t\t\t\tawait execAsync(`wsl -- echo \"WSL started\"`);\n\t\t\t}\n\t\t\treturn true;\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to start WSL: ${error}`);\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Create a new usbmuxd instance\n\t */\n\tpublic async createInstance({\n\t\tid,\n\t\tbasePort,\n\t\tverboseLogging,\n\t\tusbmuxdPath,\n\t\tonLog,\n\t\tonExit,\n\t}: {\n\t\tid: number;\n\t\tbasePort: number;\n\t\tverboseLogging: boolean;\n\t\tusbmuxdPath: string;\n\t\tonLog: (message: string) => void;\n\t\tonExit: (code: number, signal: string) => void;\n\t}): Promise<{ instance: UsbmuxdInstance; process: ChildProcess }> {\n\t\tconst port = basePort + id - 1;\n\n\t\tlogInfo(`Creating instance ${id} on port ${port}`);\n\n\t\t// Detect WSL IP for connecting from Windows\n\t\tconst host = await this.detectWslIpAddress();\n\n\t\t// Build usbmuxd arguments\n\t\tconst usbmuxdArgs = [\n\t\t\t\"-f\", // Foreground\n\t\t\t\"-v\", // Verbose (if enabled)\n\t\t\t\"-S\",\n\t\t\t`0.0.0.0:${port}`, // Listen on all interfaces (for Windows \u2192 WSL2)\n\t\t\t\"--pidfile\",\n\t\t\t\"NONE\",\n\t\t];\n\n\t\tif (!verboseLogging) {\n\t\t\t// Remove -v if not verbose\n\t\t\tusbmuxdArgs.splice(1, 1);\n\t\t}\n\n\t\t// Build WSL command: wsl -d <distro> usbmuxd <args>\n\t\tconst wslArgs = [\n\t\t\t\"-d\",\n\t\t\tthis.wslDistribution || \"alpine-usbmuxd-build\",\n\t\t\tusbmuxdPath,\n\t\t\t...usbmuxdArgs,\n\t\t];\n\n\t\t// Spawn via WSL\n\t\t// detached: true creates a new process group so the child does NOT receive\n\t\t// the parent's Ctrl+C. Without this, Windows delivers SIGINT to every process\n\t\t// in the console, killing usbmuxd before stop() can sync lockdown files back.\n\t\t// windowsHide: true prevents the detached process from opening a visible console window.\n\t\tconst process = spawn(\"wsl\", wslArgs, {\n\t\t\tstdio: [\"ignore\", \"pipe\", \"pipe\"],\n\t\t\tdetached: true,\n\t\t\twindowsHide: false,\n\t\t});\n\n\t\t// Handle process output\n\t\tprocess.stdout?.on(\"data\", (data) => {\n\t\t\tonLog(data.toString().trim());\n\t\t});\n\n\t\tprocess.stderr?.on(\"data\", (data) => {\n\t\t\tonLog(data.toString().trim());\n\t\t});\n\n\t\t// Handle process exit\n\t\tprocess.on(\"exit\", onExit);\n\n\t\t// Create instance record (pid is set synchronously by Node after spawn)\n\t\tconst pid = process.pid;\n\t\tif (pid === undefined) {\n\t\t\tprocess.kill(\"SIGKILL\");\n\t\t\tthrow new Error(\"Failed to get PID for usbmuxd instance\");\n\t\t}\n\t\tconst instance: UsbmuxdInstance = {\n\t\t\tid,\n\t\t\thost,\n\t\t\tport,\n\t\t\tpid,\n\t\t\tdeviceUdids: [],\n\t\t\tstartedAt: new Date(),\n\t\t};\n\n\t\treturn { instance, process };\n\t}\n\n\t/**\n\t * Convert a Windows path to the equivalent path inside WSL (e.g. C:\\foo\\bar -> /mnt/c/foo/bar).\n\t */\n\tprivate windowsPathToWsl(windowsPath: string): string {\n\t\tconst normalized = windowsPath.replace(/\\\\/g, \"/\").trim();\n\t\tconst driveMatch = normalized.match(/^([a-zA-Z]):\\/?(.*)$/);\n\t\tif (driveMatch) {\n\t\t\tconst drive = driveMatch[1].toLowerCase();\n\t\t\tconst rest = driveMatch[2] || \"\";\n\t\t\treturn `/mnt/${drive}${rest ? `/${rest}` : \"\"}`;\n\t\t}\n\t\treturn normalized;\n\t}\n\n\t/**\n\t * Get the native Apple lockdown directory path for the current platform.\n\t * This is where Apple/iTunes stores pairing records natively.\n\t */\n\tprivate getAppleLockdownPath(): string | null {\n\t\tif (process.platform === \"win32\") {\n\t\t\treturn join(process.env.ProgramData ?? \"C:\\\\ProgramData\", \"Apple\", \"Lockdown\");\n\t\t}\n\t\tif (process.platform === \"darwin\") {\n\t\t\treturn \"/var/db/lockdown\";\n\t\t}\n\t\tif (process.platform === \"linux\") {\n\t\t\treturn \"/var/lib/lockdown\";\n\t\t}\n\t\treturn null;\n\t}\n\n\t/**\n\t * Copy a single device's lockdown plist from the native Apple lockdown directory\n\t * to Alpine so usbmuxd can use it (skip pairing).\n\t *\n\t * Uses the platform-specific Apple/iTunes lockdown directory\n\t * (e.g. C:\\ProgramData\\Apple\\Lockdown on Windows, /var/db/lockdown on macOS).\n\t *\n\t * No-op if the lockdown directory cannot be determined or files don't exist.\n\t */\n\tpublic async syncToAlpine(udid: string): Promise<void> {\n\t\tconst lockdownDir = this.getAppleLockdownPath();\n\t\tif (!lockdownDir) {\n\t\t\tlogWarning(\"Lockdown sync to Alpine: unsupported platform, skipping\");\n\t\t\treturn;\n\t\t}\n\n\t\tlogDetail(`Lockdown sync to Alpine: source dir = ${lockdownDir}, udid = ${udid}`);\n\n\t\tconst distro = this.wslDistribution || \"alpine-usbmuxd-build\";\n\t\tconst wslDestDir = this.ALPINE_LOCKDOWN_DIR;\n\n\t\t// Sync the device-specific plist\n\t\tconst devicePlistFile = `${udid}.plist`;\n\t\tconst devicePlistPath = join(lockdownDir, devicePlistFile);\n\t\tif (existsSync(devicePlistPath)) {\n\t\t\tawait this.copyFileToAlpine(\n\t\t\t\tthis.windowsPathToWsl(devicePlistPath),\n\t\t\t\twslDestDir,\n\t\t\t\tdistro,\n\t\t\t\tdevicePlistFile\n\t\t\t);\n\t\t} else {\n\t\t\tlogDetail(\n\t\t\t\t`Lockdown sync to Alpine: ${devicePlistFile} not found at ${devicePlistPath}, skipping`\n\t\t\t);\n\t\t}\n\n\t\t// Sync SystemConfiguration.plist so Alpine usbmuxd uses the same SystemBUID\n\t\t// that was used during the original pairing on Windows.\n\t\t// Without this, lockdown sessions fail with InvalidHostID.\n\t\tconst systemConfigPath = join(lockdownDir, this.SYSTEM_CONFIG_PLIST);\n\t\tif (existsSync(systemConfigPath)) {\n\t\t\tawait this.copyFileToAlpine(\n\t\t\t\tthis.windowsPathToWsl(systemConfigPath),\n\t\t\t\twslDestDir,\n\t\t\t\tdistro,\n\t\t\t\tthis.SYSTEM_CONFIG_PLIST\n\t\t\t);\n\t\t} else {\n\t\t\tlogDetail(\n\t\t\t\t`Lockdown sync to Alpine: ${this.SYSTEM_CONFIG_PLIST} not found at ${systemConfigPath}, skipping`\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * Copy a single file into the Alpine lockdown directory.\n\t * Falls back to sudo if the initial copy fails.\n\t */\n\tprivate async copyFileToAlpine(\n\t\twslSource: string,\n\t\twslDestDir: string,\n\t\tdistro: string,\n\t\tfileName: string\n\t): Promise<void> {\n\t\ttry {\n\t\t\tawait execAsync(\n\t\t\t\t`wsl -d ${distro} -- sh -c \"mkdir -p ${wslDestDir} && cp '${wslSource}' '${wslDestDir}/'\"`\n\t\t\t);\n\t\t\tlogInfo(`Lockdown sync: copied ${fileName} to Alpine`);\n\t\t} catch (error) {\n\t\t\ttry {\n\t\t\t\tawait execAsync(\n\t\t\t\t\t`wsl -d ${distro} -- sh -c \"mkdir -p ${wslDestDir} && sudo cp '${wslSource}' '${wslDestDir}/'\"`\n\t\t\t\t);\n\t\t\t\tlogInfo(`Lockdown sync: copied ${fileName} to Alpine (via sudo)`);\n\t\t\t} catch (sudoError) {\n\t\t\t\tlogWarning(\n\t\t\t\t\t`Lockdown sync to Alpine failed for ${fileName}: ${error}. Sudo fallback failed: ${sudoError}. Ensure /var/lib/lockdown is writable or use passwordless sudo.`\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Copy lockdown plists from Alpine back to the native Apple lockdown directory\n\t * (for devices that were assigned this session).\n\t * Also copies SystemConfiguration.plist if present in Alpine.\n\t *\n\t * No-op if the lockdown directory cannot be determined.\n\t */\n\tpublic async syncFromAlpine(udids: string[]): Promise<void> {\n\t\tlogInfo(`Lockdown sync from Alpine: starting (${udids.length} device(s))`);\n\n\t\tconst lockdownDir = this.getAppleLockdownPath();\n\t\tif (!lockdownDir) {\n\t\t\tlogWarning(\"Lockdown sync: unsupported platform, skipping\");\n\t\t\treturn;\n\t\t}\n\n\t\tlogDetail(`Lockdown sync: target dir = ${lockdownDir}`);\n\n\t\ttry {\n\t\t\tmkdirSync(lockdownDir, { recursive: true });\n\t\t} catch (error) {\n\t\t\tlogWarning(`Lockdown sync: could not create lockdown dir ${lockdownDir}: ${error}`);\n\t\t\treturn;\n\t\t}\n\n\t\tconst distro = this.wslDistribution || \"alpine-usbmuxd-build\";\n\t\tconst wslDestDir = this.windowsPathToWsl(lockdownDir);\n\n\t\t// List what Alpine actually has in its lockdown dir for debugging\n\t\ttry {\n\t\t\tconst { stdout } = await execAsync(`wsl -d ${distro} -- ls -la ${this.ALPINE_LOCKDOWN_DIR}/`);\n\t\t\tlogDetail(`Lockdown sync: Alpine ${this.ALPINE_LOCKDOWN_DIR} contents:\\n${stdout}`);\n\t\t} catch (error) {\n\t\t\tlogWarning(`Lockdown sync: could not list Alpine lockdown dir: ${error}`);\n\t\t}\n\n\t\tconst filesToSync = [...udids.map((udid) => `${udid}.plist`), this.SYSTEM_CONFIG_PLIST];\n\n\t\tlogDetail(`Lockdown sync: files to sync = [${filesToSync.join(\", \")}]`);\n\n\t\tfor (const fileName of filesToSync) {\n\t\t\tawait this.copyFileFromAlpine(fileName, wslDestDir, distro, lockdownDir);\n\t\t}\n\n\t\tlogInfo(\"Lockdown sync from Alpine: done\");\n\t}\n\n\t/**\n\t * Copy a single file from the Alpine lockdown directory back to Windows.\n\t * Skips if the file does not exist in Alpine; logs a warning if the copy fails.\n\t */\n\tprivate async copyFileFromAlpine(\n\t\tfileName: string,\n\t\twslDestDir: string,\n\t\tdistro: string,\n\t\tlockdownDir: string\n\t): Promise<void> {\n\t\tconst alpinePath = `${this.ALPINE_LOCKDOWN_DIR}/${fileName}`;\n\n\t\t// Check if the file exists in Alpine first\n\t\ttry {\n\t\t\tawait execAsync(`wsl -d ${distro} -- test -f '${alpinePath}'`);\n\t\t} catch {\n\t\t\tlogDetail(`Lockdown sync: ${fileName} not found in Alpine (${alpinePath}), skipping`);\n\t\t\treturn;\n\t\t}\n\n\t\t// File exists \u2014 copy it to Windows\n\t\tlogDetail(`Lockdown sync: copying ${fileName} \u2192 ${wslDestDir}/`);\n\t\ttry {\n\t\t\tawait execAsync(`wsl -d ${distro} -- cp -f '${alpinePath}' '${wslDestDir}/'`);\n\t\t\tlogInfo(`Lockdown sync: copied ${fileName} from Alpine to ${lockdownDir}`);\n\t\t} catch (error) {\n\t\t\tlogWarning(`Lockdown sync: failed to copy ${fileName} from Alpine to ${lockdownDir}: ${error}`);\n\t\t}\n\t}\n}\n"],
5
- "mappings": "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",
6
- "names": ["exports", "module", "exports", "module", "options", "path", "config", "import_tool_debug_g4", "import_node_events", "import_tool_debug_g4", "import_tool_debug_g4", "import_node_child_process", "import_node_util", "import_tool_debug_g4", "logInfo", "logWarning", "execAsync", "process", "logWarning", "logDetail", "config", "process", "logInfo", "logDataObject", "config", "UsbDeviceListener", "dotenv", "logInfo", "logError", "logDataObject"]
4
+ "sourcesContent": ["{\n \"name\": \"dotenv\",\n \"version\": \"17.2.3\",\n \"description\": \"Loads environment variables from .env file\",\n \"main\": \"lib/main.js\",\n \"types\": \"lib/main.d.ts\",\n \"exports\": {\n \".\": {\n \"types\": \"./lib/main.d.ts\",\n \"require\": \"./lib/main.js\",\n \"default\": \"./lib/main.js\"\n },\n \"./config\": \"./config.js\",\n \"./config.js\": \"./config.js\",\n \"./lib/env-options\": \"./lib/env-options.js\",\n \"./lib/env-options.js\": \"./lib/env-options.js\",\n \"./lib/cli-options\": \"./lib/cli-options.js\",\n \"./lib/cli-options.js\": \"./lib/cli-options.js\",\n \"./package.json\": \"./package.json\"\n },\n \"scripts\": {\n \"dts-check\": \"tsc --project tests/types/tsconfig.json\",\n \"lint\": \"standard\",\n \"pretest\": \"npm run lint && npm run dts-check\",\n \"test\": \"tap run tests/**/*.js --allow-empty-coverage --disable-coverage --timeout=60000\",\n \"test:coverage\": \"tap run tests/**/*.js --show-full-coverage --timeout=60000 --coverage-report=text --coverage-report=lcov\",\n \"prerelease\": \"npm test\",\n \"release\": \"standard-version\"\n },\n \"repository\": {\n \"type\": \"git\",\n \"url\": \"git://github.com/motdotla/dotenv.git\"\n },\n \"homepage\": \"https://github.com/motdotla/dotenv#readme\",\n \"funding\": \"https://dotenvx.com\",\n \"keywords\": [\n \"dotenv\",\n \"env\",\n \".env\",\n \"environment\",\n \"variables\",\n \"config\",\n \"settings\"\n ],\n \"readmeFilename\": \"README.md\",\n \"license\": \"BSD-2-Clause\",\n \"devDependencies\": {\n \"@types/node\": \"^18.11.3\",\n \"decache\": \"^4.6.2\",\n \"sinon\": \"^14.0.1\",\n \"standard\": \"^17.0.0\",\n \"standard-version\": \"^9.5.0\",\n \"tap\": \"^19.2.0\",\n \"typescript\": \"^4.8.4\"\n },\n \"engines\": {\n \"node\": \">=12\"\n },\n \"browser\": {\n \"fs\": false\n }\n}\n", "const fs = require('fs')\nconst path = require('path')\nconst os = require('os')\nconst crypto = require('crypto')\nconst packageJson = require('../package.json')\n\nconst version = packageJson.version\n\n// Array of tips to display randomly\nconst TIPS = [\n '\uD83D\uDD10 encrypt with Dotenvx: https://dotenvx.com',\n '\uD83D\uDD10 prevent committing .env to code: https://dotenvx.com/precommit',\n '\uD83D\uDD10 prevent building .env in docker: https://dotenvx.com/prebuild',\n '\uD83D\uDCE1 add observability to secrets: https://dotenvx.com/ops',\n '\uD83D\uDC65 sync secrets across teammates & machines: https://dotenvx.com/ops',\n '\uD83D\uDDC2\uFE0F backup and recover secrets: https://dotenvx.com/ops',\n '\u2705 audit secrets and track compliance: https://dotenvx.com/ops',\n '\uD83D\uDD04 add secrets lifecycle management: https://dotenvx.com/ops',\n '\uD83D\uDD11 add access controls to secrets: https://dotenvx.com/ops',\n '\uD83D\uDEE0\uFE0F run anywhere with `dotenvx run -- yourcommand`',\n '\u2699\uFE0F specify custom .env file path with { path: \\'/custom/path/.env\\' }',\n '\u2699\uFE0F enable debug logging with { debug: true }',\n '\u2699\uFE0F override existing env vars with { override: true }',\n '\u2699\uFE0F suppress all logs with { quiet: true }',\n '\u2699\uFE0F write to custom object with { processEnv: myObject }',\n '\u2699\uFE0F load multiple .env files with { path: [\\'.env.local\\', \\'.env\\'] }'\n]\n\n// Get a random tip from the tips array\nfunction _getRandomTip () {\n return TIPS[Math.floor(Math.random() * TIPS.length)]\n}\n\nfunction parseBoolean (value) {\n if (typeof value === 'string') {\n return !['false', '0', 'no', 'off', ''].includes(value.toLowerCase())\n }\n return Boolean(value)\n}\n\nfunction supportsAnsi () {\n return process.stdout.isTTY // && process.env.TERM !== 'dumb'\n}\n\nfunction dim (text) {\n return supportsAnsi() ? `\\x1b[2m${text}\\x1b[0m` : text\n}\n\nconst LINE = /(?:^|^)\\s*(?:export\\s+)?([\\w.-]+)(?:\\s*=\\s*?|:\\s+?)(\\s*'(?:\\\\'|[^'])*'|\\s*\"(?:\\\\\"|[^\"])*\"|\\s*`(?:\\\\`|[^`])*`|[^#\\r\\n]+)?\\s*(?:#.*)?(?:$|$)/mg\n\n// Parse src into an Object\nfunction parse (src) {\n const obj = {}\n\n // Convert buffer to string\n let lines = src.toString()\n\n // Convert line breaks to same format\n lines = lines.replace(/\\r\\n?/mg, '\\n')\n\n let match\n while ((match = LINE.exec(lines)) != null) {\n const key = match[1]\n\n // Default undefined or null to empty string\n let value = (match[2] || '')\n\n // Remove whitespace\n value = value.trim()\n\n // Check if double quoted\n const maybeQuote = value[0]\n\n // Remove surrounding quotes\n value = value.replace(/^(['\"`])([\\s\\S]*)\\1$/mg, '$2')\n\n // Expand newlines if double quoted\n if (maybeQuote === '\"') {\n value = value.replace(/\\\\n/g, '\\n')\n value = value.replace(/\\\\r/g, '\\r')\n }\n\n // Add to object\n obj[key] = value\n }\n\n return obj\n}\n\nfunction _parseVault (options) {\n options = options || {}\n\n const vaultPath = _vaultPath(options)\n options.path = vaultPath // parse .env.vault\n const result = DotenvModule.configDotenv(options)\n if (!result.parsed) {\n const err = new Error(`MISSING_DATA: Cannot parse ${vaultPath} for an unknown reason`)\n err.code = 'MISSING_DATA'\n throw err\n }\n\n // handle scenario for comma separated keys - for use with key rotation\n // example: DOTENV_KEY=\"dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=prod,dotenv://:key_7890@dotenvx.com/vault/.env.vault?environment=prod\"\n const keys = _dotenvKey(options).split(',')\n const length = keys.length\n\n let decrypted\n for (let i = 0; i < length; i++) {\n try {\n // Get full key\n const key = keys[i].trim()\n\n // Get instructions for decrypt\n const attrs = _instructions(result, key)\n\n // Decrypt\n decrypted = DotenvModule.decrypt(attrs.ciphertext, attrs.key)\n\n break\n } catch (error) {\n // last key\n if (i + 1 >= length) {\n throw error\n }\n // try next key\n }\n }\n\n // Parse decrypted .env string\n return DotenvModule.parse(decrypted)\n}\n\nfunction _warn (message) {\n console.error(`[dotenv@${version}][WARN] ${message}`)\n}\n\nfunction _debug (message) {\n console.log(`[dotenv@${version}][DEBUG] ${message}`)\n}\n\nfunction _log (message) {\n console.log(`[dotenv@${version}] ${message}`)\n}\n\nfunction _dotenvKey (options) {\n // prioritize developer directly setting options.DOTENV_KEY\n if (options && options.DOTENV_KEY && options.DOTENV_KEY.length > 0) {\n return options.DOTENV_KEY\n }\n\n // secondary infra already contains a DOTENV_KEY environment variable\n if (process.env.DOTENV_KEY && process.env.DOTENV_KEY.length > 0) {\n return process.env.DOTENV_KEY\n }\n\n // fallback to empty string\n return ''\n}\n\nfunction _instructions (result, dotenvKey) {\n // Parse DOTENV_KEY. Format is a URI\n let uri\n try {\n uri = new URL(dotenvKey)\n } catch (error) {\n if (error.code === 'ERR_INVALID_URL') {\n const err = new Error('INVALID_DOTENV_KEY: Wrong format. Must be in valid uri format like dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=development')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n throw error\n }\n\n // Get decrypt key\n const key = uri.password\n if (!key) {\n const err = new Error('INVALID_DOTENV_KEY: Missing key part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get environment\n const environment = uri.searchParams.get('environment')\n if (!environment) {\n const err = new Error('INVALID_DOTENV_KEY: Missing environment part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get ciphertext payload\n const environmentKey = `DOTENV_VAULT_${environment.toUpperCase()}`\n const ciphertext = result.parsed[environmentKey] // DOTENV_VAULT_PRODUCTION\n if (!ciphertext) {\n const err = new Error(`NOT_FOUND_DOTENV_ENVIRONMENT: Cannot locate environment ${environmentKey} in your .env.vault file.`)\n err.code = 'NOT_FOUND_DOTENV_ENVIRONMENT'\n throw err\n }\n\n return { ciphertext, key }\n}\n\nfunction _vaultPath (options) {\n let possibleVaultPath = null\n\n if (options && options.path && options.path.length > 0) {\n if (Array.isArray(options.path)) {\n for (const filepath of options.path) {\n if (fs.existsSync(filepath)) {\n possibleVaultPath = filepath.endsWith('.vault') ? filepath : `${filepath}.vault`\n }\n }\n } else {\n possibleVaultPath = options.path.endsWith('.vault') ? options.path : `${options.path}.vault`\n }\n } else {\n possibleVaultPath = path.resolve(process.cwd(), '.env.vault')\n }\n\n if (fs.existsSync(possibleVaultPath)) {\n return possibleVaultPath\n }\n\n return null\n}\n\nfunction _resolveHome (envPath) {\n return envPath[0] === '~' ? path.join(os.homedir(), envPath.slice(1)) : envPath\n}\n\nfunction _configVault (options) {\n const debug = parseBoolean(process.env.DOTENV_CONFIG_DEBUG || (options && options.debug))\n const quiet = parseBoolean(process.env.DOTENV_CONFIG_QUIET || (options && options.quiet))\n\n if (debug || !quiet) {\n _log('Loading env from encrypted .env.vault')\n }\n\n const parsed = DotenvModule._parseVault(options)\n\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n\n DotenvModule.populate(processEnv, parsed, options)\n\n return { parsed }\n}\n\nfunction configDotenv (options) {\n const dotenvPath = path.resolve(process.cwd(), '.env')\n let encoding = 'utf8'\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n let debug = parseBoolean(processEnv.DOTENV_CONFIG_DEBUG || (options && options.debug))\n let quiet = parseBoolean(processEnv.DOTENV_CONFIG_QUIET || (options && options.quiet))\n\n if (options && options.encoding) {\n encoding = options.encoding\n } else {\n if (debug) {\n _debug('No encoding is specified. UTF-8 is used by default')\n }\n }\n\n let optionPaths = [dotenvPath] // default, look for .env\n if (options && options.path) {\n if (!Array.isArray(options.path)) {\n optionPaths = [_resolveHome(options.path)]\n } else {\n optionPaths = [] // reset default\n for (const filepath of options.path) {\n optionPaths.push(_resolveHome(filepath))\n }\n }\n }\n\n // Build the parsed data in a temporary object (because we need to return it). Once we have the final\n // parsed data, we will combine it with process.env (or options.processEnv if provided).\n let lastError\n const parsedAll = {}\n for (const path of optionPaths) {\n try {\n // Specifying an encoding returns a string instead of a buffer\n const parsed = DotenvModule.parse(fs.readFileSync(path, { encoding }))\n\n DotenvModule.populate(parsedAll, parsed, options)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${path} ${e.message}`)\n }\n lastError = e\n }\n }\n\n const populated = DotenvModule.populate(processEnv, parsedAll, options)\n\n // handle user settings DOTENV_CONFIG_ options inside .env file(s)\n debug = parseBoolean(processEnv.DOTENV_CONFIG_DEBUG || debug)\n quiet = parseBoolean(processEnv.DOTENV_CONFIG_QUIET || quiet)\n\n if (debug || !quiet) {\n const keysCount = Object.keys(populated).length\n const shortPaths = []\n for (const filePath of optionPaths) {\n try {\n const relative = path.relative(process.cwd(), filePath)\n shortPaths.push(relative)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${filePath} ${e.message}`)\n }\n lastError = e\n }\n }\n\n _log(`injecting env (${keysCount}) from ${shortPaths.join(',')} ${dim(`-- tip: ${_getRandomTip()}`)}`)\n }\n\n if (lastError) {\n return { parsed: parsedAll, error: lastError }\n } else {\n return { parsed: parsedAll }\n }\n}\n\n// Populates process.env from .env file\nfunction config (options) {\n // fallback to original dotenv if DOTENV_KEY is not set\n if (_dotenvKey(options).length === 0) {\n return DotenvModule.configDotenv(options)\n }\n\n const vaultPath = _vaultPath(options)\n\n // dotenvKey exists but .env.vault file does not exist\n if (!vaultPath) {\n _warn(`You set DOTENV_KEY but you are missing a .env.vault file at ${vaultPath}. Did you forget to build it?`)\n\n return DotenvModule.configDotenv(options)\n }\n\n return DotenvModule._configVault(options)\n}\n\nfunction decrypt (encrypted, keyStr) {\n const key = Buffer.from(keyStr.slice(-64), 'hex')\n let ciphertext = Buffer.from(encrypted, 'base64')\n\n const nonce = ciphertext.subarray(0, 12)\n const authTag = ciphertext.subarray(-16)\n ciphertext = ciphertext.subarray(12, -16)\n\n try {\n const aesgcm = crypto.createDecipheriv('aes-256-gcm', key, nonce)\n aesgcm.setAuthTag(authTag)\n return `${aesgcm.update(ciphertext)}${aesgcm.final()}`\n } catch (error) {\n const isRange = error instanceof RangeError\n const invalidKeyLength = error.message === 'Invalid key length'\n const decryptionFailed = error.message === 'Unsupported state or unable to authenticate data'\n\n if (isRange || invalidKeyLength) {\n const err = new Error('INVALID_DOTENV_KEY: It must be 64 characters long (or more)')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n } else if (decryptionFailed) {\n const err = new Error('DECRYPTION_FAILED: Please check your DOTENV_KEY')\n err.code = 'DECRYPTION_FAILED'\n throw err\n } else {\n throw error\n }\n }\n}\n\n// Populate process.env with parsed values\nfunction populate (processEnv, parsed, options = {}) {\n const debug = Boolean(options && options.debug)\n const override = Boolean(options && options.override)\n const populated = {}\n\n if (typeof parsed !== 'object') {\n const err = new Error('OBJECT_REQUIRED: Please check the processEnv argument being passed to populate')\n err.code = 'OBJECT_REQUIRED'\n throw err\n }\n\n // Set process.env\n for (const key of Object.keys(parsed)) {\n if (Object.prototype.hasOwnProperty.call(processEnv, key)) {\n if (override === true) {\n processEnv[key] = parsed[key]\n populated[key] = parsed[key]\n }\n\n if (debug) {\n if (override === true) {\n _debug(`\"${key}\" is already defined and WAS overwritten`)\n } else {\n _debug(`\"${key}\" is already defined and was NOT overwritten`)\n }\n }\n } else {\n processEnv[key] = parsed[key]\n populated[key] = parsed[key]\n }\n }\n\n return populated\n}\n\nconst DotenvModule = {\n configDotenv,\n _configVault,\n _parseVault,\n config,\n decrypt,\n parse,\n populate\n}\n\nmodule.exports.configDotenv = DotenvModule.configDotenv\nmodule.exports._configVault = DotenvModule._configVault\nmodule.exports._parseVault = DotenvModule._parseVault\nmodule.exports.config = DotenvModule.config\nmodule.exports.decrypt = DotenvModule.decrypt\nmodule.exports.parse = DotenvModule.parse\nmodule.exports.populate = DotenvModule.populate\n\nmodule.exports = DotenvModule\n", "#!/usr/bin/env node\n\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport dotenv from \"dotenv\";\nimport { UsbmuxdService } from \"./UsbmuxdService.js\";\ndotenv.config();\n\nconst { logInfo, logError, logHeader, logDataObject } = createLoggers(\"usbmuxd-cli\");\n\n// Parse command line arguments\nconst args = process.argv.slice(2);\nconst options: Record<string, string> = {};\n\nfor (let i = 0; i < args.length; i++) {\n\tif (args[i].startsWith(\"--\")) {\n\t\tconst key = args[i].substring(2);\n\t\tconst value = args[i + 1];\n\t\tif (value && !value.startsWith(\"--\")) {\n\t\t\toptions[key] = value;\n\t\t\ti++;\n\t\t}\n\t}\n}\n\n// Build configuration from command line\n// usbmuxdPath is the binary run inside WSL (Alpine) \u2014 must be the Linux usbmuxd, not the Windows exe\nconst config = {\n\tbatchSize: options.batchSize ? Number.parseInt(options.batchSize, 10) : 4,\n\tbasePort: options.basePort ? Number.parseInt(options.basePort, 10) : 27015,\n\tmaxInstances: options.maxInstances ? Number.parseInt(options.maxInstances, 10) : 20,\n\tusbmuxdPath: options.usbmuxdPath || \"usbmuxd\",\n\tverboseLogging: options.verbose !== \"false\",\n\tappleVendorId: options.appleVid || \"05AC\",\n};\n\nlogHeader(\"usbmuxd Instance Manager\");\nlogDataObject(\"Multi-instance manager for iOS device connections\", { config });\n\n// Create and start service\nconst service = new UsbmuxdService(config);\n\nasync function shutdown(eventName?: string) {\n\tlogInfo(\"\");\n\tlogInfo(`Received ${eventName}, shutting down gracefully...`);\n\ttry {\n\t\tawait service.stop();\n\t\tlogInfo(\"Shutdown complete\");\n\t\tprocess.exit(0);\n\t} catch (error) {\n\t\tlogError(`Error during ${eventName}:`, error);\n\t\tprocess.exit(1);\n\t}\n}\n\n// Handle graceful shutdown\nprocess.on(\"SIGINT\", async () => {\n\tvoid shutdown(\"SIGINT\");\n});\n\nprocess.on(\"SIGTERM\", async () => {\n\tvoid shutdown(\"SIGTERM\");\n});\n\n// Print stats every 30 seconds\nsetInterval(() => {\n\tconst stats = service.getStats();\n\tconst instances = service.getInstances();\n\n\tlogHeader(\n\t\t`Uptime: ${stats.uptimeSeconds}s | Instances: ${stats.instanceCount} | Devices: ${stats.deviceCount}`\n\t);\n\n\tif (instances.length > 0) {\n\t\tlogDataObject(\"Active instances:\", { instances });\n\t}\n}, 30000);\n\n// Print help if --help flag (before starting service)\nif (args.includes(\"--help\") || args.includes(\"-h\")) {\n\tlogInfo(\"Usage: usbmuxd-instance-manager [OPTIONS]\");\n\tlogInfo(\"\");\n\tlogInfo(\"Options:\");\n\tlogInfo(\" --batchSize <n> Number of devices per instance (default: 4)\");\n\tlogInfo(\" --basePort <port> Base TCP port for first instance (default: 27015)\");\n\tlogInfo(\" --maxInstances <n> Maximum number of instances (default: 20)\");\n\tlogInfo(\" --usbmuxdPath <path> Path to usbmuxd executable\");\n\tlogInfo(\" --verbose <true|false> Enable verbose logging (default: true)\");\n\tlogInfo(\" --appleVid <vid> Apple Vendor ID in hex (default: 05AC)\");\n\tlogInfo(\" -h, --help Show this help message\");\n\tlogInfo(\"\");\n\tlogInfo(\"Examples:\");\n\tlogInfo(\" usbmuxd-instance-manager\");\n\tlogInfo(\" usbmuxd-instance-manager --batchSize 6 --basePort 27020\");\n\tlogInfo(\"\");\n\tprocess.exit(0);\n}\n\n// Start the service\ntry {\n\tservice.start();\n} catch (error) {\n\tlogError(\"Failed to start service\", error);\n\tshutdown(\"Failed to start service\");\n}\n", "import { EventEmitter } from \"node:events\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type { DeviceInfo } from \"@mcesystems/usb-device-listener\";\nimport { InstanceManager } from \"./InstanceManager.js\";\nimport type { DeviceMapping, InstanceManagerConfig, UsbmuxdInstance } from \"./types/index.js\";\n\nconst { logInfo, logError, logDataObject } = createLoggers(\"usbmuxd-service\");\n\n/**\n * Main service that integrates usb-device-listener with InstanceManager\n * Monitors iOS devices and manages usbmuxd instances automatically.\n * Extends EventEmitter: forwards instance-started, instance-stopped, device-assigned,\n * device-removed, instance-log, instance-exited from the underlying manager.\n */\nexport class UsbmuxdService extends EventEmitter {\n\tprivate manager: InstanceManager;\n\tprivate isListening = false;\n\n\tconstructor(config: Partial<InstanceManagerConfig> = {}) {\n\t\tsuper();\n\t\tthis.manager = new InstanceManager(config);\n\t\tthis.setupEventHandlers();\n\t}\n\n\t/**\n\t * Forward manager events to service subscribers (no logging; CLI/examples handle logging).\n\t */\n\tprivate setupEventHandlers(): void {\n\t\tthis.manager.on(\"instance-started\", (instance: UsbmuxdInstance) => {\n\t\t\tthis.emit(\"instance-started\", instance);\n\t\t});\n\n\t\tthis.manager.on(\"instance-stopped\", (instance: UsbmuxdInstance) => {\n\t\t\tthis.emit(\"instance-stopped\", instance);\n\t\t});\n\n\t\tthis.manager.on(\n\t\t\t\"device-assigned\",\n\t\t\t(payload: { device: DeviceInfo; instance: UsbmuxdInstance; mapping: DeviceMapping; }) => {\n\t\t\t\tlogInfo(\n\t\t\t\t\t`Device ${payload.device.deviceId} connected to usbmuxd instance at ${payload.mapping.host}:${payload.mapping.port}`\n\t\t\t\t);\n\t\t\t\tthis.emit(\"device-assigned\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\n\t\t\t\"device-removed\",\n\t\t\t(payload: { device: DeviceInfo; instance: UsbmuxdInstance; }) => {\n\t\t\t\tthis.emit(\"device-removed\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\n\t\t\t\"instance-log\",\n\t\t\t(payload: { instanceId: number; level: string; message: string; }) => {\n\t\t\t\tthis.emit(\"instance-log\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\n\t\t\t\"instance-exited\",\n\t\t\t(payload: { instanceId: number; code: number | null; signal: string | null; }) => {\n\t\t\t\tthis.emit(\"instance-exited\", payload);\n\t\t\t}\n\t\t);\n\n\t\tthis.manager.on(\"device-paired\", (payload: { udid: string; mapping: DeviceMapping; }) => {\n\t\t\tthis.emit(\"device-paired\", payload);\n\t\t});\n\t}\n\n\t/**\n\t * Start the service\n\t * Begins monitoring for iOS devices\n\t */\n\tpublic start() {\n\t\tif (this.isListening) {\n\t\t\tthrow new Error(\"Service is already running\");\n\t\t}\n\n\t\tconst config = this.manager.getConfig();\n\n\t\tlogDataObject(\"Starting service...\", { config });\n\n\t\t// Start listening for USB events\n\t\ttry {\n\n\t\t\tthis.isListening = true;\n\t\t\tthis.manager.start();\n\n\t\t\tlogInfo(\"Service started successfully\");\n\t\t\tlogInfo(\"Waiting for iOS devices...\");\n\t\t\tconst onAdd = this.manager.onDeviceConnected.bind(this.manager);\n\t\t\tconst onRemove = this.manager.onDeviceDisconnected.bind(this.manager);\n\t\t\treturn {\n\t\t\t\tonAdd,\n\t\t\t\tonRemove,\n\t\t\t};\n\t\t} catch (error) {\n\t\t\tlogError(\"Failed to start USB listener:\", error);\n\t\t\tthrow error;\n\t\t}\n\t}\n\n\t/**\n\t * Stop the service\n\t * Stops monitoring and terminates all instances\n\t */\n\tpublic async stop(): Promise<void> {\n\t\tif (!this.isListening) {\n\t\t\treturn;\n\t\t}\n\n\t\tlogInfo(\"Stopping service...\");\n\n\t\tthis.isListening = false;\n\n\t\t// Stop all instances\n\t\tawait this.manager.stop();\n\n\t\tlogInfo(\"Service stopped\");\n\t}\n\n\t/**\n\t * Get device port mapping\n\t */\n\tpublic getDevicePort(udid: string): number | null {\n\t\treturn this.manager.getDevicePort(udid);\n\t}\n\n\t/**\n\t * Get all device mappings\n\t */\n\tpublic getDeviceMappings() {\n\t\treturn this.manager.getDeviceMappings();\n\t}\n\n\t/**\n\t * Get all running instances\n\t */\n\tpublic getInstances() {\n\t\treturn this.manager.getInstances();\n\t}\n\n\t/**\n\t * Get service statistics\n\t */\n\tpublic getStats() {\n\t\treturn this.manager.getStats();\n\t}\n\n\t/**\n\t * Get current configuration\n\t */\n\tpublic getConfig() {\n\t\treturn this.manager.getConfig();\n\t}\n}\n", "import type { ChildProcess } from \"node:child_process\";\nimport { EventEmitter } from \"node:events\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type {\n\tDeviceInfo,\n\tDeviceMapping,\n\tInstanceManagerConfig,\n\tManagerStats,\n\tUsbmuxdInstance,\n} from \"./types/index.js\";\nimport { Usbipd } from \"./usbipd.js\";\nimport { Wsl2 } from \"./wsl.js\";\n\nconst { logWarning, logDetail, logDataObject, logTask } = createLoggers(\"instance-manager\");\n\n/**\n * Default configuration for the instance manager\n */\nconst DEFAULT_CONFIG: InstanceManagerConfig = {\n\tbatchSize: 4,\n\tbasePort: 27015,\n\tmaxInstances: 5,\n\tusbmuxdPath: \"usbmuxd\", // Path inside WSL2\n\twslDistribution: \"alpine-usbmuxd-build\", // Alpine WSL2 distribution name\n\tverboseLogging: true,\n\tappleVendorId: \"05AC\",\n};\n\n/**\n * Manages multiple usbmuxd instances with batch device allocation\n *\n * @example\n * const manager = new InstanceManager({\n * batchSize: 4,\n * basePort: 27015,\n * });\n *\n * manager.on('instance-started', (instance) => {\n * logInfo(`Instance ${instance.id} started on port ${instance.port}`);\n * });\n *\n * manager.start();\n */\nexport class InstanceManager extends EventEmitter {\n\t/** Map of usbmuxd instance IDs to instances */\n\tprivate instances: Map<number, UsbmuxdInstance> = new Map();\n\t/** Map of device IDs to device mappings host:port */\n\tprivate deviceMappings: Map<string, DeviceMapping> = new Map();\n\t/** Map of usbmuxd instance IDs to child processes running it on wsl2*/\n\tprivate processes: Map<number, ChildProcess> = new Map();\n\t/** Tracks which devices have been attached to WSL */\n\tprivate attachedDevices: Set<string> = new Set();\n\t/** Device IDs currently in the attach flow\n\t * (ignore disconnect until attach completes) */\n\tprivate pendingAttachDevices: Set<string> = new Set();\n\n\tprivate config: InstanceManagerConfig;\n\tprivate nextInstanceId = 1;\n\tprivate startedAt: Date | null = null;\n\tprivate isRunning = false;\n\n\tprivate wsl: Wsl2;\n\tprivate usbipd: Usbipd;\n\tconstructor(config: Partial<InstanceManagerConfig> = {}) {\n\t\tsuper();\n\t\tthis.config = { ...DEFAULT_CONFIG, ...config };\n\t\tthis.wsl = new Wsl2(this.config.wslDistribution);\n\t\tthis.usbipd = new Usbipd(process.env.USBIPD_PATH ?? \"usbipd\");\n\t}\n\n\t/**\n\t * Start the instance manager\n\t */\n\tpublic start(): void {\n\t\tif (this.isRunning) {\n\t\t\tthrow new Error(\"Instance manager is already running\");\n\t\t}\n\n\t\tthis.isRunning = true;\n\t\tthis.startedAt = new Date();\n\t}\n\n\t/**\n\t * Stop the instance manager and all instances\n\t */\n\tpublic async stop(): Promise<void> {\n\t\tif (!this.isRunning) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis.isRunning = false;\n\n\t\t// Sync lockdown files from Alpine back to native Apple lockdown directory (before clearing mappings)\n\t\tconst udids = Array.from(this.deviceMappings.keys());\n\t\tawait this.wsl.syncFromAlpine(udids);\n\n\t\t// Detach all devices from WSL.\n\t\t// If you only called unbind without detach,\n\t\t// the device might still be attached to the WSL session\n\t\tawait this.usbipd.detachAllDevicesFromWsl();\n\t\tawait this.usbipd.unbindAllDevicesFromWsl();\n\t\tthis.attachedDevices.clear();\n\n\t\t// Stop all instances\n\t\tconst stopPromises = Array.from(this.instances.keys()).map((id) => this.stopInstance(id));\n\n\t\tawait Promise.all(stopPromises);\n\n\t\tthis.instances.clear();\n\t\tthis.deviceMappings.clear();\n\t\tthis.processes.clear();\n\t}\n\n\tprivate async attachToWsl({\n\t\tbusId,\n\t\tdeviceId,\n\t}: { busId: string; deviceId: string }): Promise<void> {\n\t\tif (!this.attachedDevices.has(busId)) {\n\t\t\tthis.pendingAttachDevices.add(deviceId);\n\t\t\tif (!this.config.wslDistribution) {\n\t\t\t\tthrow new Error(\"WSL distribution not configured\");\n\t\t\t}\n\t\t\tconst attached = await this.usbipd.attachDeviceToWsl(\n\t\t\t\tbusId,\n\t\t\t\tthis.wsl,\n\t\t\t\tthis.config.wslDistribution\n\t\t\t);\n\t\t\tif (attached) {\n\t\t\t\tthis.attachedDevices.add(busId);\n\t\t\t}\n\t\t}\n\t}\n\n\tprivate async removeDevice(mapping: DeviceMapping): Promise<UsbmuxdInstance> {\n\t\tconst instance = await this.detachFromWsl({ deviceId: mapping.udid });\n\t\tif (!instance) {\n\t\t\tthrow new Error(`Instance ${mapping.instanceId} not found`);\n\t\t}\n\t\tif (mapping.busId) {\n\t\t\tawait this.usbipd.detachDeviceFromWsl(mapping.busId);\n\t\t\tthis.attachedDevices.delete(mapping.busId);\n\t\t}\n\t\tconst deviceIndex = instance.deviceUdids.indexOf(mapping.udid);\n\t\tif (deviceIndex > -1) {\n\t\t\tinstance.deviceUdids.splice(deviceIndex, 1);\n\t\t}\n\n\t\tthis.deviceMappings.delete(mapping.udid);\n\t\treturn instance;\n\t}\n\n\tprivate async onUsbmuxdInstanceEnd({\n\t\tinstanceId,\n\t\tcode,\n\t\tsignal,\n\t}: { instanceId: number; code: number; signal: string }): Promise<void> {\n\t\tthis.emit(\"instance-exited\", {\n\t\t\tinstanceId,\n\t\t\tcode,\n\t\t\tsignal,\n\t\t});\n\t\tfor (const [_, mapping] of this.deviceMappings.entries()) {\n\t\t\tif (mapping.instanceId === instanceId) {\n\t\t\t\tthis.removeDevice(mapping);\n\t\t\t}\n\t\t}\n\t\tthis.instances.delete(instanceId);\n\t\tthis.processes.get(instanceId)?.kill();\n\t\tthis.processes.delete(instanceId);\n\t}\n\n\tprivate async onUsbmuxdInstanceStart({\n\t\tinstance,\n\t\tprocess,\n\t}: { instance: UsbmuxdInstance; process: ChildProcess }): Promise<void> {\n\t\tthis.instances.set(instance.id, instance);\n\t\tthis.processes.set(instance.id, process);\n\t\tthis.emit(\"instance-started\", {\n\t\t\tinstanceId: instance.id,\n\t\t});\n\t}\n\n\tprivate async getInstance() {\n\t\tlet targetInstance = this.findInstanceWithCapacity();\n\n\t\t// If no instance has capacity, create a new one\n\t\tif (!targetInstance) {\n\t\t\tif (this.instances.size >= this.config.maxInstances) {\n\t\t\t\tthrow new Error(`Maximum number of instances (${this.config.maxInstances}) reached`);\n\t\t\t}\n\n\t\t\tconst newInstanceId = this.nextInstanceId++;\n\n\t\t\tlet onExit = (code: number, signal: string) => {\n\t\t\t\tthis.onUsbmuxdInstanceEnd({ instanceId: newInstanceId, code, signal });\n\t\t\t};\n\n\t\t\tonExit = onExit.bind(this);\n\n\t\t\tconst { instance, process } = await this.wsl.createInstance({\n\t\t\t\tid: newInstanceId,\n\t\t\t\tbasePort: this.config.basePort,\n\t\t\t\tverboseLogging: this.config.verboseLogging,\n\t\t\t\tusbmuxdPath: this.config.usbmuxdPath,\n\t\t\t\tonLog: (message) =>\n\t\t\t\t\tthis.emit(\"instance-log\", {\n\t\t\t\t\t\tinstanceId: newInstanceId,\n\t\t\t\t\t\tlevel: \"info\",\n\t\t\t\t\t\tmessage,\n\t\t\t\t\t}),\n\t\t\t\tonExit,\n\t\t\t});\n\t\t\tthis.onUsbmuxdInstanceStart({ instance, process });\n\t\t\ttargetInstance = instance;\n\t\t}\n\t\treturn targetInstance;\n\t}\n\n\t/**\n\t * Handle device connection only for not attached devices\n\t * Attaches device to WSL, then assigns to an existing instance or creates a new one\n\t */\n\tpublic async onDeviceConnected(device: DeviceInfo): Promise<void> {\n\t\tconst busId = await this.usbipd.findBusIdForDevice(device);\n\n\t\tif (this.attachedDevices.has(busId)) {\n\t\t\tlogTask(`Device ${device.deviceId} is already attached. skipping onDeviceConnected...`);\n\t\t\treturn;\n\t\t}\n\n\t\tlogDataObject(\"Device connected\", { device });\n\t\tawait this.wsl.syncToAlpine(device.deviceId);\n\n\t\tawait this.attachToWsl({ busId, deviceId: device.deviceId });\n\n\t\tconst targetInstance = await this.getInstance();\n\n\t\t// Create mapping (include host and busId for cleanup on disconnect)\n\t\tconst mapping: DeviceMapping = {\n\t\t\tudid: device.deviceId,\n\t\t\tinstanceId: targetInstance.id,\n\t\t\thost: targetInstance.host,\n\t\t\tport: targetInstance.port,\n\t\t\taddedAt: new Date(),\n\t\t\tbusId: busId ?? undefined,\n\t\t};\n\n\t\tthis.deviceMappings.set(device.deviceId, mapping);\n\n\t\t// Add device to instance\n\t\ttargetInstance.deviceUdids.push(device.deviceId);\n\n\t\tthis.emit(\"device-assigned\", {\n\t\t\tdevice,\n\t\t\tinstance: targetInstance,\n\t\t\tmapping,\n\t\t});\n\n\t\tthis.pendingAttachDevices.delete(device.deviceId);\n\t}\n\n\tprivate async detachFromWsl({ deviceId }: { deviceId: string }): Promise<UsbmuxdInstance> {\n\t\tconst mapping = this.deviceMappings.get(deviceId);\n\t\tif (!mapping) {\n\t\t\tthrow new Error(`Device ${deviceId} was not tracked`);\n\t\t}\n\n\t\t// Detach device from WSL if it was attached\n\t\tif (mapping.busId) {\n\t\t\tawait this.usbipd.detachDeviceFromWsl(mapping.busId);\n\t\t\tthis.attachedDevices.delete(mapping.busId);\n\t\t}\n\n\t\tconst instance = this.instances.get(mapping.instanceId);\n\t\tif (!instance) {\n\t\t\tlogWarning(`Instance ${mapping.instanceId} not found`);\n\t\t\tthis.deviceMappings.delete(deviceId);\n\t\t\tthrow new Error(`Instance ${mapping.instanceId} not found`);\n\t\t}\n\n\t\treturn instance;\n\t}\n\n\t/**\n\t * Handle device disconnection only for attached devices\n\t * Detaches from WSL, removes device from instance, and stops instance if empty\n\t */\n\tpublic async onDeviceDisconnected(device: DeviceInfo): Promise<void> {\n\t\t// Ignore disconnect while we're still attaching this device (Windows sees disconnect when device moves to WSL)\n\t\tif (!this.attachedDevices.has(device.deviceId)) {\n\t\t\tlogTask(`Device ${device.deviceId} is not attached. skipping onDeviceDisconnected...`);\n\t\t\treturn;\n\t\t}\n\t\tlogDataObject(\"Device disconnected\", { device });\n\n\t\tconst mapping = this.deviceMappings.get(device.deviceId);\n\t\tif (!mapping) {\n\t\t\tthrow new Error(`Device ${device.deviceId} was not tracked`);\n\t\t}\n\t\t// Remove device from instance\n\t\tconst instance = await this.removeDevice(mapping);\n\n\t\tthis.emit(\"device-removed\", {\n\t\t\tdevice,\n\t\t\tinstance,\n\t\t});\n\n\t\t// If instance is now empty, stop it\n\t\tif (instance.deviceUdids.length === 0) {\n\t\t\tlogDetail(`Instance ${instance.id} is empty, stopping...`);\n\t\t\tawait this.stopInstance(instance.id);\n\t\t}\n\t}\n\n\t/**\n\t * Find an instance with available capacity\n\t */\n\tprivate findInstanceWithCapacity(): UsbmuxdInstance | null {\n\t\tfor (const instance of this.instances.values()) {\n\t\t\tif (instance.deviceUdids.length < this.config.batchSize) {\n\t\t\t\treturn instance;\n\t\t\t}\n\t\t}\n\t\treturn null;\n\t}\n\n\t/**\n\t * Stop a specific instance\n\t */\n\tprivate async stopInstance(instanceId: number): Promise<void> {\n\t\tconst instance = this.instances.get(instanceId);\n\t\tconst childProcess = this.processes.get(instanceId);\n\n\t\tif (!instance || !childProcess) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor (const process of this.processes.values()) {\n\t\t\tprocess.kill(\"SIGKILL\");\n\t\t}\n\n\t\t// Clean up\n\t\tthis.instances.delete(instanceId);\n\t\tthis.processes.delete(instanceId);\n\n\t\t// Remove device mappings\n\t\tfor (const [udid, mapping] of this.deviceMappings.entries()) {\n\t\t\tif (mapping.instanceId === instanceId) {\n\t\t\t\tthis.deviceMappings.delete(udid);\n\t\t\t}\n\t\t}\n\n\t\tthis.emit(\"instance-stopped\", instance);\n\t}\n\n\t/**\n\t * Get device-to-port mapping for a specific UDID\n\t */\n\tpublic getDevicePort(udid: string): number | null {\n\t\tconst mapping = this.deviceMappings.get(udid);\n\t\treturn mapping ? mapping.port : null;\n\t}\n\n\t/**\n\t * Get all device mappings\n\t */\n\tpublic getDeviceMappings(): DeviceMapping[] {\n\t\treturn Array.from(this.deviceMappings.values());\n\t}\n\n\t/**\n\t * Get all running instances\n\t */\n\tpublic getInstances(): UsbmuxdInstance[] {\n\t\treturn Array.from(this.instances.values());\n\t}\n\n\t/**\n\t * Get manager statistics\n\t */\n\tpublic getStats(): ManagerStats {\n\t\tconst now = Date.now();\n\t\tconst startTime = this.startedAt?.getTime() || now;\n\t\tconst uptimeSeconds = Math.floor((now - startTime) / 1000);\n\n\t\treturn {\n\t\t\tinstanceCount: this.instances.size,\n\t\t\tdeviceCount: this.deviceMappings.size,\n\t\t\tuptimeSeconds,\n\t\t\tstartedAt: this.startedAt || new Date(),\n\t\t};\n\t}\n\n\t/**\n\t * Get current configuration\n\t */\n\tpublic getConfig(): InstanceManagerConfig {\n\t\treturn { ...this.config };\n\t}\n}\n", "import { exec } from \"node:child_process\";\nimport { promisify } from \"node:util\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type { DeviceInfo } from \"./types\";\nimport type { UsbipdDevice } from \"./types/usbipd.js\";\nimport type { Wsl2 } from \"./wsl.js\";\n\nconst { logInfo, logWarning } = createLoggers(\"usbipd\");\n\nconst execAsync = promisify(exec);\n\nexport class Usbipd {\n constructor(private readonly usbipdPath: string) { }\n\n public async unbindAllDevicesFromWsl(): Promise<void> {\n try {\n await execAsync(`\"${this.usbipdPath}\" unbind -a`);\n logInfo(\"All devices unbound from WSL\");\n } catch (error) {\n logWarning(`Failed to unbind all devices from WSL: ${error}`);\n }\n }\n\n public async detachAllDevicesFromWsl(): Promise<void> {\n try {\n await execAsync(`\"${this.usbipdPath}\" detach -a`);\n logInfo(\"All devices detached from WSL\");\n } catch (error) {\n logWarning(`Failed to detach all devices from WSL: ${error}`);\n }\n }\n\n /**\n * Detach a device from WSL via usbipd\n */\n public async detachDeviceFromWsl(busId: string): Promise<void> {\n try {\n await execAsync(`\"${this.usbipdPath}\" detach --busid=${busId}`);\n logInfo(`Device ${busId} detached from WSL`);\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n if (message.includes(\"no device with busid\") || message.includes(\"There is no device\")) {\n logInfo(`Device ${busId} already detached`);\n } else {\n logWarning(`Failed to detach device ${busId}: ${error}`);\n }\n }\n }\n\n /**\n * Attach a device to WSL via usbipd\n * Note: This requires administrator privileges\n */\n public async attachDeviceToWsl(busId: string, wsl: Wsl2, distro: string): Promise<boolean> {\n try {\n // Ensure WSL is running first\n await wsl.ensureWslRunning();\n\n // Bind the device (make it shareable)\n logInfo(`Binding device ${busId}...`);\n await execAsync(`\"${this.usbipdPath}\" bind --busid ${busId} --force`);\n\n // Attach to WSL (use specific distro if configured, otherwise let usbipd pick)\n if (distro) {\n logInfo(`Attaching device ${busId} to WSL distribution ${distro}...`);\n await execAsync(`\"${this.usbipdPath}\" attach --wsl=${distro} --busid=${busId}`);\n } else {\n logInfo(`Attaching device ${busId} to default WSL...`);\n await execAsync(`\"${this.usbipdPath}\" attach --wsl --busid=${busId}`);\n }\n\n logInfo(`Device ${busId} attached to WSL successfully`);\n return true;\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n // Device already attached (e.g. from a previous run) \u2014 treat as success so we still start usbmuxd\n if (message.includes(\"already attached to a client\")) {\n logInfo(`Device ${busId} is already attached to WSL, continuing`);\n return true;\n }\n logWarning(`Failed to attach device ${busId} to WSL: ${error}`);\n return false;\n }\n }\n\n /**\n * Parse usbipd list output to extract device information\n */\n private parseUsbipdList(output: string): UsbipdDevice[] {\n const devices: UsbipdDevice[] = [];\n // Handle both \\r\\n and \\n line endings\n const lines = output.split(/\\r?\\n/);\n\n for (const line of lines) {\n // Format: \"4-5 05ac:12a8 Apple Mobile Device USB Composite Device Shared (forced)\"\n // More flexible regex: busid, vid:pid, then anything else\n const match = line.match(/^(\\d+-\\d+)\\s+([0-9a-f]{4}):([0-9a-f]{4})\\s+(.+)$/i);\n if (match) {\n // Split the rest into description and state (state is the last word(s) after multiple spaces)\n const rest = match[4].trim();\n const stateMatch = rest.match(/^(.+?)\\s{2,}(\\S.*)$/);\n const description = stateMatch ? stateMatch[1].trim() : rest;\n const state = stateMatch ? stateMatch[2].trim() : \"Unknown\";\n\n devices.push({\n busId: match[1],\n vid: match[2].toUpperCase(),\n pid: match[3].toUpperCase(),\n description,\n state,\n });\n }\n }\n\n return devices;\n }\n /**\n * Find the usbipd bus ID for a device by matching VID/PID\n */\n public async findBusIdForDevice(device: DeviceInfo): Promise<string> {\n try {\n const { stdout } = await execAsync(`\"${this.usbipdPath}\" list`);\n const usbipdDevices = this.parseUsbipdList(stdout);\n\n // Convert device VID/PID to hex string for comparison\n const deviceVid = device.vid.toString(16).toUpperCase().padStart(4, \"0\");\n const devicePid = device.pid.toString(16).toUpperCase().padStart(4, \"0\");\n\n logInfo(`Looking for device with VID:PID ${deviceVid}:${devicePid}`);\n logInfo(`Found ${usbipdDevices.length} devices from usbipd list`);\n\n // Find matching device by VID/PID\n const match = usbipdDevices.find((d) => d.vid === deviceVid && d.pid === devicePid);\n\n if (match) {\n logInfo(\n `Found usbipd bus ID ${match.busId} for device ${device.deviceId} (${deviceVid}:${devicePid})`\n );\n return match.busId;\n }\n\n throw new Error(\n `Could not find usbipd bus ID for device ${device.deviceId} (${deviceVid}:${devicePid})`\n );\n } catch (error) {\n throw new Error(`Failed to run usbipd list: ${error}`);\n }\n }\n}\n", "import { type ChildProcess, exec, spawn } from \"node:child_process\";\nimport { existsSync, mkdirSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { promisify } from \"node:util\";\nimport { createLoggers } from \"@mcesystems/tool-debug-g4\";\nimport type { UsbmuxdInstance } from \"./types\";\n\nconst { logInfo, logWarning, logDetail } = createLoggers(\"wsl\");\n\nconst execAsync = promisify(exec);\n\nexport class Wsl2 {\n\tprivate readonly ALPINE_LOCKDOWN_DIR = \"/var/lib/lockdown\";\n\tprivate readonly SYSTEM_CONFIG_PLIST = \"SystemConfiguration.plist\";\n\n\tprivate wslIpAddress: string | null = null;\n\n\tconstructor(private wslDistribution: string | undefined) {}\n\n\t/**\n\t * Detect the WSL2 IP address for the configured distribution\n\t * This IP is needed to connect from Windows to services inside WSL\n\t */\n\tpublic async detectWslIpAddress(): Promise<string> {\n\t\tif (this.wslIpAddress) {\n\t\t\treturn this.wslIpAddress;\n\t\t}\n\n\t\tconst distro = this.wslDistribution || \"alpine-usbmuxd-build\";\n\n\t\ttry {\n\t\t\t// Try to get IP from eth0 interface\n\t\t\tconst { stdout } = await execAsync(`wsl -d ${distro} -- ip -4 addr show eth0`);\n\t\t\tconst match = stdout.match(/inet\\s+(\\d+\\.\\d+\\.\\d+\\.\\d+)/);\n\t\t\tif (match) {\n\t\t\t\tthis.wslIpAddress = match[1];\n\t\t\t\tlogInfo(`Detected WSL IP address: ${this.wslIpAddress}`);\n\t\t\t\treturn this.wslIpAddress;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to detect WSL IP via ip addr: ${error}`);\n\t\t}\n\n\t\t// Fallback: try hostname -I\n\t\ttry {\n\t\t\tconst { stdout } = await execAsync(`wsl -d ${distro} -- hostname -I`);\n\t\t\tconst ip = stdout.trim().split(/\\s+/)[0];\n\t\t\tif (ip && /^\\d+\\.\\d+\\.\\d+\\.\\d+$/.test(ip)) {\n\t\t\t\tthis.wslIpAddress = ip;\n\t\t\t\tlogInfo(`Detected WSL IP address (hostname): ${this.wslIpAddress}`);\n\t\t\t\treturn this.wslIpAddress;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to detect WSL IP via hostname: ${error}`);\n\t\t}\n\n\t\t// Last resort fallback - localhost (may work with WSL2 mirrored networking)\n\t\tlogWarning(\"Could not detect WSL IP, falling back to localhost\");\n\t\tthis.wslIpAddress = \"127.0.0.1\";\n\t\treturn this.wslIpAddress;\n\t}\n\n\t/**\n\t * Ensure the WSL distribution is running\n\t */\n\tpublic async ensureWslRunning(): Promise<boolean> {\n\t\tconst distro = this.wslDistribution;\n\n\t\ttry {\n\t\t\tif (distro) {\n\t\t\t\t// Start specific distribution\n\t\t\t\tlogInfo(`Starting WSL distribution: ${distro}...`);\n\t\t\t\tawait execAsync(`wsl -d ${distro} -- echo \"WSL started\"`);\n\t\t\t} else {\n\t\t\t\t// Start default WSL\n\t\t\t\tlogInfo(\"Starting default WSL distribution...\");\n\t\t\t\tawait execAsync(`wsl -- echo \"WSL started\"`);\n\t\t\t}\n\t\t\treturn true;\n\t\t} catch (error) {\n\t\t\tlogWarning(`Failed to start WSL: ${error}`);\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Create a new usbmuxd instance\n\t */\n\tpublic async createInstance({\n\t\tid,\n\t\tbasePort,\n\t\tverboseLogging,\n\t\tusbmuxdPath,\n\t\tonLog,\n\t\tonExit,\n\t}: {\n\t\tid: number;\n\t\tbasePort: number;\n\t\tverboseLogging: boolean;\n\t\tusbmuxdPath: string;\n\t\tonLog: (message: string) => void;\n\t\tonExit: (code: number, signal: string) => void;\n\t}): Promise<{ instance: UsbmuxdInstance; process: ChildProcess }> {\n\t\tconst port = basePort + id - 1;\n\n\t\tlogInfo(`Creating instance ${id} on port ${port}`);\n\n\t\t// Detect WSL IP for connecting from Windows\n\t\tconst host = await this.detectWslIpAddress();\n\n\t\t// Build usbmuxd arguments\n\t\tconst usbmuxdArgs = [\n\t\t\t\"-f\", // Foreground\n\t\t\t\"-v\", // Verbose (if enabled)\n\t\t\t\"-S\",\n\t\t\t`0.0.0.0:${port}`, // Listen on all interfaces (for Windows \u2192 WSL2)\n\t\t\t\"--pidfile\",\n\t\t\t\"NONE\",\n\t\t];\n\n\t\tif (!verboseLogging) {\n\t\t\t// Remove -v if not verbose\n\t\t\tusbmuxdArgs.splice(1, 1);\n\t\t}\n\n\t\t// Build WSL command: wsl -d <distro> usbmuxd <args>\n\t\tconst wslArgs = [\n\t\t\t\"-d\",\n\t\t\tthis.wslDistribution || \"alpine-usbmuxd-build\",\n\t\t\tusbmuxdPath,\n\t\t\t...usbmuxdArgs,\n\t\t];\n\n\t\t// Spawn via WSL\n\t\t// detached: true creates a new process group so the child does NOT receive\n\t\t// the parent's Ctrl+C. Without this, Windows delivers SIGINT to every process\n\t\t// in the console, killing usbmuxd before stop() can sync lockdown files back.\n\t\t// windowsHide: true prevents the detached process from opening a visible console window.\n\t\tconst process = spawn(\"wsl\", wslArgs, {\n\t\t\tstdio: [\"ignore\", \"pipe\", \"pipe\"],\n\t\t\tdetached: true,\n\t\t\twindowsHide: false,\n\t\t});\n\n\t\t// Handle process output\n\t\tprocess.stdout?.on(\"data\", (data) => {\n\t\t\tonLog(data.toString().trim());\n\t\t});\n\n\t\tprocess.stderr?.on(\"data\", (data) => {\n\t\t\tonLog(data.toString().trim());\n\t\t});\n\n\t\t// Handle process exit\n\t\tprocess.on(\"exit\", onExit);\n\n\t\t// Create instance record (pid is set synchronously by Node after spawn)\n\t\tconst pid = process.pid;\n\t\tif (pid === undefined) {\n\t\t\tprocess.kill(\"SIGKILL\");\n\t\t\tthrow new Error(\"Failed to get PID for usbmuxd instance\");\n\t\t}\n\t\tconst instance: UsbmuxdInstance = {\n\t\t\tid,\n\t\t\thost,\n\t\t\tport,\n\t\t\tpid,\n\t\t\tdeviceUdids: [],\n\t\t\tstartedAt: new Date(),\n\t\t};\n\n\t\treturn { instance, process };\n\t}\n\n\t/**\n\t * Convert a Windows path to the equivalent path inside WSL (e.g. C:\\foo\\bar -> /mnt/c/foo/bar).\n\t */\n\tprivate windowsPathToWsl(windowsPath: string): string {\n\t\tconst normalized = windowsPath.replace(/\\\\/g, \"/\").trim();\n\t\tconst driveMatch = normalized.match(/^([a-zA-Z]):\\/?(.*)$/);\n\t\tif (driveMatch) {\n\t\t\tconst drive = driveMatch[1].toLowerCase();\n\t\t\tconst rest = driveMatch[2] || \"\";\n\t\t\treturn `/mnt/${drive}${rest ? `/${rest}` : \"\"}`;\n\t\t}\n\t\treturn normalized;\n\t}\n\n\t/**\n\t * Get the native Apple lockdown directory path for the current platform.\n\t * This is where Apple/iTunes stores pairing records natively.\n\t */\n\tprivate getAppleLockdownPath(): string | null {\n\t\tif (process.platform === \"win32\") {\n\t\t\treturn join(process.env.ProgramData ?? \"C:\\\\ProgramData\", \"Apple\", \"Lockdown\");\n\t\t}\n\t\tif (process.platform === \"darwin\") {\n\t\t\treturn \"/var/db/lockdown\";\n\t\t}\n\t\tif (process.platform === \"linux\") {\n\t\t\treturn \"/var/lib/lockdown\";\n\t\t}\n\t\treturn null;\n\t}\n\n\t/**\n\t * Copy a single device's lockdown plist from the native Apple lockdown directory\n\t * to Alpine so usbmuxd can use it (skip pairing).\n\t *\n\t * Uses the platform-specific Apple/iTunes lockdown directory\n\t * (e.g. C:\\ProgramData\\Apple\\Lockdown on Windows, /var/db/lockdown on macOS).\n\t *\n\t * No-op if the lockdown directory cannot be determined or files don't exist.\n\t */\n\tpublic async syncToAlpine(udid: string): Promise<void> {\n\t\tconst lockdownDir = this.getAppleLockdownPath();\n\t\tif (!lockdownDir) {\n\t\t\tlogWarning(\"Lockdown sync to Alpine: unsupported platform, skipping\");\n\t\t\treturn;\n\t\t}\n\n\t\tlogDetail(`Lockdown sync to Alpine: source dir = ${lockdownDir}, udid = ${udid}`);\n\n\t\tconst distro = this.wslDistribution || \"alpine-usbmuxd-build\";\n\t\tconst wslDestDir = this.ALPINE_LOCKDOWN_DIR;\n\n\t\t// Sync the device-specific plist\n\t\tconst devicePlistFile = `${udid}.plist`;\n\t\tconst devicePlistPath = join(lockdownDir, devicePlistFile);\n\t\tif (existsSync(devicePlistPath)) {\n\t\t\tawait this.copyFileToAlpine(\n\t\t\t\tthis.windowsPathToWsl(devicePlistPath),\n\t\t\t\twslDestDir,\n\t\t\t\tdistro,\n\t\t\t\tdevicePlistFile\n\t\t\t);\n\t\t} else {\n\t\t\tlogDetail(\n\t\t\t\t`Lockdown sync to Alpine: ${devicePlistFile} not found at ${devicePlistPath}, skipping`\n\t\t\t);\n\t\t}\n\n\t\t// Sync SystemConfiguration.plist so Alpine usbmuxd uses the same SystemBUID\n\t\t// that was used during the original pairing on Windows.\n\t\t// Without this, lockdown sessions fail with InvalidHostID.\n\t\tconst systemConfigPath = join(lockdownDir, this.SYSTEM_CONFIG_PLIST);\n\t\tif (existsSync(systemConfigPath)) {\n\t\t\tawait this.copyFileToAlpine(\n\t\t\t\tthis.windowsPathToWsl(systemConfigPath),\n\t\t\t\twslDestDir,\n\t\t\t\tdistro,\n\t\t\t\tthis.SYSTEM_CONFIG_PLIST\n\t\t\t);\n\t\t} else {\n\t\t\tlogDetail(\n\t\t\t\t`Lockdown sync to Alpine: ${this.SYSTEM_CONFIG_PLIST} not found at ${systemConfigPath}, skipping`\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * Copy a single file into the Alpine lockdown directory.\n\t * Falls back to sudo if the initial copy fails.\n\t */\n\tprivate async copyFileToAlpine(\n\t\twslSource: string,\n\t\twslDestDir: string,\n\t\tdistro: string,\n\t\tfileName: string\n\t): Promise<void> {\n\t\ttry {\n\t\t\tawait execAsync(\n\t\t\t\t`wsl -d ${distro} -- sh -c \"mkdir -p ${wslDestDir} && cp '${wslSource}' '${wslDestDir}/'\"`\n\t\t\t);\n\t\t\tlogInfo(`Lockdown sync: copied ${fileName} to Alpine`);\n\t\t} catch (error) {\n\t\t\ttry {\n\t\t\t\tawait execAsync(\n\t\t\t\t\t`wsl -d ${distro} -- sh -c \"mkdir -p ${wslDestDir} && sudo cp '${wslSource}' '${wslDestDir}/'\"`\n\t\t\t\t);\n\t\t\t\tlogInfo(`Lockdown sync: copied ${fileName} to Alpine (via sudo)`);\n\t\t\t} catch (sudoError) {\n\t\t\t\tlogWarning(\n\t\t\t\t\t`Lockdown sync to Alpine failed for ${fileName}: ${error}. Sudo fallback failed: ${sudoError}. Ensure /var/lib/lockdown is writable or use passwordless sudo.`\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Copy lockdown plists from Alpine back to the native Apple lockdown directory\n\t * (for devices that were assigned this session).\n\t * Also copies SystemConfiguration.plist if present in Alpine.\n\t *\n\t * No-op if the lockdown directory cannot be determined.\n\t */\n\tpublic async syncFromAlpine(udids: string[]): Promise<void> {\n\t\tlogInfo(`Lockdown sync from Alpine: starting (${udids.length} device(s))`);\n\n\t\tconst lockdownDir = this.getAppleLockdownPath();\n\t\tif (!lockdownDir) {\n\t\t\tlogWarning(\"Lockdown sync: unsupported platform, skipping\");\n\t\t\treturn;\n\t\t}\n\n\t\tlogDetail(`Lockdown sync: target dir = ${lockdownDir}`);\n\n\t\ttry {\n\t\t\tmkdirSync(lockdownDir, { recursive: true });\n\t\t} catch (error) {\n\t\t\tlogWarning(`Lockdown sync: could not create lockdown dir ${lockdownDir}: ${error}`);\n\t\t\treturn;\n\t\t}\n\n\t\tconst distro = this.wslDistribution || \"alpine-usbmuxd-build\";\n\t\tconst wslDestDir = this.windowsPathToWsl(lockdownDir);\n\n\t\t// List what Alpine actually has in its lockdown dir for debugging\n\t\ttry {\n\t\t\tconst { stdout } = await execAsync(`wsl -d ${distro} -- ls -la ${this.ALPINE_LOCKDOWN_DIR}/`);\n\t\t\tlogDetail(`Lockdown sync: Alpine ${this.ALPINE_LOCKDOWN_DIR} contents:\\n${stdout}`);\n\t\t} catch (error) {\n\t\t\tlogWarning(`Lockdown sync: could not list Alpine lockdown dir: ${error}`);\n\t\t}\n\n\t\tconst filesToSync = [...udids.map((udid) => `${udid}.plist`), this.SYSTEM_CONFIG_PLIST];\n\n\t\tlogDetail(`Lockdown sync: files to sync = [${filesToSync.join(\", \")}]`);\n\n\t\tfor (const fileName of filesToSync) {\n\t\t\tawait this.copyFileFromAlpine(fileName, wslDestDir, distro, lockdownDir);\n\t\t}\n\n\t\tlogInfo(\"Lockdown sync from Alpine: done\");\n\t}\n\n\t/**\n\t * Copy a single file from the Alpine lockdown directory back to Windows.\n\t * Skips if the file does not exist in Alpine; logs a warning if the copy fails.\n\t */\n\tprivate async copyFileFromAlpine(\n\t\tfileName: string,\n\t\twslDestDir: string,\n\t\tdistro: string,\n\t\tlockdownDir: string\n\t): Promise<void> {\n\t\tconst alpinePath = `${this.ALPINE_LOCKDOWN_DIR}/${fileName}`;\n\n\t\t// Check if the file exists in Alpine first\n\t\ttry {\n\t\t\tawait execAsync(`wsl -d ${distro} -- test -f '${alpinePath}'`);\n\t\t} catch {\n\t\t\tlogDetail(`Lockdown sync: ${fileName} not found in Alpine (${alpinePath}), skipping`);\n\t\t\treturn;\n\t\t}\n\n\t\t// File exists \u2014 copy it to Windows\n\t\tlogDetail(`Lockdown sync: copying ${fileName} \u2192 ${wslDestDir}/`);\n\t\ttry {\n\t\t\tawait execAsync(`wsl -d ${distro} -- cp -f '${alpinePath}' '${wslDestDir}/'`);\n\t\t\tlogInfo(`Lockdown sync: copied ${fileName} from Alpine to ${lockdownDir}`);\n\t\t} catch (error) {\n\t\t\tlogWarning(`Lockdown sync: failed to copy ${fileName} from Alpine to ${lockdownDir}: ${error}`);\n\t\t}\n\t}\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["exports", "module", "exports", "module", "options", "path", "config", "import_tool_debug_g4", "import_node_events", "import_tool_debug_g4", "import_tool_debug_g4", "import_node_child_process", "import_node_util", "import_tool_debug_g4", "logInfo", "logWarning", "execAsync", "process", "logWarning", "logDetail", "config", "process", "logInfo", "logDataObject", "config", "dotenv", "logInfo", "logError", "logDataObject"]
7
7
  }
package/dist/cli.mjs CHANGED
@@ -435,7 +435,6 @@ import { createLoggers as createLoggers5 } from "@mcesystems/tool-debug-g4";
435
435
  // src/UsbmuxdService.ts
436
436
  import { EventEmitter as EventEmitter2 } from "node:events";
437
437
  import { createLoggers as createLoggers4 } from "@mcesystems/tool-debug-g4";
438
- import UsbDeviceListener from "@mcesystems/usb-device-listener";
439
438
 
440
439
  // src/InstanceManager.ts
441
440
  import { EventEmitter } from "node:events";
@@ -1156,12 +1155,10 @@ var InstanceManager = class extends EventEmitter {
1156
1155
  var { logInfo: logInfo3, logError, logDataObject: logDataObject2 } = createLoggers4("usbmuxd-service");
1157
1156
  var UsbmuxdService = class extends EventEmitter2 {
1158
1157
  manager;
1159
- usbListener;
1160
1158
  isListening = false;
1161
1159
  constructor(config2 = {}) {
1162
1160
  super();
1163
1161
  this.manager = new InstanceManager(config2);
1164
- this.usbListener = new UsbDeviceListener();
1165
1162
  this.setupEventHandlers();
1166
1163
  }
1167
1164
  /**
@@ -1214,39 +1211,18 @@ var UsbmuxdService = class extends EventEmitter2 {
1214
1211
  throw new Error("Service is already running");
1215
1212
  }
1216
1213
  const config2 = this.manager.getConfig();
1217
- const appleVid = Number.parseInt(config2.appleVendorId, 16);
1218
1214
  logDataObject2("Starting service...", { config: config2 });
1219
- this.usbListener.onDeviceAdd(async (device) => {
1220
- if (device.vid !== appleVid) {
1221
- return;
1222
- }
1223
- logInfo3(`Apple device connected: ${device.deviceId} (${device.deviceName || "Unknown"})`);
1224
- try {
1225
- await this.manager.onDeviceConnected(device);
1226
- } catch (error) {
1227
- logError("Error handling device connection:", error);
1228
- }
1229
- });
1230
- this.usbListener.onDeviceRemove(async (device) => {
1231
- if (device.vid !== appleVid) {
1232
- return;
1233
- }
1234
- logInfo3(`Apple device disconnected: ${device.deviceId}`);
1235
- try {
1236
- await this.manager.onDeviceDisconnected(device);
1237
- } catch (error) {
1238
- logError("Error handling device disconnection:", error);
1239
- }
1240
- });
1241
1215
  try {
1242
- this.usbListener.startListening({
1243
- targetDevices: []
1244
- // Monitor all devices, we'll filter by VID
1245
- });
1246
1216
  this.isListening = true;
1247
1217
  this.manager.start();
1248
1218
  logInfo3("Service started successfully");
1249
1219
  logInfo3("Waiting for iOS devices...");
1220
+ const onAdd = this.manager.onDeviceConnected.bind(this.manager);
1221
+ const onRemove = this.manager.onDeviceDisconnected.bind(this.manager);
1222
+ return {
1223
+ onAdd,
1224
+ onRemove
1225
+ };
1250
1226
  } catch (error) {
1251
1227
  logError("Failed to start USB listener:", error);
1252
1228
  throw error;
@@ -1261,7 +1237,6 @@ var UsbmuxdService = class extends EventEmitter2 {
1261
1237
  return;
1262
1238
  }
1263
1239
  logInfo3("Stopping service...");
1264
- this.usbListener.stopListening();
1265
1240
  this.isListening = false;
1266
1241
  await this.manager.stop();
1267
1242
  logInfo3("Service stopped");