agent-device 0.15.0 → 0.15.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (26) hide show
  1. package/android-snapshot-helper/dist/{agent-device-android-snapshot-helper-0.15.0.apk → agent-device-android-snapshot-helper-0.15.1.apk} +0 -0
  2. package/android-snapshot-helper/dist/agent-device-android-snapshot-helper-0.15.1.apk.sha256 +1 -0
  3. package/android-snapshot-helper/dist/{agent-device-android-snapshot-helper-0.15.0.manifest.json → agent-device-android-snapshot-helper-0.15.1.manifest.json} +6 -6
  4. package/dist/src/1393.js +1 -0
  5. package/dist/src/1769.js +7 -7
  6. package/dist/src/1974.js +2 -2
  7. package/dist/src/208.js +1 -1
  8. package/dist/src/2151.js +27 -22
  9. package/dist/src/221.js +3 -3
  10. package/dist/src/3572.js +1 -1
  11. package/dist/src/4829.js +1 -1
  12. package/dist/src/9542.js +2 -2
  13. package/dist/src/989.js +1 -1
  14. package/dist/src/android-adb.js +1 -1
  15. package/dist/src/cli.js +44 -44
  16. package/dist/src/index.d.ts +37 -5
  17. package/dist/src/internal/daemon.js +45 -45
  18. package/ios-runner/AgentDeviceRunner/AgentDeviceRunnerUITests/RunnerTests+Alert.swift +155 -0
  19. package/ios-runner/AgentDeviceRunner/AgentDeviceRunnerUITests/RunnerTests+CommandExecution.swift +3 -25
  20. package/ios-runner/AgentDeviceRunner/AgentDeviceRunnerUITests/RunnerTests+Lifecycle.swift +1 -14
  21. package/ios-runner/AgentDeviceRunner/AgentDeviceRunnerUITests/RunnerTests+SystemModal.swift +3 -3
  22. package/ios-runner/AgentDeviceRunner/RecordingScripts/recording-overlay.swift +7 -1
  23. package/package.json +6 -1
  24. package/server.json +2 -2
  25. package/android-snapshot-helper/dist/agent-device-android-snapshot-helper-0.15.0.apk.sha256 +0 -1
  26. package/dist/src/840.js +0 -2
package/dist/src/221.js CHANGED
@@ -1,4 +1,4 @@
1
- import e from"node:crypto";import t from"node:fs";import n from"node:fs/promises";import r from"node:os";import i from"node:path";import{AppError as a}from"./9152.js";import{isScrollableType as o}from"./2842.js";import{installAndroidAdbPackage as s}from"./9639.js";let l="android-snapshot-helper",u="com.callstack.agentdevice.snapshothelper",d="com.callstack.agentdevice.snapshothelper/.SnapshotInstrumentation",c="android-snapshot-helper-v1",f="uiautomator-xml",h={"-r":"replace","-t":"allowTestPackages","-d":"allowDowngrade","-g":"grantPermissions"};async function p(e){let t=await N(e.apkPath);if(t!==e.manifest.sha256)throw new a("COMMAND_FAILED","Android snapshot helper APK checksum mismatch",{apkPath:e.apkPath,expectedSha256:e.manifest.sha256,actualSha256:t})}async function m(e){let t=e.fetch??fetch,o=await t(e.manifestUrl);if(!o.ok)throw new a("COMMAND_FAILED","Failed to download Android snapshot helper manifest",{manifestUrl:e.manifestUrl,status:o.status,statusText:o.statusText});let s=A(JSON.parse((await w(o,65536,"Android snapshot helper manifest")).toString("utf8")));if(!s.apkUrl)throw new a("COMMAND_FAILED","Android snapshot helper manifest does not include apkUrl",{manifestUrl:e.manifestUrl});let l=e.cacheDir??i.join(r.tmpdir(),`agent-device-android-snapshot-helper-${s.version}`),u=!e.cacheDir;await n.mkdir(l,{recursive:!0});let d=s.assetName??`agent-device-android-snapshot-helper-${s.version}.apk`,c=i.join(l,d),f=await t(s.apkUrl);if(!f.ok)throw new a("COMMAND_FAILED","Failed to download Android snapshot helper APK",{apkUrl:s.apkUrl,status:f.status,statusText:f.statusText});await n.writeFile(c,await w(f,0x1400000,"Android snapshot helper APK"));let h={apkPath:c,manifest:s};return await p(h),{...h,cleanup:async()=>{await n.rm(u?l:c,{recursive:u,force:!0})}}}function A(e){var t,n;if(!e||"object"!=typeof e||Array.isArray(e))throw new a("INVALID_ARGS","Android snapshot helper manifest must be an object.");return{name:I(e.name,"name",l),version:M(e.version,"version"),releaseTag:v(e.releaseTag),assetName:v(e.assetName),apkUrl:(t=e.apkUrl,n="apkUrl",null===t?null:M(t,n)),sha256:function(e){let t=M(e,"sha256").trim().toLowerCase();if(64!==t.length||!function(e){for(let t of e){let e=t.charCodeAt(0),n=e>=48&&e<=57,r=e>=97&&e<=102;if(!n&&!r)return!1}return!0}(t))throw new a("INVALID_ARGS","Android snapshot helper manifest sha256 must be a 64-character hex string.");return t}(e.sha256),checksumName:v(e.checksumName),packageName:M(e.packageName,"packageName"),versionCode:x(e.versionCode,"versionCode"),instrumentationRunner:M(e.instrumentationRunner,"instrumentationRunner"),minSdk:x(e.minSdk,"minSdk"),targetSdk:void 0===e.targetSdk?void 0:x(e.targetSdk,"targetSdk"),outputFormat:I(e.outputFormat,"outputFormat",f),statusProtocol:I(e.statusProtocol,"statusProtocol",c),installArgs:g(e.installArgs)}}async function w(e,t,n){let r=e.headers.get("content-length");if(null!==r){let e=Number(r);if(Number.isFinite(e)&&e>t)throw new a("COMMAND_FAILED",`${n} download exceeds size limit`,{contentLength:e,maxBytes:t})}if(!e.body){let r=Buffer.from(await e.arrayBuffer());if(r.length>t)throw new a("COMMAND_FAILED",`${n} download exceeds size limit`,{contentLength:r.length,maxBytes:t});return r}let i=e.body.getReader(),o=[],s=0;try{for(;;){let{done:e,value:r}=await i.read();if(e)break;if((s+=r.byteLength)>t)throw new a("COMMAND_FAILED",`${n} download exceeds size limit`,{contentLength:s,maxBytes:t});o.push(Buffer.from(r))}}finally{i.releaseLock()}return Buffer.concat(o,s)}function g(e){let t=function(e,t){if(!Array.isArray(e)||!e.every(e=>"string"==typeof e))throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} must be a string array.`);return e}(e,"installArgs");if("install"!==t[0])throw new a("INVALID_ARGS",'Android snapshot helper manifest installArgs must start with "install".');if(t.some(e=>e.includes("\0")))throw new a("INVALID_ARGS","Android snapshot helper manifest installArgs must not contain null bytes.");let n=t.slice(1).find(e=>void 0===D(e));if(n)throw new a("INVALID_ARGS",`Android snapshot helper manifest installArgs contains unsupported install flag "${n}".`);return t}async function N(n){return await new Promise((r,i)=>{let a=e.createHash("sha256"),o=t.createReadStream(n);o.on("error",i),o.on("data",e=>a.update(e)),o.on("end",()=>r(a.digest("hex")))})}function M(e,t){if("string"!=typeof e||0===e.trim().length)throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} is required.`);return e}function v(e){return"string"==typeof e&&e.trim().length>0?e:void 0}function x(e,t){if("number"!=typeof e||!Number.isInteger(e))throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} must be an integer.`);return e}function I(e,t,n){if(e!==n)throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} must be "${n}".`);return n}function D(e){if(Object.hasOwn(h,e))return h[e]}function*b(e){let t=/<node\b[^>]*>/g,n=t.exec(e);for(;n;)yield C(n[0]),n=t.exec(e)}function C(e){let t,n,r=(t=function(e){let t=new Map,n=e.indexOf(" "),r=e.lastIndexOf(">");if(n<0||r<=n)return t;let i=n;for(;i<r&&!((i=_(e,i,r))>=r);){var a;let n=e[i];if("/"===n||">"===n)break;let o=i;for(;i<r&&!("="===(a=e[i]??"")||"/"===a||">"===a||y(a));)i+=1;let s=e.slice(o,i);if(i=_(e,i,r),!s||"="!==e[i])break;i=_(e,i+1,r);let l=e[i];if('"'!==l&&"'"!==l)break;let u=i+=1;for(;i<r&&e[i]!==l;)i+=1;if(i>=r)break;t.set(s,function(e){let t="",n=0;for(;n<e.length;){let r=e.indexOf("&",n);if(r<0){t+=e.slice(n);break}t+=e.slice(n,r);let i=e.indexOf(";",r+1);if(i<0){t+=e.slice(r);break}t+=function(e){switch(e){case"amp":return"&";case"lt":return"<";case"gt":return">";case"quot":return'"';case"apos":return"'";default:return function(e){if(!e.startsWith("#"))return;let t=e[1]?.toLowerCase()==="x"?16:10,n=16===t?e.slice(2):e.slice(1);if(!n||!function(e,t){for(let n of e){let e=n.charCodeAt(0),r=e>=48&&e<=57;if(10===t){if(!r)return!1;continue}let i=e>=65&&e<=70,a=e>=97&&e<=102;if(!r&&!i&&!a)return!1}return!0}(n,t))return;let r=Number.parseInt(n,t);if(Number.isFinite(r))try{return String.fromCodePoint(r)}catch{return}}(e)}}(e.slice(r+1,i))??e.slice(r,i+1),n=i+1}return t}(e.slice(u,i))),i+=1}return t}(e),n=e=>{let n=L(t,e);if(null!==n)return"true"===n},{text:L(t,"text"),desc:L(t,"content-desc"),resourceId:L(t,"resource-id"),packageName:L(t,"package"),className:L(t,"class"),bounds:L(t,"bounds"),clickable:n("clickable"),enabled:n("enabled"),focusable:n("focusable"),focused:n("focused"),password:n("password")}),i=function(e){if(!e)return;let t=/\[(\d+),(\d+)\]\[(\d+),(\d+)\]/.exec(e);if(!t)return;let n=Number(t[1]),r=Number(t[2]);return{x:n,y:r,width:Math.max(0,Number(t[3])-n),height:Math.max(0,Number(t[4])-r)}}(r.bounds);return{...r,...i?{rect:i}:{}}}function S(e,t,n){let{sourceNodes:r,...i}=k(T(e),t,n);return i}function k(e,t,n){let r={nodes:[],sourceNodes:[],maxNodes:t,maxDepth:n.depth??1/0,options:n,analysis:function(e){let t=0,n=0,r=[...e.children];for(;r.length>0;){let e=r.pop();t+=1,n=Math.max(n,e.depth),r.push(...e.children)}return{rawNodeCount:t,maxDepth:n}}(e),interactiveDescendantMemo:new Map,truncated:!1},i=n.scope?function(e,t){let n=t.toLowerCase(),r=[...e.children],i=0;for(;i<r.length;){let e=r[i++],t=e.label?.toLowerCase()??"",a=e.value?.toLowerCase()??"",o=e.identifier?.toLowerCase()??"";if(t.includes(n)||a.includes(n)||o.includes(n))return e;r.push(...e.children)}return null}(e,n.scope):null;for(let t of i?[i]:e.children)if(function e(t,n,r,i,a=!1,s=!1){if(t.nodes.length>=t.maxNodes){t.truncated=!0;return}if(r>t.maxDepth)return;let l=t.options.raw||function(e,t,n,r,i){var a,s,l;let u=function(e){let t=O(e.type),n=!!(e.label&&e.label.trim().length>0),r=!!(e.identifier&&e.identifier.trim().length>0);return{type:t,hasMeaningfulText:n&&!E(e.label??""),hasMeaningfulId:r&&!E(e.identifier??""),isStructural:function(e){let t=e.split(".").pop()??e;return t.includes("layout")||"viewgroup"===t||"view"===t}(t),isVisual:"imageview"===t||"imagebutton"===t}}(e);return t.interactiveOnly?function(e,t,n,r,i){var a,s,l,u;return!!(e.hittable||o(t.type)&&r)||(a=t,s=n,l=r,u=i,(!!a.hasMeaningfulText||!!a.hasMeaningfulId)&&!a.isVisual&&(!a.isStructural||!!u)&&(s||l||u))}(e,u,n,r,i):t.compact?u.hasMeaningfulText||u.hasMeaningfulId||!!e.hittable:!u.isStructural&&!u.isVisual||(a=e,s=u,l=r,!!a.hittable||!!s.hasMeaningfulText||!!s.hasMeaningfulId&&!!l||l)}(n,t.options,a,function e(t,n){let r=t.interactiveDescendantMemo.get(n);if(void 0!==r)return r;for(let r of n.children)if(r.hittable||e(t,r))return t.interactiveDescendantMemo.set(n,!0),!0;return t.interactiveDescendantMemo.set(n,!1),!1}(t,n),s)?function(e,t,n,r){let i=e.nodes.length;return e.sourceNodes.push(t),e.nodes.push({index:i,type:t.type??void 0,label:t.label??void 0,value:t.value??void 0,identifier:t.identifier??void 0,rect:t.rect,enabled:t.enabled,hittable:t.hittable,depth:n,parentIndex:r,...t.hiddenContentAbove?{hiddenContentAbove:!0}:{},...t.hiddenContentBelow?{hiddenContentBelow:!0}:{}}),i}(t,n,r,i):i,u=a||!!n.hittable,d=s||function(e){if(!e)return!1;let t=O(e);return t.includes("recyclerview")||t.includes("listview")||t.includes("gridview")}(n.type);for(let i of n.children)if(e(t,i,r+1,l,u,d),t.truncated)return}(r,t,0),r.truncated)break;let a={nodes:r.nodes,sourceNodes:r.sourceNodes,analysis:r.analysis};return r.truncated?{...a,truncated:!0}:a}function _(e,t,n){for(;t<n&&y(e[t]??"");)t+=1;return t}function y(e){return" "===e||"\n"===e||"\r"===e||" "===e}function L(e,t){return e.get(t)??null}function T(e){let t={type:null,label:null,value:null,identifier:null,depth:-1,children:[]},n=[t],r=/<node\b[^>]*>|<\/node>/g,i=r.exec(e);for(;i;){let t=i[0];if(t.startsWith("</node")){n.length>1&&n.pop(),i=r.exec(e);continue}let a=C(t),o=n[n.length-1],s={type:a.className,label:a.text||a.desc,value:a.text,identifier:a.resourceId,rect:a.rect,enabled:a.enabled,hittable:a.clickable??a.focusable,depth:o.depth+1,parentIndex:void 0,children:[]};o.children.push(s),t.endsWith("/>")||n.push(s),i=r.exec(e)}return t}function O(e){return e?e.toLowerCase():""}function E(e){let t=e.trim();return!!t&&/^[\w.]+:id\/[\w.-]+$/i.test(t)}async function R(e){let t,n=e.waitForIdleTimeoutMs??500,r=e.timeoutMs??8e3,i=e.commandTimeoutMs??r+5e3,o=e.maxDepth??128,s=e.maxNodes??5e3,l=e.packageName??u,d=e.instrumentationRunner??`${l}/.SnapshotInstrumentation`,c=["shell","am","instrument","-w","-e","waitForIdleTimeoutMs",String(n),"-e","timeoutMs",String(r),"-e","maxDepth",String(o),"-e","maxNodes",String(s),d],f=await e.adb(c,{allowFailure:!0,timeoutMs:i});try{t=F(`${f.stdout}
2
- ${f.stderr}`)}catch(e){if(e instanceof a&&0!==f.exitCode&&e.details?.helper)throw e;throw new a("COMMAND_FAILED",0===f.exitCode?"Android snapshot helper output could not be parsed":"Android snapshot helper failed before returning parseable output",{stdout:f.stdout,stderr:f.stderr,exitCode:f.exitCode},e)}if(0!==f.exitCode)throw new a("COMMAND_FAILED","Android snapshot helper failed",{stdout:f.stdout,stderr:f.stderr,exitCode:f.exitCode,helper:t.metadata});return t}function F(e){var t,n;let r=function(e){var t;let n={status:[],results:[],currentStatus:null,currentResult:null};for(let t of e.split(/\r?\n/))!function(e,t){if(e.startsWith("INSTRUMENTATION_STATUS: ")){t.currentStatus??={},H(e.slice(24),t.currentStatus);return}if(e.startsWith("INSTRUMENTATION_STATUS_CODE: "))return U(t);if(e.startsWith("INSTRUMENTATION_RESULT: ")){t.currentResult??={},H(e.slice(24),t.currentResult);return}e.startsWith("INSTRUMENTATION_CODE: ")&&$(t)}(t,n);return U(t=n),$(t),{status:n.status,results:n.results}}(e),i=function(e){let t=e.find(e=>e.agentDeviceProtocol===c);if(!t)throw new a("COMMAND_FAILED","Android snapshot helper did not return a final result");if("true"!==t.ok){var n;throw new a("COMMAND_FAILED",(n=t).message&&"null"!==n.message?n.message:n.errorType||"Android snapshot helper returned an error",{errorType:t.errorType,helper:t})}return t}(r.results);return{xml:function(e,t){if(0===e.length)throw new a("COMMAND_FAILED","Android snapshot helper did not return XML chunks",{helper:t});let n=function(e){let t=e[0]?.count??e.length;if(t<1||e.length!==t||e.some(e=>e.count!==t))throw new a("COMMAND_FAILED","Android snapshot helper returned incomplete XML chunks",{expectedChunks:t,actualChunks:e.length});return t}(e),r=Buffer.concat(function(e,t){let n=[];for(let r=0;r<t;r+=1){let i=e.get(r);if(void 0===i)throw new a("COMMAND_FAILED","Android snapshot helper returned incomplete XML chunks",{missingChunkIndex:r,expectedChunks:t});n.push(Buffer.from(i,"base64"))}return n}(function(e,t){let n=new Map;for(let r of e){if(void 0===r.index||r.index<0||r.index>=t)throw new a("COMMAND_FAILED","Android snapshot helper returned invalid chunk index",{chunkIndex:r.index,expectedChunks:t});if(n.has(r.index))throw new a("COMMAND_FAILED","Android snapshot helper returned duplicate XML chunks",{chunkIndex:r.index});n.set(r.index,r.payloadBase64)}return n}(e,n),n)).toString("utf8");if(!r.includes("<hierarchy")||!r.includes("</hierarchy>"))throw new a("COMMAND_FAILED","Android snapshot helper output did not contain XML",{xml:r});return r}(r.status.filter(e=>e.agentDeviceProtocol===c&&e.outputFormat===f&&"string"==typeof e.payloadBase64).map(e=>({index:V(e.chunkIndex),count:V(e.chunkCount),payloadBase64:e.payloadBase64})),i),metadata:{helperApiVersion:(t=i).helperApiVersion,outputFormat:f,waitForIdleTimeoutMs:V(t.waitForIdleTimeoutMs),timeoutMs:V(t.timeoutMs),maxDepth:V(t.maxDepth),maxNodes:V(t.maxNodes),rootPresent:B(t.rootPresent),captureMode:"interactive-windows"===(n=t.captureMode)||"active-window"===n?n:void 0,windowCount:V(t.windowCount),nodeCount:V(t.nodeCount),truncated:B(t.truncated),elapsedMs:V(t.elapsedMs)}}}function P(e,t={outputFormat:f},n={},r=800){return{...S(e,r,n),metadata:t}}function U(e){e.currentStatus&&(e.status.push(e.currentStatus),e.currentStatus=null)}function $(e){e.currentResult&&(e.results.push(e.currentResult),e.currentResult=null)}function H(e,t){let n=e.indexOf("=");n<0||(t[e.slice(0,n)]=e.slice(n+1))}function V(e){if(void 0===e)return;let t=Number(e);return Number.isFinite(t)?t:void 0}function B(e){return"true"===e||"false"!==e&&void 0}let K=new Map;function G(e){j(W(e.deviceKey,e.packageName,e.versionCode))}function W(e,t,n){return e?`${e}\0${t}\0${n}`:void 0}function j(e){e&&K.delete(e)}async function X(e){var t,n,r;let{adb:i,artifact:o}=e,s=e.installPolicy??"missing-or-outdated",l=o.manifest.packageName,u=o.manifest.versionCode;if("never"===s)return{packageName:l,versionCode:u,installed:!1,reason:"skipped"};let d=W(e.deviceKey,l,u),c=d?K.get(d):void 0;if(d&&"always"!==s&&void 0!==c)return{packageName:l,versionCode:u,installedVersionCode:c,installed:!1,reason:"current"};let f=await z(i,l,e.timeoutMs),h=(t=s,n=f,r=u,"never"===t?"skipped":"always"===t?"forced":void 0===n?"missing":n<r?"outdated":"current");if("current"===h){if(void 0===f)throw Error("Expected installed versionCode for current Android snapshot helper");return d&&K.set(d,f),{packageName:l,versionCode:u,installedVersionCode:f,installed:!1,reason:h}}await p(o);let m=await q(i,e.adbProvider??i,o.apkPath,function(e){let t={};for(let n of e.slice(1)){let e=D(n);if(!e)throw new a("INVALID_ARGS",`Android snapshot helper manifest installArgs contains unsupported install flag "${n}".`);t[e]=!0}return t}(g(o.manifest.installArgs)),{packageName:l,timeoutMs:e.timeoutMs});if(0!==m.exitCode)throw j(d),new a("COMMAND_FAILED","Failed to install Android snapshot helper",{packageName:l,versionCode:u,stdout:m.stdout,stderr:m.stderr,exitCode:m.exitCode});return d&&K.set(d,u),{packageName:l,versionCode:u,installedVersionCode:f,installed:!0,reason:h}}async function z(e,t,n){let r=await e(["shell","cmd","package","list","packages","--show-versioncode",t],{allowFailure:!0,timeoutMs:n});if(0===r.exitCode){var i=`${r.stdout}
1
+ import e from"node:crypto";import t from"node:fs";import n from"node:fs/promises";import r from"node:os";import i from"node:path";import{AppError as a}from"./9152.js";import{isScrollableType as o}from"./2842.js";import{installAndroidAdbPackage as s}from"./9639.js";let l="android-snapshot-helper",u="com.callstack.agentdevice.snapshothelper",d="com.callstack.agentdevice.snapshothelper/.SnapshotInstrumentation",c="android-snapshot-helper-v1",f="uiautomator-xml",h={"-r":"replace","-t":"allowTestPackages","-d":"allowDowngrade","-g":"grantPermissions"};async function p(e){let t=await N(e.apkPath);if(t!==e.manifest.sha256)throw new a("COMMAND_FAILED","Android snapshot helper APK checksum mismatch",{apkPath:e.apkPath,expectedSha256:e.manifest.sha256,actualSha256:t})}async function m(e){let t=e.fetch??fetch,o=await t(e.manifestUrl);if(!o.ok)throw new a("COMMAND_FAILED","Failed to download Android snapshot helper manifest",{manifestUrl:e.manifestUrl,status:o.status,statusText:o.statusText});let s=A(JSON.parse((await w(o,65536,"Android snapshot helper manifest")).toString("utf8")));if(!s.apkUrl)throw new a("COMMAND_FAILED","Android snapshot helper manifest does not include apkUrl",{manifestUrl:e.manifestUrl});let l=e.cacheDir??i.join(r.tmpdir(),`agent-device-android-snapshot-helper-${s.version}`),u=!e.cacheDir;await n.mkdir(l,{recursive:!0});let d=s.assetName??`agent-device-android-snapshot-helper-${s.version}.apk`,c=i.join(l,d),f=await t(s.apkUrl);if(!f.ok)throw new a("COMMAND_FAILED","Failed to download Android snapshot helper APK",{apkUrl:s.apkUrl,status:f.status,statusText:f.statusText});await n.writeFile(c,await w(f,0x1400000,"Android snapshot helper APK"));let h={apkPath:c,manifest:s};return await p(h),{...h,cleanup:async()=>{await n.rm(u?l:c,{recursive:u,force:!0})}}}function A(e){var t,n;if(!e||"object"!=typeof e||Array.isArray(e))throw new a("INVALID_ARGS","Android snapshot helper manifest must be an object.");return{name:I(e.name,"name",l),version:M(e.version,"version"),releaseTag:v(e.releaseTag),assetName:v(e.assetName),apkUrl:(t=e.apkUrl,n="apkUrl",null===t?null:M(t,n)),sha256:function(e){let t=M(e,"sha256").trim().toLowerCase();if(64!==t.length||!function(e){for(let t of e){let e=t.charCodeAt(0),n=e>=48&&e<=57,r=e>=97&&e<=102;if(!n&&!r)return!1}return!0}(t))throw new a("INVALID_ARGS","Android snapshot helper manifest sha256 must be a 64-character hex string.");return t}(e.sha256),checksumName:v(e.checksumName),packageName:M(e.packageName,"packageName"),versionCode:x(e.versionCode,"versionCode"),instrumentationRunner:M(e.instrumentationRunner,"instrumentationRunner"),minSdk:x(e.minSdk,"minSdk"),targetSdk:void 0===e.targetSdk?void 0:x(e.targetSdk,"targetSdk"),outputFormat:I(e.outputFormat,"outputFormat",f),statusProtocol:I(e.statusProtocol,"statusProtocol",c),installArgs:g(e.installArgs)}}async function w(e,t,n){let r=e.headers.get("content-length");if(null!==r){let e=Number(r);if(Number.isFinite(e)&&e>t)throw new a("COMMAND_FAILED",`${n} download exceeds size limit`,{contentLength:e,maxBytes:t})}if(!e.body){let r=Buffer.from(await e.arrayBuffer());if(r.length>t)throw new a("COMMAND_FAILED",`${n} download exceeds size limit`,{contentLength:r.length,maxBytes:t});return r}let i=e.body.getReader(),o=[],s=0;try{for(;;){let{done:e,value:r}=await i.read();if(e)break;if((s+=r.byteLength)>t)throw new a("COMMAND_FAILED",`${n} download exceeds size limit`,{contentLength:s,maxBytes:t});o.push(Buffer.from(r))}}finally{i.releaseLock()}return Buffer.concat(o,s)}function g(e){let t=function(e,t){if(!Array.isArray(e)||!e.every(e=>"string"==typeof e))throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} must be a string array.`);return e}(e,"installArgs");if("install"!==t[0])throw new a("INVALID_ARGS",'Android snapshot helper manifest installArgs must start with "install".');if(t.some(e=>e.includes("\0")))throw new a("INVALID_ARGS","Android snapshot helper manifest installArgs must not contain null bytes.");let n=t.slice(1).find(e=>void 0===b(e));if(n)throw new a("INVALID_ARGS",`Android snapshot helper manifest installArgs contains unsupported install flag "${n}".`);return t}async function N(n){return await new Promise((r,i)=>{let a=e.createHash("sha256"),o=t.createReadStream(n);o.on("error",i),o.on("data",e=>a.update(e)),o.on("end",()=>r(a.digest("hex")))})}function M(e,t){if("string"!=typeof e||0===e.trim().length)throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} is required.`);return e}function v(e){return"string"==typeof e&&e.trim().length>0?e:void 0}function x(e,t){if("number"!=typeof e||!Number.isInteger(e))throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} must be an integer.`);return e}function I(e,t,n){if(e!==n)throw new a("INVALID_ARGS",`Android snapshot helper manifest ${t} must be "${n}".`);return n}function b(e){if(Object.hasOwn(h,e))return h[e]}function*D(e){let t=/<node\b[^>]*>/g,n=t.exec(e);for(;n;)yield C(n[0]),n=t.exec(e)}function C(e){let t,n,r=(t=function(e){let t=new Map,n=e.indexOf(" "),r=e.lastIndexOf(">");if(n<0||r<=n)return t;let i=n;for(;i<r&&!((i=_(e,i,r))>=r);){var a;let n=e[i];if("/"===n||">"===n)break;let o=i;for(;i<r&&!("="===(a=e[i]??"")||"/"===a||">"===a||y(a));)i+=1;let s=e.slice(o,i);if(i=_(e,i,r),!s||"="!==e[i])break;i=_(e,i+1,r);let l=e[i];if('"'!==l&&"'"!==l)break;let u=i+=1;for(;i<r&&e[i]!==l;)i+=1;if(i>=r)break;t.set(s,function(e){let t="",n=0;for(;n<e.length;){let r=e.indexOf("&",n);if(r<0){t+=e.slice(n);break}t+=e.slice(n,r);let i=e.indexOf(";",r+1);if(i<0){t+=e.slice(r);break}t+=function(e){switch(e){case"amp":return"&";case"lt":return"<";case"gt":return">";case"quot":return'"';case"apos":return"'";default:return function(e){if(!e.startsWith("#"))return;let t=e[1]?.toLowerCase()==="x"?16:10,n=16===t?e.slice(2):e.slice(1);if(!n||!function(e,t){for(let n of e){let e=n.charCodeAt(0),r=e>=48&&e<=57;if(10===t){if(!r)return!1;continue}let i=e>=65&&e<=70,a=e>=97&&e<=102;if(!r&&!i&&!a)return!1}return!0}(n,t))return;let r=Number.parseInt(n,t);if(Number.isFinite(r))try{return String.fromCodePoint(r)}catch{return}}(e)}}(e.slice(r+1,i))??e.slice(r,i+1),n=i+1}return t}(e.slice(u,i))),i+=1}return t}(e),n=e=>{let n=L(t,e);if(null!==n)return"true"===n},{text:L(t,"text"),desc:L(t,"content-desc"),resourceId:L(t,"resource-id"),packageName:L(t,"package"),className:L(t,"class"),bounds:L(t,"bounds"),clickable:n("clickable"),enabled:n("enabled"),focusable:n("focusable"),focused:n("focused"),password:n("password")}),i=function(e){if(!e)return;let t=/\[(\d+),(\d+)\]\[(\d+),(\d+)\]/.exec(e);if(!t)return;let n=Number(t[1]),r=Number(t[2]);return{x:n,y:r,width:Math.max(0,Number(t[3])-n),height:Math.max(0,Number(t[4])-r)}}(r.bounds);return{...r,...i?{rect:i}:{}}}function S(e,t,n){let{sourceNodes:r,...i}=k(T(e),t,n);return i}function k(e,t,n){let r={nodes:[],sourceNodes:[],maxNodes:t,maxDepth:n.depth??1/0,options:n,analysis:function(e){let t=0,n=0,r=[...e.children];for(;r.length>0;){let e=r.pop();t+=1,n=Math.max(n,e.depth),r.push(...e.children)}return{rawNodeCount:t,maxDepth:n}}(e),interactiveDescendantMemo:new Map,truncated:!1},i=n.scope?function(e,t){let n=t.toLowerCase(),r=[...e.children],i=0;for(;i<r.length;){let e=r[i++],t=e.label?.toLowerCase()??"",a=e.value?.toLowerCase()??"",o=e.identifier?.toLowerCase()??"";if(t.includes(n)||a.includes(n)||o.includes(n))return e;r.push(...e.children)}return null}(e,n.scope):null;for(let t of i?[i]:e.children)if(function e(t,n,r,i,a=!1,s=!1){if(t.nodes.length>=t.maxNodes){t.truncated=!0;return}if(r>t.maxDepth)return;let l=t.options.raw||function(e,t,n,r,i){var a,s,l;let u=function(e){let t=O(e.type),n=!!(e.label&&e.label.trim().length>0),r=!!(e.identifier&&e.identifier.trim().length>0);return{type:t,hasMeaningfulText:n&&!E(e.label??""),hasMeaningfulId:r&&!E(e.identifier??""),isStructural:function(e){let t=e.split(".").pop()??e;return t.includes("layout")||"viewgroup"===t||"view"===t}(t),isVisual:"imageview"===t||"imagebutton"===t}}(e);return t.interactiveOnly?function(e,t,n,r,i){var a,s,l,u;return!!(e.hittable||o(t.type)&&r)||(a=t,s=n,l=r,u=i,(!!a.hasMeaningfulText||!!a.hasMeaningfulId)&&!a.isVisual&&(!a.isStructural||!!u)&&(s||l||u))}(e,u,n,r,i):t.compact?u.hasMeaningfulText||u.hasMeaningfulId||!!e.hittable:!u.isStructural&&!u.isVisual||(a=e,s=u,l=r,!!a.hittable||!!s.hasMeaningfulText||!!s.hasMeaningfulId&&!!l||l)}(n,t.options,a,function e(t,n){let r=t.interactiveDescendantMemo.get(n);if(void 0!==r)return r;for(let r of n.children)if(r.hittable||e(t,r))return t.interactiveDescendantMemo.set(n,!0),!0;return t.interactiveDescendantMemo.set(n,!1),!1}(t,n),s)?function(e,t,n,r){let i=e.nodes.length;return e.sourceNodes.push(t),e.nodes.push({index:i,type:t.type??void 0,label:t.label??void 0,value:t.value??void 0,identifier:t.identifier??void 0,bundleId:t.packageName??void 0,rect:t.rect,enabled:t.enabled,hittable:t.hittable,depth:n,parentIndex:r,...t.hiddenContentAbove?{hiddenContentAbove:!0}:{},...t.hiddenContentBelow?{hiddenContentBelow:!0}:{}}),i}(t,n,r,i):i,u=a||!!n.hittable,d=s||function(e){if(!e)return!1;let t=O(e);return t.includes("recyclerview")||t.includes("listview")||t.includes("gridview")}(n.type);for(let i of n.children)if(e(t,i,r+1,l,u,d),t.truncated)return}(r,t,0),r.truncated)break;let a={nodes:r.nodes,sourceNodes:r.sourceNodes,analysis:r.analysis};return r.truncated?{...a,truncated:!0}:a}function _(e,t,n){for(;t<n&&y(e[t]??"");)t+=1;return t}function y(e){return" "===e||"\n"===e||"\r"===e||" "===e}function L(e,t){return e.get(t)??null}function T(e){let t={type:null,label:null,value:null,identifier:null,packageName:null,depth:-1,children:[]},n=[t],r=/<node\b[^>]*>|<\/node>/g,i=r.exec(e);for(;i;){let t=i[0];if(t.startsWith("</node")){n.length>1&&n.pop(),i=r.exec(e);continue}let a=C(t),o=n[n.length-1],s={type:a.className,label:a.text||a.desc,value:a.text,identifier:a.resourceId,packageName:a.packageName,rect:a.rect,enabled:a.enabled,hittable:a.clickable??a.focusable,depth:o.depth+1,parentIndex:void 0,children:[]};o.children.push(s),t.endsWith("/>")||n.push(s),i=r.exec(e)}return t}function O(e){return e?e.toLowerCase():""}function E(e){let t=e.trim();return!!t&&/^[\w.]+:id\/[\w.-]+$/i.test(t)}async function R(e){let t,n=e.waitForIdleTimeoutMs??500,r=e.timeoutMs??8e3,i=e.commandTimeoutMs??r+5e3,o=e.maxDepth??128,s=e.maxNodes??5e3,l=e.packageName??u,d=e.instrumentationRunner??`${l}/.SnapshotInstrumentation`,c=["shell","am","instrument","-w","-e","waitForIdleTimeoutMs",String(n),"-e","timeoutMs",String(r),"-e","maxDepth",String(o),"-e","maxNodes",String(s),d],f=await e.adb(c,{allowFailure:!0,timeoutMs:i});try{t=F(`${f.stdout}
2
+ ${f.stderr}`)}catch(e){if(e instanceof a&&0!==f.exitCode&&e.details?.helper)throw e;throw new a("COMMAND_FAILED",0===f.exitCode?"Android snapshot helper output could not be parsed":"Android snapshot helper failed before returning parseable output",{stdout:f.stdout,stderr:f.stderr,exitCode:f.exitCode},e)}if(0!==f.exitCode)throw new a("COMMAND_FAILED","Android snapshot helper failed",{stdout:f.stdout,stderr:f.stderr,exitCode:f.exitCode,helper:t.metadata});return t}function F(e){var t,n;let r=function(e){var t;let n={status:[],results:[],currentStatus:null,currentResult:null};for(let t of e.split(/\r?\n/))!function(e,t){if(e.startsWith("INSTRUMENTATION_STATUS: ")){t.currentStatus??={},H(e.slice(24),t.currentStatus);return}if(e.startsWith("INSTRUMENTATION_STATUS_CODE: "))return U(t);if(e.startsWith("INSTRUMENTATION_RESULT: ")){t.currentResult??={},H(e.slice(24),t.currentResult);return}e.startsWith("INSTRUMENTATION_CODE: ")&&$(t)}(t,n);return U(t=n),$(t),{status:n.status,results:n.results}}(e),i=function(e){let t=e.find(e=>e.agentDeviceProtocol===c);if(!t)throw new a("COMMAND_FAILED","Android snapshot helper did not return a final result");if("true"!==t.ok){var n;throw new a("COMMAND_FAILED",(n=t).message&&"null"!==n.message?n.message:n.errorType||"Android snapshot helper returned an error",{errorType:t.errorType,helper:t})}return t}(r.results);return{xml:function(e,t){if(0===e.length)throw new a("COMMAND_FAILED","Android snapshot helper did not return XML chunks",{helper:t});let n=function(e){let t=e[0]?.count??e.length;if(t<1||e.length!==t||e.some(e=>e.count!==t))throw new a("COMMAND_FAILED","Android snapshot helper returned incomplete XML chunks",{expectedChunks:t,actualChunks:e.length});return t}(e),r=Buffer.concat(function(e,t){let n=[];for(let r=0;r<t;r+=1){let i=e.get(r);if(void 0===i)throw new a("COMMAND_FAILED","Android snapshot helper returned incomplete XML chunks",{missingChunkIndex:r,expectedChunks:t});n.push(Buffer.from(i,"base64"))}return n}(function(e,t){let n=new Map;for(let r of e){if(void 0===r.index||r.index<0||r.index>=t)throw new a("COMMAND_FAILED","Android snapshot helper returned invalid chunk index",{chunkIndex:r.index,expectedChunks:t});if(n.has(r.index))throw new a("COMMAND_FAILED","Android snapshot helper returned duplicate XML chunks",{chunkIndex:r.index});n.set(r.index,r.payloadBase64)}return n}(e,n),n)).toString("utf8");if(!r.includes("<hierarchy")||!r.includes("</hierarchy>"))throw new a("COMMAND_FAILED","Android snapshot helper output did not contain XML",{xml:r});return r}(r.status.filter(e=>e.agentDeviceProtocol===c&&e.outputFormat===f&&"string"==typeof e.payloadBase64).map(e=>({index:V(e.chunkIndex),count:V(e.chunkCount),payloadBase64:e.payloadBase64})),i),metadata:{helperApiVersion:(t=i).helperApiVersion,outputFormat:f,waitForIdleTimeoutMs:V(t.waitForIdleTimeoutMs),timeoutMs:V(t.timeoutMs),maxDepth:V(t.maxDepth),maxNodes:V(t.maxNodes),rootPresent:B(t.rootPresent),captureMode:"interactive-windows"===(n=t.captureMode)||"active-window"===n?n:void 0,windowCount:V(t.windowCount),nodeCount:V(t.nodeCount),truncated:B(t.truncated),elapsedMs:V(t.elapsedMs)}}}function P(e,t={outputFormat:f},n={},r=800){return{...S(e,r,n),metadata:t}}function U(e){e.currentStatus&&(e.status.push(e.currentStatus),e.currentStatus=null)}function $(e){e.currentResult&&(e.results.push(e.currentResult),e.currentResult=null)}function H(e,t){let n=e.indexOf("=");n<0||(t[e.slice(0,n)]=e.slice(n+1))}function V(e){if(void 0===e)return;let t=Number(e);return Number.isFinite(t)?t:void 0}function B(e){return"true"===e||"false"!==e&&void 0}let K=new Map;function G(e){j(W(e.deviceKey,e.packageName,e.versionCode))}function W(e,t,n){return e?`${e}\0${t}\0${n}`:void 0}function j(e){e&&K.delete(e)}async function X(e){var t,n,r;let{adb:i,artifact:o}=e,s=e.installPolicy??"missing-or-outdated",l=o.manifest.packageName,u=o.manifest.versionCode;if("never"===s)return{packageName:l,versionCode:u,installed:!1,reason:"skipped"};let d=W(e.deviceKey,l,u),c=d?K.get(d):void 0;if(d&&"always"!==s&&void 0!==c)return{packageName:l,versionCode:u,installedVersionCode:c,installed:!1,reason:"current"};let f=await z(i,l,e.timeoutMs),h=(t=s,n=f,r=u,"never"===t?"skipped":"always"===t?"forced":void 0===n?"missing":n<r?"outdated":"current");if("current"===h){if(void 0===f)throw Error("Expected installed versionCode for current Android snapshot helper");return d&&K.set(d,f),{packageName:l,versionCode:u,installedVersionCode:f,installed:!1,reason:h}}await p(o);let m=await q(i,e.adbProvider??i,o.apkPath,function(e){let t={};for(let n of e.slice(1)){let e=b(n);if(!e)throw new a("INVALID_ARGS",`Android snapshot helper manifest installArgs contains unsupported install flag "${n}".`);t[e]=!0}return t}(g(o.manifest.installArgs)),{packageName:l,timeoutMs:e.timeoutMs});if(0!==m.exitCode)throw j(d),new a("COMMAND_FAILED","Failed to install Android snapshot helper",{packageName:l,versionCode:u,stdout:m.stdout,stderr:m.stderr,exitCode:m.exitCode});return d&&K.set(d,u),{packageName:l,versionCode:u,installedVersionCode:f,installed:!0,reason:h}}async function z(e,t,n){let r=await e(["shell","cmd","package","list","packages","--show-versioncode",t],{allowFailure:!0,timeoutMs:n});if(0===r.exitCode){var i=`${r.stdout}
3
3
  ${r.stderr}`,a=t;let e=`package:${a}`;for(let t of i.split(/\r?\n/)){if(!t.startsWith(e)||t.length>e.length&&!/\s/.test(t[e.length]??""))continue;let n=/(?:^|\s)versionCode:(\d+)(?:\s|$)/.exec(t);if(n)return Number(n[1])}return}}async function q(e,t,n,r,i){var a;let o=async()=>await s(n,{allowFailure:!0,provider:t,...r,timeoutMs:i.timeoutMs}),l=await o();if(0===l.exitCode||(a=l,!`${a.stdout}
4
- ${a.stderr}`.includes("INSTALL_FAILED_UPDATE_INCOMPATIBLE")))return l;let u=await e(["uninstall",i.packageName],{allowFailure:!0,timeoutMs:i.timeoutMs}),d=await o();return 0===d.exitCode?d:{...d,stderr:[d.stderr,u.stderr?`Previous uninstall stderr after INSTALL_FAILED_UPDATE_INCOMPATIBLE: ${u.stderr}`:""].filter(Boolean).join("\n")}}export{l as ANDROID_SNAPSHOT_HELPER_NAME,f as ANDROID_SNAPSHOT_HELPER_OUTPUT_FORMAT,u as ANDROID_SNAPSHOT_HELPER_PACKAGE,c as ANDROID_SNAPSHOT_HELPER_PROTOCOL,d as ANDROID_SNAPSHOT_HELPER_RUNNER,b as androidUiNodes,k as buildUiHierarchySnapshot,R as captureAndroidSnapshotWithHelper,X as ensureAndroidSnapshotHelper,G as forgetAndroidSnapshotHelperInstall,A as parseAndroidSnapshotHelperManifest,F as parseAndroidSnapshotHelperOutput,P as parseAndroidSnapshotHelperXml,S as parseUiHierarchy,T as parseUiHierarchyTree,m as prepareAndroidSnapshotHelperArtifactFromManifestUrl,p as verifyAndroidSnapshotHelperArtifact};
4
+ ${a.stderr}`.includes("INSTALL_FAILED_UPDATE_INCOMPATIBLE")))return l;let u=await e(["uninstall",i.packageName],{allowFailure:!0,timeoutMs:i.timeoutMs}),d=await o();return 0===d.exitCode?d:{...d,stderr:[d.stderr,u.stderr?`Previous uninstall stderr after INSTALL_FAILED_UPDATE_INCOMPATIBLE: ${u.stderr}`:""].filter(Boolean).join("\n")}}export{l as ANDROID_SNAPSHOT_HELPER_NAME,f as ANDROID_SNAPSHOT_HELPER_OUTPUT_FORMAT,u as ANDROID_SNAPSHOT_HELPER_PACKAGE,c as ANDROID_SNAPSHOT_HELPER_PROTOCOL,d as ANDROID_SNAPSHOT_HELPER_RUNNER,D as androidUiNodes,k as buildUiHierarchySnapshot,R as captureAndroidSnapshotWithHelper,X as ensureAndroidSnapshotHelper,G as forgetAndroidSnapshotHelperInstall,A as parseAndroidSnapshotHelperManifest,F as parseAndroidSnapshotHelperOutput,P as parseAndroidSnapshotHelperXml,S as parseUiHierarchy,T as parseUiHierarchyTree,m as prepareAndroidSnapshotHelperArtifactFromManifestUrl,p as verifyAndroidSnapshotHelperArtifact};
package/dist/src/3572.js CHANGED
@@ -1 +1 @@
1
- import e from"node:path";import t from"node:crypto";import n from"node:fs";import{resolveUserPath as i,expandUserHomePath as r}from"./3267.js";import{findProjectRoot as o}from"./9671.js";import{splitSelectorFromArgs as s,tryParseSelectorChain as a}from"./940.js";import{AppError as l}from"./9152.js";function u(t){let n,o=(n=(t??"").trim())?i(n):e.join(r("~"),".agent-device");return{baseDir:o,infoPath:e.join(o,"daemon.json"),lockPath:e.join(o,"daemon.lock"),logPath:e.join(o,"daemon.log"),sessionsDir:e.join(o,"sessions")}}function c(e){let t=(e??"").trim().toLowerCase();return"http"===t?"http":"dual"===t?"dual":"socket"}function d(e){let t=(e??"").trim().toLowerCase();return"auto"===t?"auto":"socket"===t?"socket":"http"===t?"http":"auto"}function p(e){return"tenant"===(e??"").trim().toLowerCase()?"tenant":"none"}function f(e){if(!e)return;let t=e.trim();if(t&&/^[a-zA-Z0-9._-]{1,128}$/.test(t))return t}let m=/(?:^|[^\w$.])(?:import|export)\s+(?:type\s+)?(?:[^'"`]*?\s+from\s+)?['"]([^'"]+)['"]/gm,h=/import\(\s*['"]([^'"]+)['"]\s*\)/gm,g=[".ts",".tsx",".js",".jsx",".mjs",".cjs"];function v(){let e=process.argv[1];return e?S(e):"unknown"}function S(i,r=o()){try{let o=e.resolve(r),s=[e.resolve(i)],a=new Set,l=[];for(;s.length>0;){let t=s.pop();if(!t||a.has(t))continue;a.add(t);let i=n.statSync(t);if(!i.isFile())continue;let r=e.relative(o,t)||t;l.push(`${r}:${i.size}:${Math.trunc(i.mtimeMs)}`);let u=n.readFileSync(t,"utf8");for(let n of function(e){let t=new Set;return b(e,m,t),b(e,h,t),[...t]}(u)){let i=function(t,n){let i=e.resolve(e.dirname(t),n),r=x(i);if(r)return r;for(let e of g){let t=x(`${i}${e}`);if(t)return t}for(let t of g){let n=x(e.join(i,`index${t}`));if(n)return n}return null}(t,n);i&&s.push(i)}}let u=l.sort().join("|"),c=t.createHash("sha1").update(u).digest("hex");return`graph:${l.length}:${c}`}catch{return"unknown"}}function b(e,t,n){t.lastIndex=0;let i=null;for(;null!==(i=t.exec(e));){let e=i[1]?.trim();e?.startsWith(".")&&n.add(e)}}function x(e){try{return n.statSync(e).isFile()?e:null}catch{return null}}let I={alert:"alert",appState:"appstate",appSwitcher:"app-switcher",apps:"apps",back:"back",batch:"batch",boot:"boot",click:"click",close:"close",clipboard:"clipboard",devices:"devices",diff:"diff",fill:"fill",find:"find",focus:"focus",get:"get",home:"home",install:"install",installFromSource:"install-from-source",is:"is",keyboard:"keyboard",logs:"logs",longPress:"longpress",network:"network",open:"open",perf:"perf",pinch:"pinch",press:"press",push:"push",record:"record",reinstall:"reinstall",replay:"replay",rotate:"rotate",scroll:"scroll",screenshot:"screenshot",settings:"settings",snapshot:"snapshot",swipe:"swipe",test:"test",trace:"trace",triggerAppEvent:"trigger-app-event",type:"type",wait:"wait"},w={installSource:"install_source",leaseAllocate:"lease_allocate",leaseHeartbeat:"lease_heartbeat",leaseRelease:"lease_release",releaseMaterializedPaths:"release_materialized_paths",sessionList:"session_list"},N={inventory:y(w.sessionList,I.devices,I.apps),state:y(I.boot,I.appState),observability:y(I.perf,I.logs,I.network),replay:y(I.replay,I.test),snapshot:y(I.snapshot,I.diff,I.wait,I.alert,I.settings),replayScopedAction:y(I.alert,I.back,I.click,I.clipboard,I.diff,I.fill,I.find,I.get,I.home,I.is,I.keyboard,I.longPress,I.pinch,I.press,I.rotate,I.screenshot,I.scroll,I.settings,I.snapshot,I.swipe,I.type,I.wait),selectorValidationExempt:y(w.sessionList,I.devices,w.releaseMaterializedPaths),leaseAdmissionExempt:y(w.sessionList,I.devices,w.releaseMaterializedPaths,w.leaseAllocate,w.leaseHeartbeat,w.leaseRelease)};function y(...e){return new Set(e)}function A(e){return void 0!==e.ref?[e.ref,..._(e.label)]:void 0!==e.selector?[e.selector]:[String(e.x),String(e.y)]}function _(e){return void 0===e?[]:[e]}function M(e){return e.join(" ").trim()||void 0}function C(e){if(!e)return null;let t=Number(e);return Number.isFinite(t)?t:null}function D(e){return{platform:e.platform,target:e.target,device:e.device,udid:e.udid,serial:e.serial,iosSimulatorDeviceSet:e.iosSimulatorDeviceSet,androidDeviceAllowlist:e.androidDeviceAllowlist}}function k(e){return{depth:e.snapshotDepth,scope:e.snapshotScope,raw:e.snapshotRaw}}function R(e){if(0===e.length)return null;let t=C(e[0]);if(null!==t)return{kind:"sleep",durationMs:t};if("text"===e[0]){let t=C(e[e.length-1]);return{kind:"text",text:(null!==t?e.slice(1,-1).join(" "):e.slice(1).join(" ")).trim(),timeoutMs:t}}if(e[0].startsWith("@")){let t=C(e[e.length-1]);return{kind:"ref",rawRef:e[0],timeoutMs:t}}let n=C(e[e.length-1]),i=s(null!==n?e.slice(0,-1):e.slice());return i&&0===i.rest.length&&a(i.selectorExpression)?{kind:"selector",selectorExpression:i.selectorExpression,timeoutMs:n}:{kind:"text",text:(null!==n?e.slice(0,-1).join(" "):e.join(" ")).trim(),timeoutMs:n}}function z(e){return null===e?{}:{timeoutMs:e}}function E(e){if(void 0===e||""===e)throw new l("INVALID_ARGS","find requires query");return e}function L(e,t){if(void 0===e||""===e.trim())throw new l("INVALID_ARGS",`settings location set requires ${t}`);return P(Number(e),t)}function F(e){return{latitude:P(e?.latitude,"latitude"),longitude:P(e?.longitude,"longitude")}}function P(e,t){let n="latitude"===t?-90:-180,i="latitude"===t?90:180;if("number"!=typeof e||!Number.isFinite(e)||e<n||e>i)throw new l("INVALID_ARGS",`${t} must be a number from ${n} to ${i}`);return e}let j=H("wifi","airplane","location","animations"),O=H("on","off"),T=H("light","dark","toggle"),q=H("faceid","touchid"),$=H("match","nonmatch","enroll","unenroll"),G=H("match","nonmatch"),V=H("grant","deny","reset"),B=H("camera","microphone","photos","contacts","contacts-limited","notifications","calendar","location","location-always","media-library","motion","reminders","siri","accessibility","screen-recording","input-monitoring");function H(...e){return new Set(e)}function K(e,t){return void 0!==e&&t.has(e)}let U={decode:function(e){if(e[0]?.startsWith("@"))return{ref:e[0],label:M(e.slice(1))};let t=s(e);return t?{selector:t.selectorExpression}:{x:Number(e[0]),y:Number(e[1])}},encode:A},W={decode:function(e){if(e[0]?.startsWith("@"))return{ref:e[0],label:M(e.slice(1))};let t=e.join(" ").trim();if(!t)throw new l("INVALID_ARGS","get requires @ref or selector expression");return{selector:t}},encode:function(e){return void 0!==e.ref?[e.ref,..._(e.label)]:[e.selector]}},Y={decode:function(e){if(e[0]?.startsWith("@")){let t=e.length>=3?e.slice(2).join(" "):e.slice(1).join(" ");return{kind:"ref",target:{ref:e[0],label:e.length>=3?M([e[1]]):void 0},text:t}}let t=s(e,{preferTrailingValue:!0});return t?{kind:"selector",target:{selector:t.selectorExpression},text:t.rest.join(" ")}:{kind:"point",target:{x:Number(e[0]),y:Number(e[1])},text:e.slice(2).join(" ")}},encode:function(e){return[...A(e),e.text]}},Z={decode:function(e,t){let n=R(e);if(!n)throw new l("INVALID_ARGS","wait requires <ms>, text <text>, @ref, or <selector> [timeoutMs].");let i={...D(t),...k(t)};if("sleep"===n.kind)return{...i,durationMs:n.durationMs};if("text"===n.kind){if(!n.text)throw new l("INVALID_ARGS","wait requires text.");return{...i,text:n.text,...z(n.timeoutMs)}}return"ref"===n.kind?{...i,ref:n.rawRef,...z(n.timeoutMs)}:{...i,selector:n.selectorExpression,...z(n.timeoutMs)}},encode:function(e){if(1!==[void 0!==e.durationMs?"durationMs":void 0,void 0!==e.text?"text":void 0,void 0!==e.ref?"ref":void 0,void 0!==e.selector?"selector":void 0].filter(Boolean).length)throw new l("INVALID_ARGS","wait command requires exactly one of durationMs, text, ref, or selector.");if(void 0!==e.durationMs)return[String(e.durationMs)];let t=void 0!==e.timeoutMs?[String(e.timeoutMs)]:[];if(void 0!==e.text)return["text",e.text,...t];if(void 0!==e.ref)return[e.ref,...t];let n=e.selector;return function(e){if(!a(e))throw new l("INVALID_ARGS",`Invalid wait selector: ${e}`)}(n),[n,...t]}},J={decode:function(e,t){var n,i;let r={...{depth:(n=t).snapshotDepth,raw:n.snapshotRaw},...D(t),first:t.findFirst,last:t.findLast},o=function(e){if("text"===e||"label"===e||"value"===e||"role"===e||"id"===e)return e}(e[0]),s=void 0!==o,a=s?e[1]:e[0],u=s?2:1,c=e[u];if(void 0===c)return{...r,locator:o,query:E(a)};if("get"===c){let t=e[u+1];if("text"===t)return{...r,locator:o,query:E(a),action:"getText"};if("attrs"===t)return{...r,locator:o,query:E(a),action:"getAttrs"};throw new l("INVALID_ARGS","find get only supports text or attrs")}if("wait"===c){return{...r,locator:o,query:E(a),action:"wait",timeoutMs:void 0===(i=e[u+1])?void 0:Number(i)}}if("fill"===c||"type"===c)return{...r,locator:o,query:E(a),action:c,value:e.slice(u+1).join(" ")};if("click"===c||"focus"===c||"exists"===c)return{...r,locator:o,query:E(a),action:c};throw new l("INVALID_ARGS",`Unsupported find action: ${c}`)},encode:function(e){let t=e.locator&&"any"!==e.locator?[e.locator,e.query]:[e.query];switch(e.action){case void 0:case"click":case"focus":case"exists":return e.action?[...t,e.action]:t;case"getText":return[...t,"get","text"];case"getAttrs":return[...t,"get","attrs"];case"wait":var n;return[...t,"wait",...void 0===(n=e.timeoutMs)?[]:[String(n)]];case"fill":case"type":return[...t,e.action,e.value]}}},Q={decode:function(e,t){let n={...k(t),...D(t)},i=e[0],r=s(e.slice(1),{preferTrailingValue:"text"===i});if(!r)throw new l("INVALID_ARGS","is requires a selector expression");if("text"===i)return{...n,predicate:i,selector:r.selectorExpression,value:r.rest.join(" ")};if("visible"===i||"hidden"===i||"exists"===i||"editable"===i||"selected"===i)return{...n,predicate:i,selector:r.selectorExpression};throw new l("INVALID_ARGS","is requires predicate: visible|hidden|exists|editable|selected|text")},encode:function(e){return[e.predicate,e.selector,..."text"===e.predicate?[e.value]:[]]}},X={decode:function(e,t){let n=D(t),i=e[0],r=e[1];if(K(i,j)&&K(r,O))return{...n,setting:i,state:r};if("location"===i&&"set"===r)return{...n,setting:i,state:r,latitude:L(e[2],"latitude"),longitude:L(e[3],"longitude")};if("appearance"===i&&K(r,T)||K(i,q)&&K(r,$)||"fingerprint"===i&&K(r,G))return{...n,setting:i,state:r};if("permission"===i&&K(r,V))return{...n,setting:i,state:r,permission:function(e){if(K(e,B))return e;throw new l("INVALID_ARGS","settings permission requires a permission target.")}(e[2]),mode:function(e){if(void 0===e||"full"===e||"limited"===e)return e;throw new l("INVALID_ARGS","settings permission mode must be full or limited.")}(e[3])};throw new l("INVALID_ARGS","Invalid settings arguments.")},encode:function(e){var t;return"location"===e.setting&&"set"===e.state?[e.setting,e.state,String(e.latitude),String(e.longitude)]:"permission"===e.setting?[e.setting,e.state,e.permission,...void 0===(t=e.mode)?[]:[t]]:[e.setting,e.state]}},ee=["out","overlayRefs","screenshotFullscreen","screenshotMaxSize","screenshotNoStabilize"],et=["screenshotFullscreen","screenshotMaxSize","screenshotNoStabilize"],en=[{key:"screenshotFullscreen",names:["--fullscreen"],type:"boolean",usageLabel:"--fullscreen",usageDescription:"Screenshot: capture the full screen instead of the app window"},{key:"screenshotMaxSize",names:["--max-size"],type:"int",min:1,usageLabel:"--max-size <px>",usageDescription:"Screenshot: downscale so the longest edge is at most <px>"},{key:"screenshotNoStabilize",names:["--no-stabilize"],type:"boolean",usageLabel:"--no-stabilize",usageDescription:"Screenshot: skip Android demo-mode/status-bar stabilization and settle delay for low-latency capture loops"}];function ei(e){return ea({overlayRefs:e?.overlayRefs,fullscreen:e?.screenshotFullscreen,maxSize:e?.screenshotMaxSize,stabilize:!e?.screenshotNoStabilize&&void 0})}function er(e={}){return ea({overlayRefs:e.overlayRefs,screenshotFullscreen:e.screenshotFullscreen??e.fullscreen,screenshotMaxSize:e.screenshotMaxSize??e.maxSize,screenshotNoStabilize:e.screenshotNoStabilize??(!1===e.stabilize||void 0)})}function eo(e,t){t?.screenshotFullscreen&&e.push("--fullscreen"),"number"==typeof t?.screenshotMaxSize&&e.push("--max-size",String(t.screenshotMaxSize)),t?.screenshotNoStabilize&&e.push("--no-stabilize")}function es(e){let{args:t,flags:n,index:i}=e,r=t[i];if("--fullscreen"===r)return n.screenshotFullscreen=!0,{handled:!0,nextIndex:i};if("--no-stabilize"===r)return n.screenshotNoStabilize=!0,{handled:!0,nextIndex:i};if("--max-size"===r){let e=t[i+1],r=void 0===e?NaN:Number(e);if(!Number.isInteger(r)||r<1)throw new l("INVALID_ARGS","screenshot --max-size requires a positive integer");return n.screenshotMaxSize=r,{handled:!0,nextIndex:i+1}}return{handled:!1}}function ea(e){return Object.fromEntries(Object.entries(e).filter(e=>void 0!==e[1]))}let el={apple:{simulator:!0,device:!0},android:{emulator:!0,device:!0,unknown:!0},linux:{device:!0}};function eu(e){return e}function ec(e){return Object.fromEntries(e.map(e=>[e.name,e.schema]))}function ed(e){return Object.fromEntries(e.map(e=>[e.name,e.capability]))}let ep={decode:(e,t)=>({text:e.join(" "),delayMs:t?.delayMs}),encode:e=>[e.text]},ef=eu({name:I.type,schema:{helpDescription:"Type text in focused field",positionalArgs:["text"],allowsExtraPositionals:!0,allowedFlags:["delayMs"]},capability:el,codec:ep}),em=[ef],eh=ec(em),eg=ed(em);function ev(e){return e?{message:e}:{}}function eS(e,t){return t?{...e,message:t}:e}function eb(e){return"string"==typeof e?.message&&e.message.length>0?e.message:null}function ex(e){let t=e.appId??e.bundleId??e.packageName;return{session:e.session,appId:t,appBundleId:e.bundleId,package:e.packageName}}function eI(e,t,n){return{deviceId:t,deviceName:n,..."android"===e?{serial:t}:"ios"===e?{udid:t}:{}}}function ew(e,t={}){let n=t.includeAndroidSerial??!0;return{platform:e.platform,target:e.target,device:e.name,id:e.id,..."ios"===e.platform?{device_udid:e.ios?.udid??e.id,ios_simulator_device_set:e.ios?.simulatorSetPath??null}:{},..."android"===e.platform&&n?{serial:e.android?.serial??e.id}:{}}}function eN(e){return{name:e.name,...ew(e.device,{includeAndroidSerial:!1}),createdAt:e.createdAt}}function ey(e){return{platform:e.platform,id:e.id,name:e.name,kind:e.kind,target:e.target,..."boolean"==typeof e.booted?{booted:e.booted}:{}}}function eA(e){return e.bundleId??e.package??e.app}function e_(e){return eS({app:e.app,appPath:e.appPath,platform:e.platform,...e.appId?{appId:e.appId}:{},...e.bundleId?{bundleId:e.bundleId}:{},...e.package?{package:e.package}:{}},`Installed: ${eA(e)}`)}function eM(e){return e.appName??e.bundleId??e.packageName??e.launchTarget}function eC(e){return eS({launchTarget:e.launchTarget,...e.appName?{appName:e.appName}:{},...e.appId?{appId:e.appId}:{},...e.bundleId?{bundleId:e.bundleId}:{},...e.packageName?{package:e.packageName}:{},...e.installablePath?{installablePath:e.installablePath}:{},...e.archivePath?{archivePath:e.archivePath}:{},...e.materializationId?{materializationId:e.materializationId}:{},...e.materializationExpiresAt?{materializationExpiresAt:e.materializationExpiresAt}:{}},`Installed: ${eM(e)}`)}function eD(e){let t=e.appName??e.appBundleId??e.session;return eS({session:e.session,...e.appName?{appName:e.appName}:{},...e.appBundleId?{appBundleId:e.appBundleId}:{},...e.startup?{startup:e.startup}:{},...e.runtime?{runtime:e.runtime}:{},...e.device?ew(e.device):{}},t?`Opened: ${t}`:"Opened")}function ek(e){return{session:e.session,...e.shutdown?{shutdown:e.shutdown}:{},...ev(e.session?`Closed: ${e.session}`:"Closed")}}function eR(e){return{nodes:e.nodes,truncated:e.truncated,...e.appName?{appName:e.appName}:{},...e.appBundleId?{appBundleId:e.appBundleId}:{},...e.visibility?{visibility:e.visibility}:{},...e.androidSnapshot?{androidSnapshot:e.androidSnapshot}:{},...e.warnings&&e.warnings.length>0?{warnings:e.warnings}:{},...e.unchanged?{unchanged:e.unchanged}:{}}}function ez(e,t,n,i){let r=n(e[t]);if(void 0===r)throw new l("COMMAND_FAILED",i,{response:e});return r}function eE(e,t){return ez(e,t,eV,`Daemon response is missing "${t}".`)}function eL(e,t){return eV(e[t])}function eF(e,t){var n;let i;return n=eV,null===(i=e[t])?null:n(i)}function eP(e,t){return ez(e,t,eB,`Daemon response is missing numeric "${t}".`)}function ej(e,t){return ez(e,t,eH,`Daemon response has invalid "${t}".`)}function eO(e,t){return ez(e,t,eK,`Daemon response has invalid "${t}".`)}function eT(e,t){return function(e){return"tv"===e||"mobile"===e||"desktop"===e?e:void 0}(e[t])??"mobile"}function eq(e,t){let n=e[t];if(!eW(n))return;let i=eG(n,"x"),r=eG(n,"y"),o=eG(n,"width"),s=eG(n,"height");if(void 0!==i&&void 0!==r&&void 0!==o&&void 0!==s)return{x:i,y:r,width:o,height:s}}function e$(e,t){let n=e[t];if(!eW(n))return;let i=eG(n,"x"),r=eG(n,"y");if(void 0!==i&&void 0!==r)return{x:i,y:r}}function eG(e,t){let n=e[t];return"number"==typeof n?n:void 0}function eV(e){return"string"==typeof e&&e.length>0?e:void 0}function eB(e){return"number"==typeof e&&Number.isFinite(e)?e:void 0}function eH(e){return"ios"===e||"macos"===e||"android"===e||"linux"===e?e:void 0}function eK(e){return"simulator"===e||"emulator"===e||"device"===e?e:void 0}function eU(e){if(!eW(e))throw new l("COMMAND_FAILED","Daemon returned an unexpected response shape.",{value:e});return e}function eW(e){return"object"==typeof e&&null!==e}function eY(e){let t={};for(let[n,i]of Object.entries(e))void 0!==i&&(t[n]=i);return t}function eZ(e){return e.split("\n").map(e=>e.trim()).filter(e=>e.length>0)}export{el as ALL_DEVICE_COMMAND_CAPABILITY,N as DAEMON_COMMAND_GROUPS,eg as INTERACTION_COMMAND_CAPABILITIES,eh as INTERACTION_COMMAND_SCHEMAS,w as INTERNAL_COMMANDS,I as PUBLIC_COMMANDS,et as SCREENSHOT_ACTION_FLAG_KEYS,ee as SCREENSHOT_COMMAND_FLAG_KEYS,en as SCREENSHOT_SPECIFIC_FLAG_DEFINITIONS,eo as appendScreenshotScriptFlags,eU as asRecord,ex as buildAppIdentifiers,eI as buildDeviceIdentifiers,ed as commandCapabilityMap,ec as commandSchemaMap,S as computeDaemonCodeSignature,eu as defineCommand,W as elementTargetCodec,Y as fillCommandCodec,J as findCommandCodec,U as interactionTargetCodec,Q as isCommandCodec,eW as isRecord,f as normalizeTenantId,C as parseTimeout,R as parseWaitPositionals,eY as parsing_stripUndefined,eb as readCommandMessage,eT as readDeviceTarget,L as readLocationCoordinate,eF as readNullableString,eL as readOptionalString,e$ as readPoint,eq as readRect,eO as readRequiredDeviceKind,eP as readRequiredNumber,ej as readRequiredPlatform,eE as readRequiredString,es as readScreenshotScriptFlag,F as requireLocationCoordinates,v as resolveDaemonCodeSignature,u as resolveDaemonPaths,c as resolveDaemonServerMode,d as resolveDaemonTransportPreference,eA as resolveDeployResultTarget,eM as resolveInstallFromSourceResultTarget,p as resolveSessionIsolationMode,er as screenshotFlagsFromOptions,ei as screenshotOptionsFromFlags,D as selectionOptionsFromFlags,k as selectorSnapshotOptionsFromFlags,ek as serializeCloseResult,e_ as serializeDeployResult,ey as serializeDevice,eC as serializeInstallFromSourceResult,eD as serializeOpenResult,eN as serializeSessionListEntry,eR as serializeSnapshotResult,X as settingsCommandCodec,eZ as splitNonEmptyTrimmedLines,ev as successText,ep as typeCommandCodec,ef as typeCommandDefinition,Z as waitCommandCodec,eS as withSuccessText};
1
+ import e from"node:path";import t from"node:crypto";import n from"node:fs";import{resolveUserPath as i,expandUserHomePath as r}from"./3267.js";import{findProjectRoot as o}from"./9671.js";import{splitSelectorFromArgs as s,tryParseSelectorChain as a}from"./940.js";import{AppError as l}from"./9152.js";function u(t){let n,o=(n=(t??"").trim())?i(n):e.join(r("~"),".agent-device");return{baseDir:o,infoPath:e.join(o,"daemon.json"),lockPath:e.join(o,"daemon.lock"),logPath:e.join(o,"daemon.log"),sessionsDir:e.join(o,"sessions")}}function c(e){let t=(e??"").trim().toLowerCase();return"http"===t?"http":"dual"===t?"dual":"socket"}function d(e){let t=(e??"").trim().toLowerCase();return"auto"===t?"auto":"socket"===t?"socket":"http"===t?"http":"auto"}function p(e){return"tenant"===(e??"").trim().toLowerCase()?"tenant":"none"}function f(e){if(!e)return;let t=e.trim();if(t&&/^[a-zA-Z0-9._-]{1,128}$/.test(t))return t}let m=/(?:^|[^\w$.])(?:import|export)\s+(?:type\s+)?(?:[^'"`]*?\s+from\s+)?['"]([^'"]+)['"]/gm,h=/import\(\s*['"]([^'"]+)['"]\s*\)/gm,g=[".ts",".tsx",".js",".jsx",".mjs",".cjs"];function v(){let e=process.argv[1];return e?S(e):"unknown"}function S(i,r=o()){try{let o=e.resolve(r),s=[e.resolve(i)],a=new Set,l=[];for(;s.length>0;){let t=s.pop();if(!t||a.has(t))continue;a.add(t);let i=n.statSync(t);if(!i.isFile())continue;let r=e.relative(o,t)||t;l.push(`${r}:${i.size}:${Math.trunc(i.mtimeMs)}`);let u=n.readFileSync(t,"utf8");for(let n of function(e){let t=new Set;return b(e,m,t),b(e,h,t),[...t]}(u)){let i=function(t,n){let i=e.resolve(e.dirname(t),n),r=x(i);if(r)return r;for(let e of g){let t=x(`${i}${e}`);if(t)return t}for(let t of g){let n=x(e.join(i,`index${t}`));if(n)return n}return null}(t,n);i&&s.push(i)}}let u=l.sort().join("|"),c=t.createHash("sha1").update(u).digest("hex");return`graph:${l.length}:${c}`}catch{return"unknown"}}function b(e,t,n){t.lastIndex=0;let i=null;for(;null!==(i=t.exec(e));){let e=i[1]?.trim();e?.startsWith(".")&&n.add(e)}}function x(e){try{return n.statSync(e).isFile()?e:null}catch{return null}}let I={alert:"alert",appState:"appstate",appSwitcher:"app-switcher",apps:"apps",back:"back",batch:"batch",boot:"boot",click:"click",close:"close",clipboard:"clipboard",devices:"devices",diff:"diff",fill:"fill",find:"find",focus:"focus",get:"get",home:"home",install:"install",installFromSource:"install-from-source",is:"is",keyboard:"keyboard",logs:"logs",longPress:"longpress",network:"network",open:"open",perf:"perf",pinch:"pinch",press:"press",push:"push",record:"record",reactNative:"react-native",reinstall:"reinstall",replay:"replay",rotate:"rotate",scroll:"scroll",screenshot:"screenshot",settings:"settings",snapshot:"snapshot",swipe:"swipe",test:"test",trace:"trace",triggerAppEvent:"trigger-app-event",type:"type",wait:"wait"},w={installSource:"install_source",leaseAllocate:"lease_allocate",leaseHeartbeat:"lease_heartbeat",leaseRelease:"lease_release",releaseMaterializedPaths:"release_materialized_paths",sessionList:"session_list"},N={inventory:y(w.sessionList,I.devices,I.apps),state:y(I.boot,I.appState),observability:y(I.perf,I.logs,I.network),replay:y(I.replay,I.test),snapshot:y(I.snapshot,I.diff,I.wait,I.alert,I.settings),replayScopedAction:y(I.alert,I.back,I.click,I.clipboard,I.diff,I.fill,I.find,I.get,I.home,I.is,I.keyboard,I.longPress,I.pinch,I.press,I.record,I.reactNative,I.rotate,I.screenshot,I.scroll,I.settings,I.snapshot,I.swipe,I.type,I.wait),selectorValidationExempt:y(w.sessionList,I.devices,w.releaseMaterializedPaths),leaseAdmissionExempt:y(w.sessionList,I.devices,w.releaseMaterializedPaths,w.leaseAllocate,w.leaseHeartbeat,w.leaseRelease)};function y(...e){return new Set(e)}function A(e){if(e[0]?.startsWith("@")){let t=C(e.slice(1));return{ref:e[0],...void 0===t?{}:{label:t}}}let t=s(e);return t?{selector:t.selectorExpression}:{x:Number(e[0]),y:Number(e[1])}}function _(e){return void 0!==e.ref?[e.ref,...M(e.label)]:void 0!==e.selector?[e.selector]:[String(e.x),String(e.y)]}function M(e){return void 0===e?[]:[e]}function C(e){return e.join(" ").trim()||void 0}function D(e){return void 0!==e&&""!==e.trim()&&Number.isFinite(Number(e))}function k(e){if(!e)return null;let t=Number(e);return Number.isFinite(t)?t:null}function R(e){return{platform:e.platform,target:e.target,device:e.device,udid:e.udid,serial:e.serial,iosSimulatorDeviceSet:e.iosSimulatorDeviceSet,androidDeviceAllowlist:e.androidDeviceAllowlist}}function z(e){return{depth:e.snapshotDepth,scope:e.snapshotScope,raw:e.snapshotRaw}}function E(e){if(0===e.length)return null;let t=k(e[0]);if(null!==t)return{kind:"sleep",durationMs:t};if("text"===e[0]){let t=k(e[e.length-1]);return{kind:"text",text:(null!==t?e.slice(1,-1).join(" "):e.slice(1).join(" ")).trim(),timeoutMs:t}}if(e[0].startsWith("@")){let t=k(e[e.length-1]);return{kind:"ref",rawRef:e[0],timeoutMs:t}}let n=k(e[e.length-1]),i=s(null!==n?e.slice(0,-1):e.slice());return i&&0===i.rest.length&&a(i.selectorExpression)?{kind:"selector",selectorExpression:i.selectorExpression,timeoutMs:n}:{kind:"text",text:(null!==n?e.slice(0,-1).join(" "):e.join(" ")).trim(),timeoutMs:n}}function L(e){return null===e?{}:{timeoutMs:e}}function F(e){if(void 0===e||""===e)throw new l("INVALID_ARGS","find requires query");return e}function P(e,t){if(void 0===e||""===e.trim())throw new l("INVALID_ARGS",`settings location set requires ${t}`);return O(Number(e),t)}function j(e){return{latitude:O(e?.latitude,"latitude"),longitude:O(e?.longitude,"longitude")}}function O(e,t){let n="latitude"===t?-90:-180,i="latitude"===t?90:180;if("number"!=typeof e||!Number.isFinite(e)||e<n||e>i)throw new l("INVALID_ARGS",`${t} must be a number from ${n} to ${i}`);return e}let T=U("wifi","airplane","location","animations"),q=U("on","off"),$=U("light","dark","toggle"),G=U("faceid","touchid"),V=U("match","nonmatch","enroll","unenroll"),B=U("match","nonmatch"),H=U("grant","deny","reset"),K=U("camera","microphone","photos","contacts","contacts-limited","notifications","calendar","location","location-always","media-library","motion","reminders","siri","accessibility","screen-recording","input-monitoring");function U(...e){return new Set(e)}function W(e,t){return void 0!==e&&t.has(e)}let Y={decode:A,encode:_},Z={decode:function(e){if(e[0]?.startsWith("@"))return{ref:e[0],label:C(e.slice(1))};let t=e.join(" ").trim();if(!t)throw new l("INVALID_ARGS","get requires @ref or selector expression");return{selector:t}},encode:function(e){return void 0!==e.ref?[e.ref,...M(e.label)]:[e.selector]}},J={decode:function(e){if(e[0]?.startsWith("@")){let t=e.length>=3?e.slice(2).join(" "):e.slice(1).join(" ");return{kind:"ref",target:{ref:e[0],label:e.length>=3?C([e[1]]):void 0},text:t}}let t=s(e,{preferTrailingValue:!0});return t?{kind:"selector",target:{selector:t.selectorExpression},text:t.rest.join(" ")}:{kind:"point",target:{x:Number(e[0]),y:Number(e[1])},text:e.slice(2).join(" ")}},encode:function(e){return[..._(e),e.text]}},Q={decode:function(e){let t=function(e){if(D(e[0])&&D(e[1]))return{target:e.slice(0,2),...void 0!==e[2]?{durationMs:Number(e[2])}:{}};let t=e.at(-1);return e.length>1&&D(t)?{target:e.slice(0,-1),durationMs:Number(t)}:{target:e}}(e);return{...A(t.target),...void 0!==t.durationMs?{durationMs:t.durationMs}:{}}},encode:function(e){return[..._(e),...void 0===e.durationMs?[]:[String(e.durationMs)]]}},X={decode:function(e,t){let n=E(e);if(!n)throw new l("INVALID_ARGS","wait requires <ms>, text <text>, @ref, or <selector> [timeoutMs].");let i={...R(t),...z(t)};if("sleep"===n.kind)return{...i,durationMs:n.durationMs};if("text"===n.kind){if(!n.text)throw new l("INVALID_ARGS","wait requires text.");return{...i,text:n.text,...L(n.timeoutMs)}}return"ref"===n.kind?{...i,ref:n.rawRef,...L(n.timeoutMs)}:{...i,selector:n.selectorExpression,...L(n.timeoutMs)}},encode:function(e){if(1!==[void 0!==e.durationMs?"durationMs":void 0,void 0!==e.text?"text":void 0,void 0!==e.ref?"ref":void 0,void 0!==e.selector?"selector":void 0].filter(Boolean).length)throw new l("INVALID_ARGS","wait command requires exactly one of durationMs, text, ref, or selector.");if(void 0!==e.durationMs)return[String(e.durationMs)];let t=void 0!==e.timeoutMs?[String(e.timeoutMs)]:[];if(void 0!==e.text)return["text",e.text,...t];if(void 0!==e.ref)return[e.ref,...t];let n=e.selector;return function(e){if(!a(e))throw new l("INVALID_ARGS",`Invalid wait selector: ${e}`)}(n),[n,...t]}},ee={decode:function(e,t){var n,i;let r={...{depth:(n=t).snapshotDepth,raw:n.snapshotRaw},...R(t),first:t.findFirst,last:t.findLast},o=function(e){if("text"===e||"label"===e||"value"===e||"role"===e||"id"===e)return e}(e[0]),s=void 0!==o,a=s?e[1]:e[0],u=s?2:1,c=e[u];if(void 0===c)return{...r,locator:o,query:F(a)};if("get"===c){let t=e[u+1];if("text"===t)return{...r,locator:o,query:F(a),action:"getText"};if("attrs"===t)return{...r,locator:o,query:F(a),action:"getAttrs"};throw new l("INVALID_ARGS","find get only supports text or attrs")}if("wait"===c){return{...r,locator:o,query:F(a),action:"wait",timeoutMs:void 0===(i=e[u+1])?void 0:Number(i)}}if("fill"===c||"type"===c)return{...r,locator:o,query:F(a),action:c,value:e.slice(u+1).join(" ")};if("click"===c||"focus"===c||"exists"===c)return{...r,locator:o,query:F(a),action:c};throw new l("INVALID_ARGS",`Unsupported find action: ${c}`)},encode:function(e){let t=e.locator&&"any"!==e.locator?[e.locator,e.query]:[e.query];switch(e.action){case void 0:case"click":case"focus":case"exists":return e.action?[...t,e.action]:t;case"getText":return[...t,"get","text"];case"getAttrs":return[...t,"get","attrs"];case"wait":var n;return[...t,"wait",...void 0===(n=e.timeoutMs)?[]:[String(n)]];case"fill":case"type":return[...t,e.action,e.value]}}},et={decode:function(e,t){let n={...z(t),...R(t)},i=e[0],r=s(e.slice(1),{preferTrailingValue:"text"===i});if(!r)throw new l("INVALID_ARGS","is requires a selector expression");if("text"===i)return{...n,predicate:i,selector:r.selectorExpression,value:r.rest.join(" ")};if("visible"===i||"hidden"===i||"exists"===i||"editable"===i||"selected"===i)return{...n,predicate:i,selector:r.selectorExpression};throw new l("INVALID_ARGS","is requires predicate: visible|hidden|exists|editable|selected|text")},encode:function(e){return[e.predicate,e.selector,..."text"===e.predicate?[e.value]:[]]}},en={decode:function(e,t){let n=R(t),i=e[0],r=e[1];if(W(i,T)&&W(r,q))return{...n,setting:i,state:r};if("location"===i&&"set"===r)return{...n,setting:i,state:r,latitude:P(e[2],"latitude"),longitude:P(e[3],"longitude")};if("appearance"===i&&W(r,$)||W(i,G)&&W(r,V)||"fingerprint"===i&&W(r,B))return{...n,setting:i,state:r};if("permission"===i&&W(r,H))return{...n,setting:i,state:r,permission:function(e){if(W(e,K))return e;throw new l("INVALID_ARGS","settings permission requires a permission target.")}(e[2]),mode:function(e){if(void 0===e||"full"===e||"limited"===e)return e;throw new l("INVALID_ARGS","settings permission mode must be full or limited.")}(e[3])};throw new l("INVALID_ARGS","Invalid settings arguments.")},encode:function(e){var t;return"location"===e.setting&&"set"===e.state?[e.setting,e.state,String(e.latitude),String(e.longitude)]:"permission"===e.setting?[e.setting,e.state,e.permission,...void 0===(t=e.mode)?[]:[t]]:[e.setting,e.state]}},ei=["out","overlayRefs","screenshotFullscreen","screenshotMaxSize","screenshotNoStabilize"],er=["screenshotFullscreen","screenshotMaxSize","screenshotNoStabilize"],eo=[{key:"screenshotFullscreen",names:["--fullscreen"],type:"boolean",usageLabel:"--fullscreen",usageDescription:"Screenshot: capture the full screen instead of the app window"},{key:"screenshotMaxSize",names:["--max-size"],type:"int",min:1,usageLabel:"--max-size <px>",usageDescription:"Screenshot: downscale so the longest edge is at most <px>"},{key:"screenshotNoStabilize",names:["--no-stabilize"],type:"boolean",usageLabel:"--no-stabilize",usageDescription:"Screenshot: skip Android demo-mode/status-bar stabilization and settle delay for low-latency capture loops"}];function es(e){return ec({overlayRefs:e?.overlayRefs,fullscreen:e?.screenshotFullscreen,maxSize:e?.screenshotMaxSize,stabilize:!e?.screenshotNoStabilize&&void 0})}function ea(e={}){return ec({overlayRefs:e.overlayRefs,screenshotFullscreen:e.screenshotFullscreen??e.fullscreen,screenshotMaxSize:e.screenshotMaxSize??e.maxSize,screenshotNoStabilize:e.screenshotNoStabilize??(!1===e.stabilize||void 0)})}function el(e,t){t?.screenshotFullscreen&&e.push("--fullscreen"),"number"==typeof t?.screenshotMaxSize&&e.push("--max-size",String(t.screenshotMaxSize)),t?.screenshotNoStabilize&&e.push("--no-stabilize")}function eu(e){let{args:t,flags:n,index:i}=e,r=t[i];if("--fullscreen"===r)return n.screenshotFullscreen=!0,{handled:!0,nextIndex:i};if("--no-stabilize"===r)return n.screenshotNoStabilize=!0,{handled:!0,nextIndex:i};if("--max-size"===r){let e=t[i+1],r=void 0===e?NaN:Number(e);if(!Number.isInteger(r)||r<1)throw new l("INVALID_ARGS","screenshot --max-size requires a positive integer");return n.screenshotMaxSize=r,{handled:!0,nextIndex:i+1}}return{handled:!1}}function ec(e){return Object.fromEntries(Object.entries(e).filter(e=>void 0!==e[1]))}let ed={apple:{simulator:!0,device:!0},android:{emulator:!0,device:!0,unknown:!0},linux:{device:!0}};function ep(e){return e}function ef(e){return Object.fromEntries(e.map(e=>[e.name,e.schema]))}function em(e){return Object.fromEntries(e.map(e=>[e.name,e.capability]))}let eh={decode:(e,t)=>({text:e.join(" "),delayMs:t?.delayMs}),encode:e=>[e.text]},eg=ep({name:I.type,schema:{helpDescription:"Type text in focused field",positionalArgs:["text"],allowsExtraPositionals:!0,allowedFlags:["delayMs"]},capability:ed,codec:eh}),ev=[eg],eS=ef(ev),eb=em(ev);function ex(e){return e?{message:e}:{}}function eI(e,t){return t?{...e,message:t}:e}function ew(e){return"string"==typeof e?.message&&e.message.length>0?e.message:null}function eN(e){let t=e.appId??e.bundleId??e.packageName;return{session:e.session,appId:t,appBundleId:e.bundleId,package:e.packageName}}function ey(e,t,n){return{deviceId:t,deviceName:n,..."android"===e?{serial:t}:"ios"===e?{udid:t}:{}}}function eA(e,t={}){let n=t.includeAndroidSerial??!0;return{platform:e.platform,target:e.target,device:e.name,id:e.id,..."ios"===e.platform?{device_udid:e.ios?.udid??e.id,ios_simulator_device_set:e.ios?.simulatorSetPath??null}:{},..."android"===e.platform&&n?{serial:e.android?.serial??e.id}:{}}}function e_(e){return{name:e.name,...eA(e.device,{includeAndroidSerial:!1}),createdAt:e.createdAt}}function eM(e){return{platform:e.platform,id:e.id,name:e.name,kind:e.kind,target:e.target,..."boolean"==typeof e.booted?{booted:e.booted}:{}}}function eC(e){return e.bundleId??e.package??e.app}function eD(e){return eI({app:e.app,appPath:e.appPath,platform:e.platform,...e.appId?{appId:e.appId}:{},...e.bundleId?{bundleId:e.bundleId}:{},...e.package?{package:e.package}:{}},`Installed: ${eC(e)}`)}function ek(e){return e.appName??e.bundleId??e.packageName??e.launchTarget}function eR(e){return eI({launchTarget:e.launchTarget,...e.appName?{appName:e.appName}:{},...e.appId?{appId:e.appId}:{},...e.bundleId?{bundleId:e.bundleId}:{},...e.packageName?{package:e.packageName}:{},...e.installablePath?{installablePath:e.installablePath}:{},...e.archivePath?{archivePath:e.archivePath}:{},...e.materializationId?{materializationId:e.materializationId}:{},...e.materializationExpiresAt?{materializationExpiresAt:e.materializationExpiresAt}:{}},`Installed: ${ek(e)}`)}function ez(e){let t=e.appName??e.appBundleId??e.session;return eI({session:e.session,...e.appName?{appName:e.appName}:{},...e.appBundleId?{appBundleId:e.appBundleId}:{},...e.startup?{startup:e.startup}:{},...e.runtime?{runtime:e.runtime}:{},...e.device?eA(e.device):{}},t?`Opened: ${t}`:"Opened")}function eE(e){return{session:e.session,...e.shutdown?{shutdown:e.shutdown}:{},...ex(e.session?`Closed: ${e.session}`:"Closed")}}function eL(e){return{nodes:e.nodes,truncated:e.truncated,...e.appName?{appName:e.appName}:{},...e.appBundleId?{appBundleId:e.appBundleId}:{},...e.visibility?{visibility:e.visibility}:{},...e.androidSnapshot?{androidSnapshot:e.androidSnapshot}:{},...e.warnings&&e.warnings.length>0?{warnings:e.warnings}:{},...e.unchanged?{unchanged:e.unchanged}:{}}}function eF(e,t,n,i){let r=n(e[t]);if(void 0===r)throw new l("COMMAND_FAILED",i,{response:e});return r}function eP(e,t){return eF(e,t,eK,`Daemon response is missing "${t}".`)}function ej(e,t){return eK(e[t])}function eO(e,t){var n;let i;return n=eK,null===(i=e[t])?null:n(i)}function eT(e,t){return eF(e,t,eU,`Daemon response is missing numeric "${t}".`)}function eq(e,t){return eF(e,t,eW,`Daemon response has invalid "${t}".`)}function e$(e,t){return eF(e,t,eY,`Daemon response has invalid "${t}".`)}function eG(e,t){return function(e){return"tv"===e||"mobile"===e||"desktop"===e?e:void 0}(e[t])??"mobile"}function eV(e,t){let n=e[t];if(!eJ(n))return;let i=eH(n,"x"),r=eH(n,"y"),o=eH(n,"width"),s=eH(n,"height");if(void 0!==i&&void 0!==r&&void 0!==o&&void 0!==s)return{x:i,y:r,width:o,height:s}}function eB(e,t){let n=e[t];if(!eJ(n))return;let i=eH(n,"x"),r=eH(n,"y");if(void 0!==i&&void 0!==r)return{x:i,y:r}}function eH(e,t){let n=e[t];return"number"==typeof n?n:void 0}function eK(e){return"string"==typeof e&&e.length>0?e:void 0}function eU(e){return"number"==typeof e&&Number.isFinite(e)?e:void 0}function eW(e){return"ios"===e||"macos"===e||"android"===e||"linux"===e?e:void 0}function eY(e){return"simulator"===e||"emulator"===e||"device"===e?e:void 0}function eZ(e){if(!eJ(e))throw new l("COMMAND_FAILED","Daemon returned an unexpected response shape.",{value:e});return e}function eJ(e){return"object"==typeof e&&null!==e}function eQ(e){let t={};for(let[n,i]of Object.entries(e))void 0!==i&&(t[n]=i);return t}function eX(e){return e.split("\n").map(e=>e.trim()).filter(e=>e.length>0)}export{ed as ALL_DEVICE_COMMAND_CAPABILITY,N as DAEMON_COMMAND_GROUPS,eb as INTERACTION_COMMAND_CAPABILITIES,eS as INTERACTION_COMMAND_SCHEMAS,w as INTERNAL_COMMANDS,I as PUBLIC_COMMANDS,er as SCREENSHOT_ACTION_FLAG_KEYS,ei as SCREENSHOT_COMMAND_FLAG_KEYS,eo as SCREENSHOT_SPECIFIC_FLAG_DEFINITIONS,el as appendScreenshotScriptFlags,eZ as asRecord,eN as buildAppIdentifiers,ey as buildDeviceIdentifiers,em as commandCapabilityMap,ef as commandSchemaMap,S as computeDaemonCodeSignature,ep as defineCommand,Z as elementTargetCodec,J as fillCommandCodec,ee as findCommandCodec,Y as interactionTargetCodec,et as isCommandCodec,eJ as isRecord,Q as longPressCommandCodec,f as normalizeTenantId,k as parseTimeout,E as parseWaitPositionals,eQ as parsing_stripUndefined,ew as readCommandMessage,eG as readDeviceTarget,P as readLocationCoordinate,eO as readNullableString,ej as readOptionalString,eB as readPoint,eV as readRect,e$ as readRequiredDeviceKind,eT as readRequiredNumber,eq as readRequiredPlatform,eP as readRequiredString,eu as readScreenshotScriptFlag,j as requireLocationCoordinates,v as resolveDaemonCodeSignature,u as resolveDaemonPaths,c as resolveDaemonServerMode,d as resolveDaemonTransportPreference,eC as resolveDeployResultTarget,ek as resolveInstallFromSourceResultTarget,p as resolveSessionIsolationMode,ea as screenshotFlagsFromOptions,es as screenshotOptionsFromFlags,R as selectionOptionsFromFlags,z as selectorSnapshotOptionsFromFlags,eE as serializeCloseResult,eD as serializeDeployResult,eM as serializeDevice,eR as serializeInstallFromSourceResult,ez as serializeOpenResult,e_ as serializeSessionListEntry,eL as serializeSnapshotResult,en as settingsCommandCodec,eX as splitNonEmptyTrimmedLines,ex as successText,eh as typeCommandCodec,eg as typeCommandDefinition,X as waitCommandCodec,eI as withSuccessText};
package/dist/src/4829.js CHANGED
@@ -1 +1 @@
1
- function e(e,t,r){if(!e)return t;let n=Number(e);return Number.isFinite(n)?Math.max(r,Math.floor(n)):t}function t(e){return new Promise(t=>setTimeout(t,e))}function r(t,r,n){return e(t,r,n)}export{e as resolveTimeoutMs,r as resolveTimeoutSeconds,t as sleep};
1
+ function e(e){return new Promise(t=>setTimeout(t,e))}export{e as sleep};
package/dist/src/9542.js CHANGED
@@ -1,2 +1,2 @@
1
- import e from"node:net";import t from"node:http";import a from"node:https";import r from"node:fs";import o from"node:path";import{pipeline as n}from"node:stream/promises";import i from"node:os";import{createHash as s,randomUUID as l}from"node:crypto";import{Writable as d}from"node:stream";import{toAppErrorCode as c,AppError as u}from"./9152.js";import{runCmdSync as p,runCmd as m,runCmdDetached as f}from"./9818.js";import{readVersion as h,findProjectRoot as y}from"./9671.js";import{createRequestId as w,withDiagnosticTimer as I,emitDiagnostic as g}from"./7599.js";import{isAgentDeviceDaemonProcess as v,stopProcessForTakeover as A}from"./8656.js";import{readRect as b,settingsCommandCodec as M,PUBLIC_COMMANDS as _,elementTargetCodec as S,asRecord as E,buildDeviceIdentifiers as P,buildAppIdentifiers as k,computeDaemonCodeSignature as D,resolveDaemonTransportPreference as T,readRequiredNumber as N,readPoint as U,parsing_stripUndefined as C,readOptionalString as R,readRequiredPlatform as O,readRequiredDeviceKind as L,findCommandCodec as x,INTERNAL_COMMANDS as F,fillCommandCodec as B,typeCommandCodec as q,isRecord as z,readRequiredString as $,readNullableString as j,readDeviceTarget as H,isCommandCodec as K,screenshotFlagsFromOptions as V,resolveDaemonPaths as G,resolveDaemonServerMode as J,waitCommandCodec as W,interactionTargetCodec as Y}from"./3572.js";import{sleep as Q}from"./4829.js";import{reloadMetro as X,prepareMetroRuntime as Z}from"./1974.js";function ee(e){return new Promise((t,a)=>{let r="";e.setEncoding("utf8"),e.on("data",e=>{r+=e}),e.on("end",()=>t(r)),e.on("error",a)})}let et="sha256";async function ea(e){let t=await er(e.localPath,e.platform),a=e.baseUrl.endsWith("/")?e.baseUrl:`${e.baseUrl}/`;try{let r=await es({normalizedBase:a,token:e.token,artifact:t});if(r?.kind==="cache-hit")return r.uploadId;if(r?.kind==="direct-upload")try{return await el(t.payloadPath,r),await ec({normalizedBase:a,token:e.token,uploadId:r.uploadId})}catch{}return await ei({normalizedBase:a,token:e.token,artifact:t})}finally{t.cleanup()}}async function er(e,t){var a,n,i;let s,l=r.statSync(e),d=o.basename(e),c=l.isDirectory(),u=("ios"===(a=t)||"android"===a?a:void 0)??(n=e,i=l,s=n.toLowerCase(),i.isDirectory()&&s.endsWith(".app")||s.endsWith(".ipa")?"ios":s.endsWith(".apk")||s.endsWith(".aab")?"android":void 0),p=[];try{let t=c?await eo(e,p):e,a=r.statSync(t);return{payloadPath:t,fileName:d,artifactType:c?"app-bundle":"file",platform:u,contentType:c?"application/gzip":"application/octet-stream",sha256:await eu(t),sizeBytes:a.size,cleanup:()=>en(p)}}catch(e){throw en(p),e}}async function eo(e,t){let a=r.mkdtempSync(o.join(i.tmpdir(),`agent-device-upload-${l()}-`));t.push(a);let n=o.join(a,`${o.basename(e)}.tar.gz`);return await m("tar",["czf",n,"-C",o.dirname(e),o.basename(e)],{env:{...process.env,COPYFILE_DISABLE:"1"}}),n}function en(e){for(let t of e)r.rmSync(t,{recursive:!0,force:!0})}async function ei(e){let{normalizedBase:t,token:a,artifact:r}=e,o=new URL("upload",t),n={"content-type":r.contentType,"x-artifact-type":r.artifactType,"x-artifact-filename":r.fileName,"x-artifact-hash":r.sha256,"x-artifact-hash-algorithm":et,"transfer-encoding":"chunked"};a&&(n.authorization=`Bearer ${a}`,n["x-agent-device-token"]=a);let i=await ed({url:o,method:"POST",headers:n,payloadPath:r.payloadPath,timeoutMessage:"Artifact upload timed out",timeoutHint:"The upload to the remote daemon exceeded the 5-minute timeout.",errorMessage:"Failed to upload artifact to remote daemon",errorHint:"Verify the remote daemon is reachable and supports artifact uploads."});try{let e=JSON.parse(i.body);if(!e.ok||!e.uploadId)throw new u("COMMAND_FAILED",`Upload failed: ${i.body}`);return e.uploadId}catch(e){if(e instanceof u)throw e;throw new u("COMMAND_FAILED",`Invalid upload response: ${i.body}`)}}async function es(e){let t=new URL("upload/preflight",e.normalizedBase),a={"content-type":"application/json"};e.token&&(a.authorization=`Bearer ${e.token}`,a["x-agent-device-token"]=e.token);let r=await fetch(t,{method:"POST",headers:a,signal:AbortSignal.timeout(3e4),body:JSON.stringify({sha256:e.artifact.sha256,fileName:e.artifact.fileName,sizeBytes:e.artifact.sizeBytes,artifactType:e.artifact.artifactType,...e.artifact.platform?{platform:e.artifact.platform}:{},contentType:e.artifact.contentType})}).catch(()=>void 0);if(r?.ok)return function(e){var t;if(!e||"object"!=typeof e||!0!==e.ok||"string"!=typeof e.uploadId)return;if(!0===e.cacheHit)return{kind:"cache-hit",uploadId:e.uploadId};let a=e.upload;if(!a||"string"!=typeof a.url)return;let r=a.headers??{};if(!(!(t=r)||"object"!=typeof t||Array.isArray(t))&&Object.values(t).every(e=>"string"==typeof e))return{kind:"direct-upload",uploadId:e.uploadId,url:a.url,headers:r}}(await r.json().catch(()=>void 0))}async function el(e,t){let a=await ed({url:new URL(t.url),method:"PUT",headers:t.headers,payloadPath:e,timeoutMessage:"Direct artifact upload timed out",timeoutHint:"The direct upload ticket did not accept the artifact within the timeout.",errorMessage:"Failed to upload artifact with direct upload ticket"});if(a.statusCode<200||a.statusCode>=300)throw new u("COMMAND_FAILED","Direct artifact upload failed",{statusCode:a.statusCode,statusMessage:a.statusMessage})}async function ed(e){let o="https:"===e.url.protocol?a:t;return await new Promise((t,a)=>{let i=o.request({protocol:e.url.protocol,host:e.url.hostname,port:e.url.port,method:e.method,path:e.url.pathname+e.url.search,headers:e.headers},e=>{ee(e).then(a=>{clearTimeout(s),t({statusCode:e.statusCode??500,statusMessage:e.statusMessage,body:a})}).catch(a)}),s=setTimeout(()=>{i.destroy(),a(new u("COMMAND_FAILED",e.timeoutMessage,{timeoutMs:3e5,...e.timeoutHint?{hint:e.timeoutHint}:{}}))},3e5);i.on("error",t=>{clearTimeout(s),a(new u("COMMAND_FAILED",e.errorMessage,e.errorHint?{hint:e.errorHint}:{},t))}),i.on("close",()=>clearTimeout(s)),n(r.createReadStream(e.payloadPath),i).catch(e=>{i.destroy(),a(new u("COMMAND_FAILED","Failed to read local artifact",{},e instanceof Error?e:Error(String(e))))})})}async function ec(e){let t=new URL("upload/finalize",e.normalizedBase),a={"content-type":"application/json"};e.token&&(a.authorization=`Bearer ${e.token}`,a["x-agent-device-token"]=e.token);let r=await fetch(t,{method:"POST",headers:a,signal:AbortSignal.timeout(3e4),body:JSON.stringify({uploadId:e.uploadId})}).catch(e=>{throw new u("COMMAND_FAILED","Failed to finalize direct artifact upload",{},e)});if(!r.ok)throw new u("COMMAND_FAILED","Direct artifact upload finalize failed",{status:r.status,statusText:r.statusText});let o=await r.json().catch(()=>void 0);if(!o?.ok||!o.uploadId)throw new u("COMMAND_FAILED","Invalid upload finalize response");return o.uploadId}async function eu(e){let t=s(et),a=new d({write(e,a,r){t.update(e),r()}});return await n(r.createReadStream(e),a).catch(e=>{throw new u("COMMAND_FAILED","Failed to read local artifact",{},e instanceof Error?e:void 0)}),t.digest("hex")}let ep=eV(),em=function(e=process.env.AGENT_DEVICE_DAEMON_STARTUP_TIMEOUT_MS){if(!e)return 15e3;let t=Number(e);return Number.isFinite(t)?Math.max(1e3,Math.floor(t)):15e3}(),ef=function(e=process.env.AGENT_DEVICE_DAEMON_STARTUP_ATTEMPTS){if(!e)return 2;let t=Number(e);return Number.isFinite(t)?Math.min(5,Math.max(1,Math.floor(t))):2}(),eh=["xcodebuild .*AgentDeviceRunnerUITests/RunnerTests/testCommand","xcodebuild .*AgentDeviceRunner\\.env\\.session-","xcodebuild build-for-testing .*ios-runner/AgentDeviceRunner/AgentDeviceRunner\\.xcodeproj"],ey=new e.BlockList;async function ew(t){let a=t.meta?.requestId??w(),r=!!(t.meta?.debug||t.flags?.verbose),o=function(t){let a,r=t.flags?.stateDir??process.env.AGENT_DEVICE_STATE_DIR,o=function(e){let t;if(e){try{t=new URL(e)}catch(t){throw new u("INVALID_ARGS","Invalid daemon base URL",{daemonBaseUrl:e},t instanceof Error?t:void 0)}if("http:"!==t.protocol&&"https:"!==t.protocol)throw new u("INVALID_ARGS","Daemon base URL must use http or https",{daemonBaseUrl:e});return t.toString().replace(/\/+$/,"")}}(t.flags?.daemonBaseUrl??process.env.AGENT_DEVICE_DAEMON_BASE_URL),n=t.flags?.daemonAuthToken??process.env.AGENT_DEVICE_DAEMON_AUTH_TOKEN;var i=o,s=n;if(!(!i||"localhost"===(a=new URL(i).hostname.trim().toLowerCase().replace(/^\[(.*)\]$/,"$1"))||(e.isIPv4(a)?ey.check(a,"ipv4"):!!e.isIPv6(a)&&ey.check(a,"ipv6")))&&("string"!=typeof s||!(s.trim().length>0)))throw new u("INVALID_ARGS","Remote daemon base URL for non-loopback hosts requires daemon authentication",{daemonBaseUrl:i,hint:"Provide --daemon-auth-token or AGENT_DEVICE_DAEMON_AUTH_TOKEN when using a non-loopback remote daemon URL."});let l=t.flags?.daemonTransport??process.env.AGENT_DEVICE_DAEMON_TRANSPORT,d=T(l);if(o&&"socket"===d)throw new u("INVALID_ARGS","Remote daemon base URL only supports HTTP transport. Remove --daemon-transport socket.",{daemonBaseUrl:o});let c=J(t.flags?.daemonServerMode??process.env.AGENT_DEVICE_DAEMON_SERVER_MODE??("dual"===l?"dual":void 0));return{paths:G(r),transportPreference:d,serverMode:c,remoteBaseUrl:o,remoteAuthToken:n}}(t),n=function(e,t=process.env.AGENT_DEVICE_DAEMON_TIMEOUT_MS){if("test"!==e)return eV(t)}(t.command),i=await I("daemon_startup",async()=>await eM(o),{requestId:a,session:t.session}),s=await eI(t,i),l={...t,positionals:s.positionals,flags:s.flags,token:i.token,meta:{...t.meta??{},requestId:a,debug:r,cwd:t.meta?.cwd,tenantId:t.meta?.tenantId??t.flags?.tenant,runId:t.meta?.runId??t.flags?.runId,leaseId:t.meta?.leaseId??t.flags?.leaseId,sessionIsolation:t.meta?.sessionIsolation??t.flags?.sessionIsolation,lockPolicy:t.meta?.lockPolicy,lockPlatform:t.meta?.lockPlatform,...s.uploadedArtifactId?{uploadedArtifactId:s.uploadedArtifactId}:{},...s.clientArtifactPaths?{clientArtifactPaths:s.clientArtifactPaths}:{},...s.installSource?{installSource:s.installSource}:{}}};return g({level:"info",phase:"daemon_request_prepare",data:{requestId:a,command:t.command,session:t.session}}),await I("daemon_request",async()=>await eO(i,l,o.transportPreference,n),{requestId:a,command:t.command})}async function eI(e,t){let a,n=[...e.positionals??[]],i=e.flags?{...e.flags}:void 0,s=e.meta?.installSource,l={};if(e$(t)){let r=function(e,t){if("screenshot"===e.command){let a=eA(e,"path",".png");return t[0]?{field:"path",localPath:a,positionalIndex:0,positionalPath:eb("screenshot",".png")}:{field:"path",localPath:a,positionalIndex:0,flagPath:eb("screenshot",".png")}}if("record"===e.command&&"start"===(t[0]??"").toLowerCase()){let t=eA(e,"outPath",".mp4",1);return{field:"outPath",localPath:t,positionalIndex:1,positionalPath:eb("recording",o.extname(t)||".mp4")}}return null}(e,n);r&&(void 0!==r.positionalPath&&(n[r.positionalIndex]=r.positionalPath),void 0!==r.flagPath&&((i??={}).out=r.flagPath),l[r.field]=r.localPath);let d=await ev(e,t);d&&(s=d.installSource,a=d.uploadedArtifactId??a)}let d=()=>eg({positionals:n,flags:i,installSource:s,uploadedArtifactId:a,clientArtifactPaths:l});if(!e$(t)||"install"!==e.command&&"reinstall"!==e.command||n.length<2)return d();let c=n[1];if(c.startsWith("remote:"))return n[1]=c.slice(7),eg({positionals:n,flags:i,clientArtifactPaths:l});let u=o.isAbsolute(c)?c:o.resolve(e.meta?.cwd??process.cwd(),c);return r.existsSync(u)?(a=await ea({localPath:u,baseUrl:t.baseUrl,token:t.token,platform:e.flags?.platform}),d()):eg({positionals:n,flags:i,clientArtifactPaths:l})}function eg(e){return{positionals:e.positionals,flags:e.flags,installSource:e.installSource,uploadedArtifactId:e.uploadedArtifactId,...Object.keys(e.clientArtifactPaths).length>0?{clientArtifactPaths:e.clientArtifactPaths}:{}}}async function ev(e,t){let a=e.meta?.installSource;if("install_source"!==e.command||!a||"path"!==a.kind)return null;let n=a.path.trim();if(!n)return{installSource:a};if(n.startsWith("remote:"))return{installSource:{...a,path:n.slice(7)}};let i=o.isAbsolute(n)?n:o.resolve(e.meta?.cwd??process.cwd(),n);if(!r.existsSync(i))return{installSource:{...a,path:i}};let s=await ea({localPath:i,baseUrl:t.baseUrl,token:t.token,platform:e.flags?.platform});return{installSource:{...a,path:i},uploadedArtifactId:s}}function eA(e,t,a,r=0){let n=e.positionals?.[r]??e.flags?.out,i=`${"path"===t?"screenshot":"recording"}-${Date.now()}${a}`,s=n&&n.trim().length>0?n:i;return o.isAbsolute(s)?s:o.resolve(e.meta?.cwd??process.cwd(),s)}function eb(e,t){let a=t.startsWith(".")?t:`.${t}`;return o.posix.join("/tmp",`agent-device-${e}-${Date.now()}-${Math.random().toString(36).slice(2,8)}${a}`)}async function eM(e){let t;if(e.remoteBaseUrl){let t={transport:"http",token:e.remoteAuthToken??"",pid:0,baseUrl:e.remoteBaseUrl};if(await eU(t,"http"))return t;throw new u("COMMAND_FAILED","Remote daemon is unavailable",{daemonBaseUrl:e.remoteBaseUrl,hint:"Verify AGENT_DEVICE_DAEMON_BASE_URL points to a reachable daemon with GET /health and POST /rpc."})}let a=eP(e.paths.infoPath),r=h(),o=D((t=eR()).useSrc?t.srcPath:t.distPath,t.root),n=!!a&&await eU(a,e.transportPreference);if(a&&a.version===r&&a.codeSignature===o&&n)return a;a&&(a.version!==r||a.codeSignature!==o||!n)&&(await eE(a),eN(e.paths.infoPath)),function(e){let t=eD(e);if(!t.hasLock||t.hasInfo)return;let a=ek(e.lockPath);if(!a)return eN(e.lockPath);v(a.pid,a.processStartTime)||eN(e.lockPath)}(e.paths);let i=0;for(let t=1;t<=ef;t+=1){await eC(e);let a=await e_(em,e);if(a)return a;if(await eS(e.paths)){i+=1;continue}let r=eD(e.paths);if(!(t<ef))break;if(!r.hasInfo&&!r.hasLock){await Q(150);continue}}let s=eD(e.paths);throw new u("COMMAND_FAILED","Failed to start daemon",{kind:"daemon_startup_failed",infoPath:e.paths.infoPath,lockPath:e.paths.lockPath,startupTimeoutMs:em,startupAttempts:ef,lockRecoveryCount:i,metadataState:s,hint:function(e,t=G(process.env.AGENT_DEVICE_STATE_DIR)){return e.hasLock&&!e.hasInfo?`Detected ${t.lockPath} without ${t.infoPath}. If no agent-device daemon process is running, delete ${t.lockPath} and retry.`:e.hasLock&&e.hasInfo?`Daemon metadata may be stale. If no agent-device daemon process is running, delete ${t.infoPath} and ${t.lockPath}, then retry.`:`Daemon metadata is missing or stale. Delete ${t.infoPath} if present and retry.`}(s,e.paths)})}async function e_(e,t){let a=Date.now();for(;Date.now()-a<e;){let e=eP(t.paths.infoPath);if(e&&await eU(e,t.transportPreference))return e;await Q(100)}return null}async function eS(e){let t=eD(e);if(!t.hasLock||t.hasInfo)return!1;let a=ek(e.lockPath);return a&&v(a.pid,a.processStartTime)&&await A(a.pid,{termTimeoutMs:3e3,killTimeoutMs:1e3,expectedStartTime:a.processStartTime}),eN(e.lockPath),!0}async function eE(e){await A(e.pid,{termTimeoutMs:3e3,killTimeoutMs:1e3,expectedStartTime:e.processStartTime})}function eP(e){let t=eT(e);if(!t||"object"!=typeof t)return null;let a="string"==typeof t.token&&t.token.length>0?t.token:null;if(!a)return null;let r=Number.isInteger(t.port)&&Number(t.port)>0,o=Number.isInteger(t.httpPort)&&Number(t.httpPort)>0;if(!r&&!o)return null;let n=t.transport,i="string"==typeof t.version?t.version:void 0,s="string"==typeof t.codeSignature?t.codeSignature:void 0,l="string"==typeof t.processStartTime?t.processStartTime:void 0,d=Number.isInteger(t.pid)&&Number(t.pid)>0;return{token:a,port:r?Number(t.port):void 0,httpPort:o?Number(t.httpPort):void 0,transport:"socket"===n||"http"===n||"dual"===n?n:void 0,pid:d?Number(t.pid):0,version:i,codeSignature:s,processStartTime:l}}function ek(e){let t=eT(e);return t&&"object"==typeof t&&Number.isInteger(t.pid)&&Number(t.pid)>0?{pid:Number(t.pid),processStartTime:"string"==typeof t.processStartTime?t.processStartTime:void 0,startedAt:"number"==typeof t.startedAt?t.startedAt:void 0}:null}ey.addSubnet("127.0.0.0",8,"ipv4"),ey.addAddress("::1","ipv6"),ey.addSubnet("::ffff:127.0.0.0",104,"ipv6");function eD(e){return{hasInfo:r.existsSync(e.infoPath),hasLock:r.existsSync(e.lockPath)}}function eT(e){if(!r.existsSync(e))return null;try{return JSON.parse(r.readFileSync(e,"utf8"))}catch{return null}}function eN(e){try{r.existsSync(e)&&r.unlinkSync(e)}catch{}}async function eU(r,o){var n;return"http"===eL(r,o)?await function(e){let r=e.baseUrl?ej(e.baseUrl,"health"):e.httpPort?`http://127.0.0.1:${e.httpPort}/health`:null;if(!r)return Promise.resolve(!1);let o=new URL(r),n="https:"===o.protocol?a:t,i=e.baseUrl?3e3:500;return new Promise(e=>{let t=n.request({protocol:o.protocol,host:o.hostname,port:o.port,path:o.pathname+o.search,method:"GET",timeout:i},t=>{t.resume(),e((t.statusCode??500)<500)});t.on("timeout",()=>{t.destroy(),e(!1)}),t.on("error",()=>{e(!1)}),t.end()})}(r):await ((n=r.port)?new Promise(t=>{let a=e.createConnection({host:"127.0.0.1",port:n},()=>{a.destroy(),t(!0)});a.on("error",()=>{t(!1)})}):Promise.resolve(!1))}async function eC(e){let t=eR(),a=t.useSrc?["--experimental-strip-types",t.srcPath]:[t.distPath],r={...process.env,AGENT_DEVICE_STATE_DIR:e.paths.baseDir,AGENT_DEVICE_DAEMON_SERVER_MODE:e.serverMode};f(process.execPath,a,{env:r})}function eR(){let e=y(),t=[o.join(e,"dist","src","internal","daemon.js"),o.join(e,"dist","src","daemon.js")],a=t.find(e=>r.existsSync(e))??t[0],n=o.join(e,"src","daemon.ts"),i=t.some(e=>r.existsSync(e)),s=r.existsSync(n);if(!i&&!s)throw new u("COMMAND_FAILED","Daemon entry not found",{distPaths:t,srcPath:n});return{root:e,distPath:a,distPaths:t,srcPath:n,useSrc:process.execArgv.includes("--experimental-strip-types")?s:!i&&s}}async function eO(e,t,a,r){return"http"===eL(e,a)?await ez(e,t,r):await eq(e,t,r)}function eL(e,t){if(e.baseUrl){if("socket"===t)throw new u("COMMAND_FAILED","Remote daemon endpoint only supports HTTP transport",{daemonBaseUrl:e.baseUrl});return"http"}if("http"===t||"socket"===t){var a=e,r=t;if(ex(a,r))return r;throw new u("COMMAND_FAILED","http"===r?"Daemon HTTP endpoint is unavailable":"Daemon socket endpoint is unavailable")}let o=("socket"===e.transport||"dual"===e.transport?["socket","http"]:["http","socket"]).find(t=>ex(e,t));if(o)return o;throw new u("COMMAND_FAILED","Daemon metadata has no reachable transport")}function ex(e,t){return"http"===t?!!e.httpPort:!!e.port}function eF(e,t,a,r,o,n){let i=o?{terminated:0}:function(){let e=0;try{for(let t of eh){let a=p("pkill",["-f",t],{allowFailure:!0});0===a.exitCode&&(e+=1)}return{terminated:e}}catch(t){return{terminated:e,error:t instanceof Error?t.message:String(t)}}}(),s=o?{forcedKill:!1}:function(e,t){let a=!1;try{v(e.pid,e.processStartTime)&&(process.kill(e.pid,"SIGKILL"),a=!0)}catch{A(e.pid,{termTimeoutMs:3e3,killTimeoutMs:1e3,expectedStartTime:e.processStartTime})}finally{eN(t.infoPath),eN(t.lockPath)}return{forcedKill:a}}(e,t);return g({level:"error",phase:"daemon_request_timeout",data:{timeoutMs:n,requestId:a,command:r,timedOutRunnerPidsTerminated:i.terminated,timedOutRunnerCleanupError:i.error,daemonPidReset:o?void 0:e.pid,daemonPidForceKilled:o?void 0:s.forcedKill,daemonBaseUrl:e.baseUrl}}),new u("COMMAND_FAILED","Daemon request timed out",{timeoutMs:n,requestId:a,hint:o?"Retry with --debug and verify the remote daemon URL, auth token, and remote host logs.":"Retry with --debug and check daemon diagnostics logs. Timed-out iOS runner xcodebuild processes were terminated when detected."})}function eB(e,t,a){return g({level:"error",phase:"daemon_request_socket_error",data:{requestId:t,message:e instanceof Error?e.message:String(e)}}),new u("COMMAND_FAILED","Failed to communicate with daemon",{requestId:t,hint:a?"Retry command. If this persists, verify the remote daemon URL, auth token, and remote host reachability.":"Retry command. If this persists, clean stale daemon metadata and start a fresh session."},e instanceof Error?e:void 0)}async function eq(t,a,r){let o=t.port;if(!o)throw new u("COMMAND_FAILED","Daemon socket endpoint is unavailable");return new Promise((n,i)=>{let s=e.createConnection({host:"127.0.0.1",port:o},()=>{s.write(`${JSON.stringify(a)}
2
- `)}),l=G(a.flags?.stateDir??process.env.AGENT_DEVICE_STATE_DIR),d="number"==typeof r?setTimeout(()=>{s.destroy(),i(eF(t,l,a.meta?.requestId,a.command,!1,r))},r):void 0,c="";s.setEncoding("utf8"),s.on("data",e=>{let t=(c+=e).indexOf("\n");if(-1===t)return;let r=c.slice(0,t).trim();if(r)try{let e=JSON.parse(r);s.end(),d&&clearTimeout(d),n(e)}catch(e){d&&clearTimeout(d),i(new u("COMMAND_FAILED","Invalid daemon response",{requestId:a.meta?.requestId,line:r},e instanceof Error?e:void 0))}}),s.on("error",e=>{d&&clearTimeout(d),i(eB(e,a.meta?.requestId,!1))})})}async function ez(e,r,o){var n,i,s;let l,d=e.baseUrl?new URL(ej(e.baseUrl,"rpc")):e.httpPort?new URL(`http://127.0.0.1:${e.httpPort}/rpc`):null;if(!d)throw new u("COMMAND_FAILED","Daemon HTTP endpoint is unavailable");let p=JSON.stringify((n=r,i={includeTokenParam:!e.baseUrl},l=n.meta?.requestId??w(),"lease_allocate"!==(s=n.command)&&"lease_heartbeat"!==s&&"lease_release"!==s?{jsonrpc:"2.0",id:l,method:"agent_device.command",params:n}:{jsonrpc:"2.0",id:l,method:function(e){switch(e){case"lease_allocate":return"agent_device.lease.allocate";case"lease_heartbeat":return"agent_device.lease.heartbeat";case"lease_release":return"agent_device.lease.release"}}(n.command),params:function(e,t,a){let r={...a.includeTokenParam?{token:e.token}:{},session:e.session,tenantId:e.meta?.tenantId,runId:e.meta?.runId};switch(t){case"lease_allocate":return{...r,ttlMs:e.meta?.leaseTtlMs,backend:e.meta?.leaseBackend};case"lease_heartbeat":return{...r,leaseId:e.meta?.leaseId,ttlMs:e.meta?.leaseTtlMs};case"lease_release":return{...r,leaseId:e.meta?.leaseId}}}(n,n.command,i)})),m={"content-type":"application/json","content-length":Buffer.byteLength(p)};return e.baseUrl&&e.token&&(m.authorization=`Bearer ${e.token}`,m["x-agent-device-token"]=e.token),await new Promise((n,i)=>{let s=G(r.flags?.stateDir??process.env.AGENT_DEVICE_STATE_DIR),l=("https:"===d.protocol?a:t).request({protocol:d.protocol,host:d.hostname,port:d.port,method:"POST",path:d.pathname+d.search,headers:m},t=>{ee(t).then(t=>{h&&clearTimeout(h),function(e,t){let{info:a,req:r,resolve:o,reject:n}=t;try{let t=JSON.parse(e);if(t.error){let e=t.error.data??{};n(new u(c(null!=e.code?String(e.code):void 0,"COMMAND_FAILED"),String(e.message??t.error.message??"Daemon RPC request failed"),{..."object"==typeof e.details&&e.details?e.details:{},hint:"string"==typeof e.hint?e.hint:void 0,diagnosticId:"string"==typeof e.diagnosticId?e.diagnosticId:void 0,logPath:"string"==typeof e.logPath?e.logPath:void 0,requestId:r.meta?.requestId}));return}if(!t.result||"object"!=typeof t.result)return void n(new u("COMMAND_FAILED","Invalid daemon RPC response",{requestId:r.meta?.requestId}));if(a.baseUrl&&t.result.ok)return void eH(a,r,t.result).then(o).catch(n);o(t.result)}catch(t){n(new u("COMMAND_FAILED","Invalid daemon response",{requestId:r.meta?.requestId,line:e},t instanceof Error?t:void 0))}}(t,{info:e,req:r,resolve:n,reject:i})}).catch(e=>{h&&clearTimeout(h),i(new u("COMMAND_FAILED","Failed to read daemon response",{requestId:r.meta?.requestId},e instanceof Error?e:void 0))})}),f=e$(e),h="number"==typeof o?setTimeout(()=>{l.destroy(),i(eF(e,s,r.meta?.requestId,r.command,f,o))},o):void 0;l.on("error",e=>{h&&clearTimeout(h),i(eB(e,r.meta?.requestId,f))}),l.write(p),l.end()})}function e$(e){return"string"==typeof e.baseUrl&&e.baseUrl.length>0}function ej(e,t){return new URL(t,e.endsWith("/")?e:`${e}/`).toString()}async function eH(e,t,a){let r=Array.isArray(a.data?.artifacts)?a.data.artifacts:[];if(0===r.length||!e.baseUrl)return a;let n=a.data?{...a.data}:{},i=[];for(let a of r){if(!a||"object"!=typeof a||"string"!=typeof a.artifactId){i.push(a);continue}let r=function(e,t){if(e.localPath&&e.localPath.trim().length>0)return e.localPath;let a=t.meta?.clientArtifactPaths?.[e.field];if(a&&a.trim().length>0)return a;let r=e.fileName?.trim()||`${e.field}-${Date.now()}`;return o.resolve(t.meta?.cwd??process.cwd(),r)}(a,t);await eK({baseUrl:e.baseUrl,token:e.token,artifactId:a.artifactId,destinationPath:r,requestId:t.meta?.requestId}),n[a.field]=r,i.push({...a,localPath:r})}return n.artifacts=i,{ok:!0,data:n}}async function eK(e){var i,s;let l,d=new URL((i=e.baseUrl,s=e.artifactId,l=i.endsWith("/")?i:`${i}/`,new URL(`artifacts/${encodeURIComponent(s)}`,l).toString())),c="https:"===d.protocol?a:t;await r.promises.mkdir(o.dirname(e.destinationPath),{recursive:!0}),await new Promise((t,a)=>{let o=!1,i=e.timeoutMs??ep,s=n=>{if(!o){if(o=!0,clearTimeout(p),n)return void r.promises.rm(e.destinationPath,{force:!0}).finally(()=>a(n));t()}},l=c.request({protocol:d.protocol,host:d.hostname,port:d.port,method:"GET",path:d.pathname+d.search,headers:e.token?{authorization:`Bearer ${e.token}`,"x-agent-device-token":e.token}:void 0},t=>{if((t.statusCode??500)>=400){let a="";t.setEncoding("utf8"),t.on("data",e=>{a+=e}),t.on("end",()=>{s(new u("COMMAND_FAILED","Failed to download remote artifact",{artifactId:e.artifactId,statusCode:t.statusCode,requestId:e.requestId,body:a}))});return}t.on("aborted",()=>{s(new u("COMMAND_FAILED","Remote artifact download was interrupted",{artifactId:e.artifactId,requestId:e.requestId}))}),n(t,r.createWriteStream(e.destinationPath)).then(()=>s(),e=>s(e instanceof Error?e:Error(String(e))))}),p=setTimeout(()=>{l.destroy(new u("COMMAND_FAILED","Remote artifact download timed out",{artifactId:e.artifactId,requestId:e.requestId,timeoutMs:i}))},i);l.on("error",t=>{t instanceof u?s(t):s(new u("COMMAND_FAILED","Failed to download remote artifact",{artifactId:e.artifactId,requestId:e.requestId,timeoutMs:i},t instanceof Error?t:void 0))}),l.end()})}function eV(e=process.env.AGENT_DEVICE_DAEMON_TIMEOUT_MS){if(!e)return 9e4;let t=Number(e);return Number.isFinite(t)?Math.max(1e3,Math.floor(t)):9e4}function eG(e,t){let a=R(e,"bundleId"),r=R(e,"package");return{app:$(e,"app"),appPath:$(e,"appPath"),platform:O(e,"platform"),appId:a??r,bundleId:a,package:r,identifiers:k({session:t,bundleId:a,packageName:r})}}function eJ(e){let{record:t,platform:a,id:r,name:o,target:n}=eY(e,"name");return{platform:a,target:n,kind:L(t,"kind"),id:r,name:o,booted:"boolean"==typeof t.booted?t.booted:void 0,identifiers:P(a,r,o),...eQ(a,r)}}function eW(e){let{record:t,platform:a,id:r,name:o,target:n}=eY(e,"name"),i=$(t,"device"),s={session:o,...P(a,r,i)};return{name:o,createdAt:N(t,"createdAt"),device:{platform:a,target:n,id:r,name:i,identifiers:s,...eQ(a,r,j(t,"ios_simulator_device_set"))},identifiers:s}}function eY(e,t){let a=E(e);return{record:a,platform:O(a,"platform"),id:$(a,"id"),name:$(a,t),target:H(a,"target")}}function eQ(e,t,a){return{ios:"ios"===e?{udid:t,...void 0!==a?{simulatorSetPath:a}:{}}:void 0,android:"android"===e?{serial:t}:void 0}}function eX(e){return e??"default"}function eZ(e={},t={}){var a;let r,o=t.transport??ew,n=async(t,a=[],r={})=>{var n,i;let s=(n=e,i=r,{...n,...i}),l=await o({session:eX(s.session),command:t,positionals:a,flags:C({stateDir:s.stateDir,daemonBaseUrl:s.daemonBaseUrl,daemonAuthToken:s.daemonAuthToken,daemonTransport:s.daemonTransport,daemonServerMode:s.daemonServerMode,tenant:s.tenant,sessionIsolation:s.sessionIsolation,runId:s.runId,leaseId:s.leaseId,leaseBackend:s.leaseBackend,platform:s.platform,target:s.target,device:s.device,udid:s.udid,serial:s.serial,iosSimulatorDeviceSet:s.iosSimulatorDeviceSet,androidDeviceAllowlist:s.androidDeviceAllowlist,surface:s.surface,activity:s.activity,relaunch:s.relaunch,shutdown:s.shutdown,saveScript:s.saveScript,noRecord:s.noRecord,backMode:s.backMode,metroHost:s.metroHost,metroPort:s.metroPort,bundleUrl:s.bundleUrl,launchUrl:s.launchUrl,snapshotInteractiveOnly:s.interactiveOnly,snapshotCompact:s.compact,snapshotDepth:s.depth,snapshotScope:s.scope,snapshotRaw:s.raw,snapshotForceFull:s.forceFull,...V(s),appsFilter:s.appsFilter,out:s.out,count:s.count,fps:s.fps,quality:s.quality,hideTouches:s.hideTouches,intervalMs:s.intervalMs,delayMs:s.delayMs,holdMs:s.holdMs,jitterPx:s.jitterPx,pixels:s.pixels,doubleTap:s.doubleTap,clickButton:s.clickButton,pauseMs:s.pauseMs,pattern:s.pattern,headless:s.headless,restart:s.restart,replayUpdate:s.replayUpdate,replayMaestro:s.replayMaestro,replayEnv:s.replayEnv,replayShellEnv:s.replayShellEnv,failFast:s.failFast,timeoutMs:s.timeoutMs,retries:s.retries,artifactsDir:s.artifactsDir,reportJunit:s.reportJunit,findFirst:s.findFirst,findLast:s.findLast,networkInclude:s.networkInclude,batchOnError:s.batchOnError,batchMaxSteps:s.batchMaxSteps,batchSteps:s.batchSteps,verbose:s.debug}),runtime:s.runtime,meta:C({requestId:s.requestId,cwd:s.cwd,debug:s.debug,lockPolicy:s.lockPolicy,lockPlatform:s.lockPlatform,tenantId:s.tenant,runId:s.runId,leaseId:s.leaseId,leaseBackend:s.leaseBackend,leaseTtlMs:s.leaseTtlMs,sessionIsolation:s.sessionIsolation,installSource:s.installSource,retainMaterializedPaths:s.retainMaterializedPaths,materializedPathRetentionMs:s.materializedPathRetentionMs,materializationId:s.materializationId})});return l.ok||function(e){throw new u(c(e.code),e.message,{...e.details??{},hint:e.hint,diagnosticId:e.diagnosticId,logPath:e.logPath})}(l.error),l.data??{}},i=async(e={})=>{let t=await n(F.sessionList,[],e);return(Array.isArray(t.sessions)?t.sessions:[]).map(eW)},s=async(e,t=[],a={})=>await n(e,t,a),l=(t={})=>{var a,r;return eX((a=e,r=t,{...a,...r}).session)};return{command:(a=async e=>await n(e.command,e.positionals,e.options),r=async e=>await a(e),{wait:async e=>{var t;return await r((t=e,{command:_.wait,positionals:W.encode(t),options:t}))},alert:async(e={})=>{var t;let a;return await r((a=(t=e).action??"get",{command:_.alert,positionals:[a,...void 0!==t.timeoutMs?[String(t.timeoutMs)]:[]],options:t}))},appState:async(e={})=>await r({command:_.appState,positionals:[],options:e}),back:async(e={})=>await r({command:_.back,positionals:[],options:{...e,backMode:e.mode}}),home:async(e={})=>await r({command:_.home,positionals:[],options:e}),rotate:async e=>await r({command:_.rotate,positionals:[e.orientation],options:e}),appSwitcher:async(e={})=>await r({command:_.appSwitcher,positionals:[],options:e}),keyboard:async(e={})=>await r({command:_.keyboard,positionals:e.action?[e.action]:[],options:e}),clipboard:async e=>{var t;return await r("read"===(t=e).action?{command:_.clipboard,positionals:["read"],options:t}:{command:_.clipboard,positionals:["write",t.text],options:t})}}),devices:{list:async(e={})=>{let t=await n(_.devices,[],e);return(Array.isArray(t.devices)?t.devices:[]).map(eJ)},boot:async(e={})=>await s(_.boot,[],e)},sessions:{list:async(e={})=>await i(e),close:async(e={})=>{let t=l(e),a=(await n(_.close,[],e)).shutdown;return{session:t,shutdown:"object"==typeof a&&null!==a?a:void 0,identifiers:{session:t}}}},apps:{install:async e=>eG(await n(_.install,[e.app,e.appPath],e),l(e)),reinstall:async e=>eG(await n(_.reinstall,[e.app,e.appPath],e),l(e)),installFromSource:async e=>(function(e,t){let a=R(e,"bundleId"),r=R(e,"packageName"),o=a??r??R(e,"appId"),n=R(e,"launchTarget")??r??a??o;if(!n)throw new u("COMMAND_FAILED",'Daemon response is missing "launchTarget".',{response:e});return{appName:R(e,"appName"),appId:o,bundleId:a,packageName:r,launchTarget:n,installablePath:R(e,"installablePath"),archivePath:R(e,"archivePath"),materializationId:R(e,"materializationId"),materializationExpiresAt:R(e,"materializationExpiresAt"),identifiers:k({session:t,bundleId:a,packageName:r,appId:o})}})(await n(F.installSource,[],{...e,installSource:e.source,retainMaterializedPaths:e.retainPaths,materializedPathRetentionMs:e.retentionMs}),l(e)),list:async(e={})=>{let t=await n(_.apps,[],e);return Array.isArray(t.apps)?t.apps.filter(e=>"string"==typeof e):[]},open:async e=>{let t=l(e),a=e.app?e.url?[e.app,e.url]:[e.app]:[],r=await n(_.open,a,e),o=function(e){let t=e.platform,a=R(e,"id"),r=R(e,"device");if("ios"!==t&&"macos"!==t&&"android"!==t&&"linux"!==t||!a||!r)return;let o=H(e,"target"),n=P(t,a,r);return{platform:t,target:o,id:a,name:r,identifiers:n,ios:"ios"===t?{udid:R(e,"device_udid")??a,simulatorSetPath:j(e,"ios_simulator_device_set")}:void 0,android:"android"===t?{serial:R(e,"serial")??a}:void 0}}(r),i=R(r,"appBundleId");return{session:t,appName:R(r,"appName"),appBundleId:i,appId:i,startup:function(e){if(z(e)&&"number"==typeof e.durationMs&&"string"==typeof e.measuredAt&&"string"==typeof e.method)return{durationMs:e.durationMs,measuredAt:e.measuredAt,method:e.method,appTarget:R(e,"appTarget"),appBundleId:R(e,"appBundleId")}}(r.startup),runtime:function(e){if(!z(e))return;let t=e.platform,a=R(e,"metroHost"),r="number"==typeof e.metroPort?e.metroPort:void 0;return{platform:"ios"===t||"android"===t?t:void 0,metroHost:a,metroPort:r,bundleUrl:R(e,"bundleUrl"),launchUrl:R(e,"launchUrl")}}(r.runtime),device:o,identifiers:{session:t,deviceId:o?.id,deviceName:o?.name,udid:o?.ios?.udid,serial:o?.android?.serial,appId:i,appBundleId:i}}},close:async(e={})=>{let t=l(e),a=(await n(_.close,e.app?[e.app]:[],e)).shutdown;return{session:t,closedApp:e.app,shutdown:"object"==typeof a&&null!==a?a:void 0,identifiers:{session:t}}},push:async e=>{var t;return await s(_.push,[e.app,"string"==typeof(t=e.payload)?t:JSON.stringify(t)],e)},triggerEvent:async e=>{var t;return await s(_.triggerAppEvent,[(t=e).event,...t.payload?[JSON.stringify(t.payload)]:[]],e)}},materializations:{release:async e=>{var t;return{released:!0===(t=await n(F.releaseMaterializedPaths,[],{...e,materializationId:e.materializationId})).released,materializationId:$(t,"materializationId"),identifiers:{}}}},leases:{allocate:async e=>e5(await n(F.leaseAllocate,[],{...e,leaseId:void 0,leaseTtlMs:e.ttlMs})),heartbeat:async e=>e5(await n(F.leaseHeartbeat,[],{...e,leaseTtlMs:e.ttlMs})),release:async e=>({released:!0===(await n(F.leaseRelease,[],e)).released})},metro:{prepare:async t=>await Z({projectRoot:t.projectRoot??e.cwd,kind:t.kind,publicBaseUrl:t.publicBaseUrl,proxyBaseUrl:t.proxyBaseUrl,proxyBearerToken:t.bearerToken,bridgeScope:t.bridgeScope,launchUrl:t.launchUrl,companionProfileKey:t.companionProfileKey,companionConsumerKey:t.companionConsumerKey,metroPort:t.port,listenHost:t.listenHost,statusHost:t.statusHost,startupTimeoutMs:t.startupTimeoutMs,probeTimeoutMs:t.probeTimeoutMs,reuseExisting:t.reuseExisting,installDependenciesIfNeeded:t.installDependenciesIfNeeded,runtimeFilePath:t.runtimeFilePath,logPath:t.logPath}),reload:async(t={})=>await X({metroHost:t.metroHost,metroPort:t.metroPort,bundleUrl:t.bundleUrl,runtime:e.runtime,timeoutMs:t.timeoutMs})},capture:{snapshot:async(e={})=>{var t,a,r,o;let i,s,d,c,u,p=l(e);return t=await n(_.snapshot,[],e),a=p,i=R(t,"appBundleId"),{nodes:Array.isArray(r=t.nodes)?r:[],truncated:!0===t.truncated,appName:R(t,"appName"),appBundleId:i,...(s=e0((o=t).visibility),d=e0(o.androidSnapshot),c=e0(o.unchanged),u=Array.isArray(o.warnings)?o.warnings.filter(e=>"string"==typeof e):void 0,{...s?{visibility:s}:{},...d?{androidSnapshot:d}:{},...c?{unchanged:c}:{},...u?{warnings:u}:{}}),identifiers:{session:a,appId:i,appBundleId:i}}},screenshot:async(e={})=>{let t=l(e),a=await n(_.screenshot,e.path?[e.path]:[],{...e,...V(e)});return{path:$(a,"path"),overlayRefs:function(e){let t=e.overlayRefs;if(!Array.isArray(t))return;let a=[];for(let e of t){if(!z(e))continue;let t=R(e,"ref"),r=b(e,"rect"),o=b(e,"overlayRect"),n=U(e,"center");t&&r&&o&&n&&a.push({ref:t,label:R(e,"label"),rect:r,overlayRect:o,center:n})}return a}(a),identifiers:{session:t}}},diff:async e=>await s(_.diff,[e.kind],{...e,interactiveOnly:e.interactiveOnly,compact:e.compact,depth:e.depth,scope:e.scope,raw:e.raw})},interactions:{click:async e=>await s(_.click,Y.encode(e),{...e,clickButton:e.button}),press:async e=>await s(_.press,Y.encode(e),e),longPress:async e=>await s(_.longPress,[String(e.x),String(e.y),...e3(e.durationMs)],e),swipe:async e=>await s(_.swipe,[String(e.from.x),String(e.from.y),String(e.to.x),String(e.to.y),...e3(e.durationMs)],e),focus:async e=>await s(_.focus,[String(e.x),String(e.y)],e),type:async e=>await s(_.type,q.encode(e),e),fill:async e=>await s(_.fill,B.encode(e),e),scroll:async e=>await s(_.scroll,[e.direction,...e3(e.amount)],e),pinch:async e=>await s(_.pinch,[String(e.scale),...e3(e.x),...e3(e.y)],e),get:async e=>await s(_.get,[e.format,...S.encode(e)],e),is:async e=>await s(_.is,K.encode(e),e),find:async e=>await s(_.find,x.encode(e),{...e,findFirst:e.first,findLast:e.last})},replay:{run:async e=>await s(_.replay,[e.path],{...e,replayUpdate:e.update,replayMaestro:e.maestro,replayEnv:e.env,replayShellEnv:e2(process.env)}),test:async e=>await s(_.test,e.paths,{...e,replayUpdate:e.update,replayEnv:e.env,replayShellEnv:e2(process.env)})},batch:{run:async e=>await s(_.batch,[],{...e,batchSteps:e.steps,batchOnError:e.onError,batchMaxSteps:e.maxSteps})},observability:{perf:async(e={})=>await s(_.perf,[],e),logs:async(e={})=>{var t;return await s(_.logs,[(t=e).action??"path",...e1(t.message)],e)},network:async(e={})=>{var t;return await s(_.network,[...(t=e).action?[t.action]:[],...e3(t.limit)],{...e,networkInclude:e.include})}},recording:{record:async e=>await s(_.record,[e.action,...e1(e.path)],e),trace:async e=>await s(_.trace,[e.action,...e1(e.path)],e)},settings:{update:async e=>await s(_.settings,M.encode(e),e)}}}function e0(e){return"object"==typeof e&&null!==e?e:void 0}function e1(e){return void 0===e?[]:[e]}function e3(e){return void 0===e?[]:[String(e)]}function e2(e){let t={};for(let[a,r]of Object.entries(e))"string"==typeof r&&a.startsWith("AD_VAR_")&&(t[a]=r);return t}function e5(e){let t=e.lease;if(!t||"object"!=typeof t||Array.isArray(t))throw Error("Invalid lease response from daemon");return{leaseId:$(t,"leaseId"),tenantId:$(t,"tenantId"),runId:$(t,"runId"),backend:$(t,"backend"),createdAt:"number"==typeof t.createdAt?t.createdAt:void 0,heartbeatAt:"number"==typeof t.heartbeatAt?t.heartbeatAt:void 0,expiresAt:"number"==typeof t.expiresAt?t.expiresAt:void 0}}export{eZ as createAgentDeviceClient,ew as sendToDaemon};
1
+ import e from"node:net";import t from"node:http";import a from"node:https";import r from"node:fs";import o from"node:path";import{pipeline as n}from"node:stream/promises";import i from"node:os";import{createHash as s,randomUUID as l}from"node:crypto";import{Writable as c}from"node:stream";import{toAppErrorCode as d,AppError as p}from"./9152.js";import{runCmdSync as u,runCmd as m,runCmdDetached as f}from"./9818.js";import{readVersion as h,findProjectRoot as y}from"./9671.js";import{createRequestId as w,withDiagnosticTimer as I,emitDiagnostic as v}from"./7599.js";import{isAgentDeviceDaemonProcess as g,stopProcessForTakeover as b}from"./8656.js";import{readRect as A,settingsCommandCodec as k,PUBLIC_COMMANDS as P,elementTargetCodec as S,asRecord as _,longPressCommandCodec as M,buildAppIdentifiers as E,buildDeviceIdentifiers as D,computeDaemonCodeSignature as T,resolveDaemonTransportPreference as N,readRequiredNumber as U,readPoint as C,parsing_stripUndefined as L,readOptionalString as R,readRequiredPlatform as O,readRequiredDeviceKind as x,findCommandCodec as F,INTERNAL_COMMANDS as B,fillCommandCodec as q,typeCommandCodec as z,isRecord as j,readRequiredString as $,readNullableString as H,readDeviceTarget as K,isCommandCodec as V,screenshotFlagsFromOptions as G,resolveDaemonPaths as J,resolveDaemonServerMode as W,waitCommandCodec as Y,interactionTargetCodec as Q}from"./3572.js";import{sleep as X}from"./4829.js";import{reloadMetro as Z,prepareMetroRuntime as ee}from"./1974.js";function et(e){return new Promise((t,a)=>{let r="";e.setEncoding("utf8"),e.on("data",e=>{r+=e}),e.on("end",()=>t(r)),e.on("error",a)})}let ea="sha256";async function er(e){let t=await eo(e.localPath,e.platform),a=e.baseUrl.endsWith("/")?e.baseUrl:`${e.baseUrl}/`;try{let r=await el({normalizedBase:a,token:e.token,artifact:t});if(r?.kind==="cache-hit")return r.uploadId;if(r?.kind==="direct-upload")try{return await ec(t.payloadPath,r),await ep({normalizedBase:a,token:e.token,uploadId:r.uploadId})}catch{}return await es({normalizedBase:a,token:e.token,artifact:t})}finally{t.cleanup()}}async function eo(e,t){var a,n,i;let s,l=r.statSync(e),c=o.basename(e),d=l.isDirectory(),p=("ios"===(a=t)||"android"===a?a:void 0)??(n=e,i=l,s=n.toLowerCase(),i.isDirectory()&&s.endsWith(".app")||s.endsWith(".ipa")?"ios":s.endsWith(".apk")||s.endsWith(".aab")?"android":void 0),u=[];try{let t=d?await en(e,u):e,a=r.statSync(t);return{payloadPath:t,fileName:c,artifactType:d?"app-bundle":"file",platform:p,contentType:d?"application/gzip":"application/octet-stream",sha256:await eu(t),sizeBytes:a.size,cleanup:()=>ei(u)}}catch(e){throw ei(u),e}}async function en(e,t){let a=r.mkdtempSync(o.join(i.tmpdir(),`agent-device-upload-${l()}-`));t.push(a);let n=o.join(a,`${o.basename(e)}.tar.gz`);return await m("tar",["czf",n,"-C",o.dirname(e),o.basename(e)],{env:{...process.env,COPYFILE_DISABLE:"1"}}),n}function ei(e){for(let t of e)r.rmSync(t,{recursive:!0,force:!0})}async function es(e){let{normalizedBase:t,token:a,artifact:r}=e,o=new URL("upload",t),n={"content-type":r.contentType,"x-artifact-type":r.artifactType,"x-artifact-filename":r.fileName,"x-artifact-hash":r.sha256,"x-artifact-hash-algorithm":ea,"transfer-encoding":"chunked"};a&&(n.authorization=`Bearer ${a}`,n["x-agent-device-token"]=a);let i=await ed({url:o,method:"POST",headers:n,payloadPath:r.payloadPath,timeoutMessage:"Artifact upload timed out",timeoutHint:"The upload to the remote daemon exceeded the 5-minute timeout.",errorMessage:"Failed to upload artifact to remote daemon",errorHint:"Verify the remote daemon is reachable and supports artifact uploads."});try{let e=JSON.parse(i.body);if(!e.ok||!e.uploadId)throw new p("COMMAND_FAILED",`Upload failed: ${i.body}`);return e.uploadId}catch(e){if(e instanceof p)throw e;throw new p("COMMAND_FAILED",`Invalid upload response: ${i.body}`)}}async function el(e){let t=new URL("upload/preflight",e.normalizedBase),a={"content-type":"application/json"};e.token&&(a.authorization=`Bearer ${e.token}`,a["x-agent-device-token"]=e.token);let r=await fetch(t,{method:"POST",headers:a,signal:AbortSignal.timeout(3e4),body:JSON.stringify({sha256:e.artifact.sha256,fileName:e.artifact.fileName,sizeBytes:e.artifact.sizeBytes,artifactType:e.artifact.artifactType,...e.artifact.platform?{platform:e.artifact.platform}:{},contentType:e.artifact.contentType})}).catch(()=>void 0);if(r?.ok)return function(e){var t;if(!e||"object"!=typeof e||!0!==e.ok||"string"!=typeof e.uploadId)return;if(!0===e.cacheHit)return{kind:"cache-hit",uploadId:e.uploadId};let a=e.upload;if(!a||"string"!=typeof a.url)return;let r=a.headers??{};if(!(!(t=r)||"object"!=typeof t||Array.isArray(t))&&Object.values(t).every(e=>"string"==typeof e))return{kind:"direct-upload",uploadId:e.uploadId,url:a.url,headers:r}}(await r.json().catch(()=>void 0))}async function ec(e,t){let a=await ed({url:new URL(t.url),method:"PUT",headers:t.headers,payloadPath:e,timeoutMessage:"Direct artifact upload timed out",timeoutHint:"The direct upload ticket did not accept the artifact within the timeout.",errorMessage:"Failed to upload artifact with direct upload ticket"});if(a.statusCode<200||a.statusCode>=300)throw new p("COMMAND_FAILED","Direct artifact upload failed",{statusCode:a.statusCode,statusMessage:a.statusMessage})}async function ed(e){let o="https:"===e.url.protocol?a:t;return await new Promise((t,a)=>{let i=o.request({protocol:e.url.protocol,host:e.url.hostname,port:e.url.port,method:e.method,path:e.url.pathname+e.url.search,headers:e.headers},e=>{et(e).then(a=>{clearTimeout(s),t({statusCode:e.statusCode??500,statusMessage:e.statusMessage,body:a})}).catch(a)}),s=setTimeout(()=>{i.destroy(),a(new p("COMMAND_FAILED",e.timeoutMessage,{timeoutMs:3e5,...e.timeoutHint?{hint:e.timeoutHint}:{}}))},3e5);i.on("error",t=>{clearTimeout(s),a(new p("COMMAND_FAILED",e.errorMessage,e.errorHint?{hint:e.errorHint}:{},t))}),i.on("close",()=>clearTimeout(s)),n(r.createReadStream(e.payloadPath),i).catch(e=>{i.destroy(),a(new p("COMMAND_FAILED","Failed to read local artifact",{},e instanceof Error?e:Error(String(e))))})})}async function ep(e){let t=new URL("upload/finalize",e.normalizedBase),a={"content-type":"application/json"};e.token&&(a.authorization=`Bearer ${e.token}`,a["x-agent-device-token"]=e.token);let r=await fetch(t,{method:"POST",headers:a,signal:AbortSignal.timeout(3e4),body:JSON.stringify({uploadId:e.uploadId})}).catch(e=>{throw new p("COMMAND_FAILED","Failed to finalize direct artifact upload",{},e)});if(!r.ok)throw new p("COMMAND_FAILED","Direct artifact upload finalize failed",{status:r.status,statusText:r.statusText});let o=await r.json().catch(()=>void 0);if(!o?.ok||!o.uploadId)throw new p("COMMAND_FAILED","Invalid upload finalize response");return o.uploadId}async function eu(e){let t=s(ea),a=new c({write(e,a,r){t.update(e),r()}});return await n(r.createReadStream(e),a).catch(e=>{throw new p("COMMAND_FAILED","Failed to read local artifact",{},e instanceof Error?e:void 0)}),t.digest("hex")}let em=["xcodebuild .*AgentDeviceRunnerUITests/RunnerTests/testCommand","xcodebuild .*AgentDeviceRunner\\.env\\.session-","xcodebuild build-for-testing .*ios-runner/AgentDeviceRunner/AgentDeviceRunner\\.xcodeproj"],ef=new e.BlockList;async function eh(t){let a=t.meta?.requestId??w(),r=!!(t.meta?.debug||t.flags?.verbose),o=function(t){let a,r=t.flags?.stateDir??process.env.AGENT_DEVICE_STATE_DIR,o=function(e){let t;if(e){try{t=new URL(e)}catch(t){throw new p("INVALID_ARGS","Invalid daemon base URL",{daemonBaseUrl:e},t instanceof Error?t:void 0)}if("http:"!==t.protocol&&"https:"!==t.protocol)throw new p("INVALID_ARGS","Daemon base URL must use http or https",{daemonBaseUrl:e});return t.toString().replace(/\/+$/,"")}}(t.flags?.daemonBaseUrl??process.env.AGENT_DEVICE_DAEMON_BASE_URL),n=t.flags?.daemonAuthToken??process.env.AGENT_DEVICE_DAEMON_AUTH_TOKEN;var i=o,s=n;if(!(!i||"localhost"===(a=new URL(i).hostname.trim().toLowerCase().replace(/^\[(.*)\]$/,"$1"))||(e.isIPv4(a)?ef.check(a,"ipv4"):!!e.isIPv6(a)&&ef.check(a,"ipv6")))&&("string"!=typeof s||!(s.trim().length>0)))throw new p("INVALID_ARGS","Remote daemon base URL for non-loopback hosts requires daemon authentication",{daemonBaseUrl:i,hint:"Provide --daemon-auth-token or AGENT_DEVICE_DAEMON_AUTH_TOKEN when using a non-loopback remote daemon URL."});let l=t.flags?.daemonTransport??process.env.AGENT_DEVICE_DAEMON_TRANSPORT,c=N(l);if(o&&"socket"===c)throw new p("INVALID_ARGS","Remote daemon base URL only supports HTTP transport. Remove --daemon-transport socket.",{daemonBaseUrl:o});let d=W(t.flags?.daemonServerMode??process.env.AGENT_DEVICE_DAEMON_SERVER_MODE??("dual"===l?"dual":void 0));return{paths:J(r),transportPreference:c,serverMode:d,remoteBaseUrl:o,remoteAuthToken:n}}(t),n="test"===t.command?void 0:9e4,i=await I("daemon_startup",async()=>await eb(o),{requestId:a,session:t.session}),s=await ey(t,i),l={...t,positionals:s.positionals,flags:s.flags,token:i.token,meta:{...t.meta??{},requestId:a,debug:r,cwd:t.meta?.cwd,tenantId:t.meta?.tenantId??t.flags?.tenant,runId:t.meta?.runId??t.flags?.runId,leaseId:t.meta?.leaseId??t.flags?.leaseId,sessionIsolation:t.meta?.sessionIsolation??t.flags?.sessionIsolation,lockPolicy:t.meta?.lockPolicy,lockPlatform:t.meta?.lockPlatform,...s.uploadedArtifactId?{uploadedArtifactId:s.uploadedArtifactId}:{},...s.clientArtifactPaths?{clientArtifactPaths:s.clientArtifactPaths}:{},...s.installSource?{installSource:s.installSource}:{}}};return v({level:"info",phase:"daemon_request_prepare",data:{requestId:a,command:t.command,session:t.session}}),await I("daemon_request",async()=>await eL(i,l,o.transportPreference,n),{requestId:a,command:t.command})}async function ey(e,t){let a,n=[...e.positionals??[]],i=e.flags?{...e.flags}:void 0,s=e.meta?.installSource,l={};if(ez(t)){let r=function(e,t){if("screenshot"===e.command){let a=ev(e,"path",".png");return t[0]?{field:"path",localPath:a,positionalIndex:0,positionalPath:eg("screenshot",".png")}:{field:"path",localPath:a,positionalIndex:0,flagPath:eg("screenshot",".png")}}if("record"===e.command&&"start"===(t[0]??"").toLowerCase()){let t=ev(e,"outPath",".mp4",1);return{field:"outPath",localPath:t,positionalIndex:1,positionalPath:eg("recording",o.extname(t)||".mp4")}}return null}(e,n);r&&(void 0!==r.positionalPath&&(n[r.positionalIndex]=r.positionalPath),void 0!==r.flagPath&&((i??={}).out=r.flagPath),l[r.field]=r.localPath);let c=await eI(e,t);c&&(s=c.installSource,a=c.uploadedArtifactId??a)}let c=()=>ew({positionals:n,flags:i,installSource:s,uploadedArtifactId:a,clientArtifactPaths:l});if(!ez(t)||"install"!==e.command&&"reinstall"!==e.command||n.length<2)return c();let d=n[1];if(d.startsWith("remote:"))return n[1]=d.slice(7),ew({positionals:n,flags:i,clientArtifactPaths:l});let p=o.isAbsolute(d)?d:o.resolve(e.meta?.cwd??process.cwd(),d);return r.existsSync(p)?(a=await er({localPath:p,baseUrl:t.baseUrl,token:t.token,platform:e.flags?.platform}),c()):ew({positionals:n,flags:i,clientArtifactPaths:l})}function ew(e){return{positionals:e.positionals,flags:e.flags,installSource:e.installSource,uploadedArtifactId:e.uploadedArtifactId,...Object.keys(e.clientArtifactPaths).length>0?{clientArtifactPaths:e.clientArtifactPaths}:{}}}async function eI(e,t){let a=e.meta?.installSource;if("install_source"!==e.command||!a||"path"!==a.kind)return null;let n=a.path.trim();if(!n)return{installSource:a};if(n.startsWith("remote:"))return{installSource:{...a,path:n.slice(7)}};let i=o.isAbsolute(n)?n:o.resolve(e.meta?.cwd??process.cwd(),n);if(!r.existsSync(i))return{installSource:{...a,path:i}};let s=await er({localPath:i,baseUrl:t.baseUrl,token:t.token,platform:e.flags?.platform});return{installSource:{...a,path:i},uploadedArtifactId:s}}function ev(e,t,a,r=0){let n=e.positionals?.[r]??e.flags?.out,i=`${"path"===t?"screenshot":"recording"}-${Date.now()}${a}`,s=n&&n.trim().length>0?n:i;return o.isAbsolute(s)?s:o.resolve(e.meta?.cwd??process.cwd(),s)}function eg(e,t){let a=t.startsWith(".")?t:`.${t}`;return o.posix.join("/tmp",`agent-device-${e}-${Date.now()}-${Math.random().toString(36).slice(2,8)}${a}`)}async function eb(e){let t,a;if(e.remoteBaseUrl){let t={transport:"http",token:e.remoteAuthToken??"",pid:0,baseUrl:e.remoteBaseUrl};if(await eN(t,"http"))return t;throw new p("COMMAND_FAILED","Remote daemon is unavailable",{daemonBaseUrl:e.remoteBaseUrl,hint:"Verify AGENT_DEVICE_DAEMON_BASE_URL points to a reachable daemon with GET /health and POST /rpc."})}let r=eS(e.paths.infoPath),o=h(),n=T((a=eC()).useSrc?a.srcPath:a.distPath,a.root),i=!!r&&await eN(r,e.transportPreference);if(r&&r.version===o&&r.codeSignature===n&&i)return r;r&&(r.version!==o||r.codeSignature!==n||!i)&&(await eP(r),eT(e.paths.infoPath)),function(e){let t=eE(e);if(!t.hasLock||t.hasInfo)return;let a=e_(e.lockPath);if(!a)return eT(e.lockPath);g(a.pid,a.processStartTime)||eT(e.lockPath)}(e.paths);let s=0,l=[];for(let a=1;a<=2;a+=1){try{await eU(e)}catch(r){if(t=r instanceof Error?r.message:String(r),l.push(await eM(e.paths,"start_error")),a<2){await X(150);continue}break}let r=await eA(15e3,e);if(r)return r;if(await ek(e.paths)){s+=1;continue}let o=eE(e.paths),n=a<2,i=await eM(e.paths,"startup_timeout",{stopLiveProcesses:!1});if(l.push(i),i.retainedInfoProcess||i.retainedLockProcess){let t=await eA(15e3,e);if(t)return t;break}if(!n)break;o.hasInfo||o.hasLock||await X(150)}let c=eE(e.paths);throw new p("COMMAND_FAILED","Failed to start daemon",{kind:"daemon_startup_failed",infoPath:e.paths.infoPath,lockPath:e.paths.lockPath,startupTimeoutMs:15e3,startupAttempts:2,lockRecoveryCount:s,cleanupResults:l,startError:t,metadataState:c,hint:function(e,t=J(process.env.AGENT_DEVICE_STATE_DIR)){return e.hasLock&&!e.hasInfo?`agent-device attempted to clean stale daemon metadata automatically, but ${t.lockPath} still exists without ${t.infoPath}. Retry with --debug; if this persists, remove ${t.lockPath} after confirming no agent-device daemon process is running.`:e.hasLock&&e.hasInfo?`agent-device attempted to clean stale daemon metadata automatically, but ${t.infoPath} and ${t.lockPath} still remain. Retry with --debug; if this persists, remove both files after confirming no agent-device daemon process is running.`:"agent-device did not observe reachable daemon metadata after retrying. Stale metadata was cleaned automatically when safe; retry with --debug and check daemon diagnostics logs."}(c,e.paths)})}async function eA(e,t){let a=Date.now();for(;Date.now()-a<e;){let e=eS(t.paths.infoPath);if(e&&await eN(e,t.transportPreference))return e;await X(100)}return null}async function ek(e){let t=eE(e);if(!t.hasLock||t.hasInfo)return!1;let a=e_(e.lockPath);return!(a&&g(a.pid,a.processStartTime))&&(eT(e.lockPath),!0)}async function eP(e){await b(e.pid,{termTimeoutMs:3e3,killTimeoutMs:1e3,expectedStartTime:e.processStartTime})}function eS(e){let t=eD(e);if(!t||"object"!=typeof t)return null;let a="string"==typeof t.token&&t.token.length>0?t.token:null;if(!a)return null;let r=Number.isInteger(t.port)&&Number(t.port)>0,o=Number.isInteger(t.httpPort)&&Number(t.httpPort)>0;if(!r&&!o)return null;let n=t.transport,i="string"==typeof t.version?t.version:void 0,s="string"==typeof t.codeSignature?t.codeSignature:void 0,l="string"==typeof t.processStartTime?t.processStartTime:void 0,c=Number.isInteger(t.pid)&&Number(t.pid)>0;return{token:a,port:r?Number(t.port):void 0,httpPort:o?Number(t.httpPort):void 0,transport:"socket"===n||"http"===n||"dual"===n?n:void 0,pid:c?Number(t.pid):0,version:i,codeSignature:s,processStartTime:l}}function e_(e){let t=eD(e);return t&&"object"==typeof t&&Number.isInteger(t.pid)&&Number(t.pid)>0?{pid:Number(t.pid),processStartTime:"string"==typeof t.processStartTime?t.processStartTime:void 0,startedAt:"number"==typeof t.startedAt?t.startedAt:void 0}:null}ef.addSubnet("127.0.0.0",8,"ipv4"),ef.addAddress("::1","ipv6"),ef.addSubnet("::ffff:127.0.0.0",104,"ipv6");async function eM(e,t,a={}){let o=a.stopLiveProcesses??!0,n={reason:t,removedInfo:!1,removedLock:!1,stoppedInfoProcess:!1,stoppedLockProcess:!1};try{var i,s,l,c;let t=r.existsSync(e.infoPath),a=eS(e.infoPath);if(a){let t=g(a.pid,a.processStartTime);t&&!o?n.retainedInfoProcess=!0:(t&&(await eP(a),n.stoppedInfoProcess=!0),i=e.infoPath,eT(i),n.removedInfo=!0)}else t&&(s=e.infoPath,eT(s),n.removedInfo=!0);let d=r.existsSync(e.lockPath),p=e_(e.lockPath);if(p){let t=g(p.pid,p.processStartTime);t&&!o?n.retainedLockProcess=!0:(t&&(await b(p.pid,{termTimeoutMs:3e3,killTimeoutMs:1e3,expectedStartTime:p.processStartTime}),n.stoppedLockProcess=!0),l=e.lockPath,eT(l),n.removedLock=!0)}else d&&(c=e.lockPath,eT(c),n.removedLock=!0)}catch(e){n.error=e instanceof Error?e.message:String(e)}return v({level:n.error?"warn":"info",phase:"daemon_startup_metadata_cleanup",data:n}),n}function eE(e){return{hasInfo:r.existsSync(e.infoPath),hasLock:r.existsSync(e.lockPath)}}function eD(e){if(!r.existsSync(e))return null;try{return JSON.parse(r.readFileSync(e,"utf8"))}catch{return null}}function eT(e){try{r.existsSync(e)&&r.unlinkSync(e)}catch{}}async function eN(r,o){var n;return"http"===eR(r,o)?await function(e){let r=e.baseUrl?ej(e.baseUrl,"health"):e.httpPort?`http://127.0.0.1:${e.httpPort}/health`:null;if(!r)return Promise.resolve(!1);let o=new URL(r),n="https:"===o.protocol?a:t,i=e.baseUrl?3e3:500;return new Promise(e=>{let t=n.request({protocol:o.protocol,host:o.hostname,port:o.port,path:o.pathname+o.search,method:"GET",timeout:i},t=>{t.resume(),e((t.statusCode??500)<500)});t.on("timeout",()=>{t.destroy(),e(!1)}),t.on("error",()=>{e(!1)}),t.end()})}(r):await ((n=r.port)?new Promise(t=>{let a=e.createConnection({host:"127.0.0.1",port:n},()=>{a.destroy(),t(!0)});a.on("error",()=>{t(!1)})}):Promise.resolve(!1))}async function eU(e){let t=eC(),a=t.useSrc?["--experimental-strip-types",t.srcPath]:[t.distPath],r={...process.env,AGENT_DEVICE_STATE_DIR:e.paths.baseDir,AGENT_DEVICE_DAEMON_SERVER_MODE:e.serverMode};f(process.execPath,a,{env:r})}function eC(){let e=y(),t=[o.join(e,"dist","src","internal","daemon.js"),o.join(e,"dist","src","daemon.js")],a=t.find(e=>r.existsSync(e))??t[0],n=o.join(e,"src","daemon.ts"),i=t.some(e=>r.existsSync(e)),s=r.existsSync(n);if(!i&&!s)throw new p("COMMAND_FAILED","Daemon entry not found",{distPaths:t,srcPath:n});return{root:e,distPath:a,distPaths:t,srcPath:n,useSrc:process.execArgv.includes("--experimental-strip-types")?s:!i&&s}}async function eL(e,t,a,r){return"http"===eR(e,a)?await eq(e,t,r):await eB(e,t,r)}function eR(e,t){if(e.baseUrl){if("socket"===t)throw new p("COMMAND_FAILED","Remote daemon endpoint only supports HTTP transport",{daemonBaseUrl:e.baseUrl});return"http"}if("http"===t||"socket"===t){var a=e,r=t;if(eO(a,r))return r;throw new p("COMMAND_FAILED","http"===r?"Daemon HTTP endpoint is unavailable":"Daemon socket endpoint is unavailable")}let o=("socket"===e.transport||"dual"===e.transport?["socket","http"]:["http","socket"]).find(t=>eO(e,t));if(o)return o;throw new p("COMMAND_FAILED","Daemon metadata has no reachable transport")}function eO(e,t){return"http"===t?!!e.httpPort:!!e.port}function ex(e,t,a,r,o,n){let i=o?{terminated:0}:function(){let e=0;try{for(let t of em){let a=u("pkill",["-f",t],{allowFailure:!0});0===a.exitCode&&(e+=1)}return{terminated:e}}catch(t){return{terminated:e,error:t instanceof Error?t.message:String(t)}}}(),s=o?{forcedKill:!1}:function(e,t){let a=!1;try{g(e.pid,e.processStartTime)&&(process.kill(e.pid,"SIGKILL"),a=!0)}catch{b(e.pid,{termTimeoutMs:3e3,killTimeoutMs:1e3,expectedStartTime:e.processStartTime})}finally{eT(t.infoPath),eT(t.lockPath)}return{forcedKill:a}}(e,t);return v({level:"error",phase:"daemon_request_timeout",data:{timeoutMs:n,requestId:a,command:r,timedOutRunnerPidsTerminated:i.terminated,timedOutRunnerCleanupError:i.error,daemonPidReset:o?void 0:e.pid,daemonPidForceKilled:o?void 0:s.forcedKill,daemonBaseUrl:e.baseUrl}}),new p("COMMAND_FAILED","Daemon request timed out",{timeoutMs:n,requestId:a,hint:o?"Retry with --debug and verify the remote daemon URL, auth token, and remote host logs.":"Retry with --debug and check daemon diagnostics logs. Timed-out iOS runner xcodebuild processes were terminated when detected."})}function eF(e,t,a){return v({level:"error",phase:"daemon_request_socket_error",data:{requestId:t,message:e instanceof Error?e.message:String(e)}}),new p("COMMAND_FAILED","Failed to communicate with daemon",{requestId:t,hint:a?"Retry command. If this persists, verify the remote daemon URL, auth token, and remote host reachability.":"Retry command. If this persists, clean stale daemon metadata and start a fresh session."},e instanceof Error?e:void 0)}async function eB(t,a,r){let o=t.port;if(!o)throw new p("COMMAND_FAILED","Daemon socket endpoint is unavailable");return new Promise((n,i)=>{let s=e.createConnection({host:"127.0.0.1",port:o},()=>{s.write(`${JSON.stringify(a)}
2
+ `)}),l=J(a.flags?.stateDir??process.env.AGENT_DEVICE_STATE_DIR),c="number"==typeof r?setTimeout(()=>{s.destroy(),i(ex(t,l,a.meta?.requestId,a.command,!1,r))},r):void 0,d="";s.setEncoding("utf8"),s.on("data",e=>{let t=(d+=e).indexOf("\n");if(-1===t)return;let r=d.slice(0,t).trim();if(r)try{let e=JSON.parse(r);s.end(),c&&clearTimeout(c),n(e)}catch(e){c&&clearTimeout(c),i(new p("COMMAND_FAILED","Invalid daemon response",{requestId:a.meta?.requestId,line:r},e instanceof Error?e:void 0))}}),s.on("error",e=>{c&&clearTimeout(c),i(eF(e,a.meta?.requestId,!1))})})}async function eq(e,r,o){var n,i,s;let l,c=e.baseUrl?new URL(ej(e.baseUrl,"rpc")):e.httpPort?new URL(`http://127.0.0.1:${e.httpPort}/rpc`):null;if(!c)throw new p("COMMAND_FAILED","Daemon HTTP endpoint is unavailable");let u=JSON.stringify((n=r,i={includeTokenParam:!e.baseUrl},l=n.meta?.requestId??w(),"lease_allocate"!==(s=n.command)&&"lease_heartbeat"!==s&&"lease_release"!==s?{jsonrpc:"2.0",id:l,method:"agent_device.command",params:n}:{jsonrpc:"2.0",id:l,method:function(e){switch(e){case"lease_allocate":return"agent_device.lease.allocate";case"lease_heartbeat":return"agent_device.lease.heartbeat";case"lease_release":return"agent_device.lease.release"}}(n.command),params:function(e,t,a){let r={...a.includeTokenParam?{token:e.token}:{},session:e.session,tenantId:e.meta?.tenantId,runId:e.meta?.runId};switch(t){case"lease_allocate":return{...r,ttlMs:e.meta?.leaseTtlMs,backend:e.meta?.leaseBackend};case"lease_heartbeat":return{...r,leaseId:e.meta?.leaseId,ttlMs:e.meta?.leaseTtlMs};case"lease_release":return{...r,leaseId:e.meta?.leaseId}}}(n,n.command,i)})),m={"content-type":"application/json","content-length":Buffer.byteLength(u)};return e.baseUrl&&e.token&&(m.authorization=`Bearer ${e.token}`,m["x-agent-device-token"]=e.token),await new Promise((n,i)=>{let s=J(r.flags?.stateDir??process.env.AGENT_DEVICE_STATE_DIR),l=("https:"===c.protocol?a:t).request({protocol:c.protocol,host:c.hostname,port:c.port,method:"POST",path:c.pathname+c.search,headers:m},t=>{et(t).then(t=>{h&&clearTimeout(h),function(e,t){let{info:a,req:r,resolve:o,reject:n}=t;try{let t=JSON.parse(e);if(t.error){let e=t.error.data??{};n(new p(d(null!=e.code?String(e.code):void 0,"COMMAND_FAILED"),String(e.message??t.error.message??"Daemon RPC request failed"),{..."object"==typeof e.details&&e.details?e.details:{},hint:"string"==typeof e.hint?e.hint:void 0,diagnosticId:"string"==typeof e.diagnosticId?e.diagnosticId:void 0,logPath:"string"==typeof e.logPath?e.logPath:void 0,requestId:r.meta?.requestId}));return}if(!t.result||"object"!=typeof t.result)return void n(new p("COMMAND_FAILED","Invalid daemon RPC response",{requestId:r.meta?.requestId}));if(a.baseUrl&&t.result.ok)return void e$(a,r,t.result).then(o).catch(n);o(t.result)}catch(t){n(new p("COMMAND_FAILED","Invalid daemon response",{requestId:r.meta?.requestId,line:e},t instanceof Error?t:void 0))}}(t,{info:e,req:r,resolve:n,reject:i})}).catch(e=>{h&&clearTimeout(h),i(new p("COMMAND_FAILED","Failed to read daemon response",{requestId:r.meta?.requestId},e instanceof Error?e:void 0))})}),f=ez(e),h="number"==typeof o?setTimeout(()=>{l.destroy(),i(ex(e,s,r.meta?.requestId,r.command,f,o))},o):void 0;l.on("error",e=>{h&&clearTimeout(h),i(eF(e,r.meta?.requestId,f))}),l.write(u),l.end()})}function ez(e){return"string"==typeof e.baseUrl&&e.baseUrl.length>0}function ej(e,t){return new URL(t,e.endsWith("/")?e:`${e}/`).toString()}async function e$(e,t,a){let r=Array.isArray(a.data?.artifacts)?a.data.artifacts:[];if(0===r.length||!e.baseUrl)return a;let n=a.data?{...a.data}:{},i=[];for(let a of r){if(!a||"object"!=typeof a||"string"!=typeof a.artifactId){i.push(a);continue}let r=function(e,t){if(e.localPath&&e.localPath.trim().length>0)return e.localPath;let a=t.meta?.clientArtifactPaths?.[e.field];if(a&&a.trim().length>0)return a;let r=e.fileName?.trim()||`${e.field}-${Date.now()}`;return o.resolve(t.meta?.cwd??process.cwd(),r)}(a,t);await eH({baseUrl:e.baseUrl,token:e.token,artifactId:a.artifactId,destinationPath:r,requestId:t.meta?.requestId}),n[a.field]=r,i.push({...a,localPath:r})}return n.artifacts=i,{ok:!0,data:n}}async function eH(e){var i,s;let l,c=new URL((i=e.baseUrl,s=e.artifactId,l=i.endsWith("/")?i:`${i}/`,new URL(`artifacts/${encodeURIComponent(s)}`,l).toString())),d="https:"===c.protocol?a:t;await r.promises.mkdir(o.dirname(e.destinationPath),{recursive:!0}),await new Promise((t,a)=>{let o=!1,i=e.timeoutMs??9e4,s=n=>{if(!o){if(o=!0,clearTimeout(u),n)return void r.promises.rm(e.destinationPath,{force:!0}).finally(()=>a(n));t()}},l=d.request({protocol:c.protocol,host:c.hostname,port:c.port,method:"GET",path:c.pathname+c.search,headers:e.token?{authorization:`Bearer ${e.token}`,"x-agent-device-token":e.token}:void 0},t=>{if((t.statusCode??500)>=400){let a="";t.setEncoding("utf8"),t.on("data",e=>{a+=e}),t.on("end",()=>{s(new p("COMMAND_FAILED","Failed to download remote artifact",{artifactId:e.artifactId,statusCode:t.statusCode,requestId:e.requestId,body:a}))});return}t.on("aborted",()=>{s(new p("COMMAND_FAILED","Remote artifact download was interrupted",{artifactId:e.artifactId,requestId:e.requestId}))}),n(t,r.createWriteStream(e.destinationPath)).then(()=>s(),e=>s(e instanceof Error?e:Error(String(e))))}),u=setTimeout(()=>{l.destroy(new p("COMMAND_FAILED","Remote artifact download timed out",{artifactId:e.artifactId,requestId:e.requestId,timeoutMs:i}))},i);l.on("error",t=>{t instanceof p?s(t):s(new p("COMMAND_FAILED","Failed to download remote artifact",{artifactId:e.artifactId,requestId:e.requestId,timeoutMs:i},t instanceof Error?t:void 0))}),l.end()})}function eK(e,t){let a=R(e,"bundleId"),r=R(e,"package");return{app:$(e,"app"),appPath:$(e,"appPath"),platform:O(e,"platform"),appId:a??r,bundleId:a,package:r,identifiers:E({session:t,bundleId:a,packageName:r})}}function eV(e){let{record:t,platform:a,id:r,name:o,target:n}=eJ(e,"name");return{platform:a,target:n,kind:x(t,"kind"),id:r,name:o,booted:"boolean"==typeof t.booted?t.booted:void 0,identifiers:D(a,r,o),...eW(a,r)}}function eG(e){let{record:t,platform:a,id:r,name:o,target:n}=eJ(e,"name"),i=$(t,"device"),s={session:o,...D(a,r,i)};return{name:o,createdAt:U(t,"createdAt"),device:{platform:a,target:n,id:r,name:i,identifiers:s,...eW(a,r,H(t,"ios_simulator_device_set"))},identifiers:s}}function eJ(e,t){let a=_(e);return{record:a,platform:O(a,"platform"),id:$(a,"id"),name:$(a,t),target:K(a,"target")}}function eW(e,t,a){return{ios:"ios"===e?{udid:t,...void 0!==a?{simulatorSetPath:a}:{}}:void 0,android:"android"===e?{serial:t}:void 0}}function eY(e){return e??"default"}function eQ(e={},t={}){var a;let r,o=t.transport??eh,n=async(t,a=[],r={})=>{var n,i;let s=(n=e,i=r,{...n,...i}),l=await o({session:eY(s.session),command:t,positionals:a,flags:L({stateDir:s.stateDir,daemonBaseUrl:s.daemonBaseUrl,daemonAuthToken:s.daemonAuthToken,daemonTransport:s.daemonTransport,daemonServerMode:s.daemonServerMode,tenant:s.tenant,sessionIsolation:s.sessionIsolation,runId:s.runId,leaseId:s.leaseId,leaseBackend:s.leaseBackend,platform:s.platform,target:s.target,device:s.device,udid:s.udid,serial:s.serial,iosSimulatorDeviceSet:s.iosSimulatorDeviceSet,androidDeviceAllowlist:s.androidDeviceAllowlist,surface:s.surface,activity:s.activity,relaunch:s.relaunch,shutdown:s.shutdown,saveScript:s.saveScript,noRecord:s.noRecord,backMode:s.backMode,metroHost:s.metroHost,metroPort:s.metroPort,bundleUrl:s.bundleUrl,launchUrl:s.launchUrl,snapshotInteractiveOnly:s.interactiveOnly,snapshotCompact:s.compact,snapshotDepth:s.depth,snapshotScope:s.scope,snapshotRaw:s.raw,snapshotForceFull:s.forceFull,...G(s),appsFilter:s.appsFilter,out:s.out,count:s.count,fps:s.fps,quality:s.quality,hideTouches:s.hideTouches,intervalMs:s.intervalMs,delayMs:s.delayMs,holdMs:s.holdMs,jitterPx:s.jitterPx,pixels:s.pixels,doubleTap:s.doubleTap,clickButton:s.clickButton,pauseMs:s.pauseMs,pattern:s.pattern,headless:s.headless,restart:s.restart,replayUpdate:s.replayUpdate,replayBackend:s.replayBackend,replayEnv:s.replayEnv,replayShellEnv:s.replayShellEnv,failFast:s.failFast,timeoutMs:s.timeoutMs,retries:s.retries,artifactsDir:s.artifactsDir,reportJunit:s.reportJunit,findFirst:s.findFirst,findLast:s.findLast,networkInclude:s.networkInclude,batchOnError:s.batchOnError,batchMaxSteps:s.batchMaxSteps,batchSteps:s.batchSteps,verbose:s.debug}),runtime:s.runtime,meta:L({requestId:s.requestId,cwd:s.cwd,debug:s.debug,lockPolicy:s.lockPolicy,lockPlatform:s.lockPlatform,tenantId:s.tenant,runId:s.runId,leaseId:s.leaseId,leaseBackend:s.leaseBackend,leaseTtlMs:s.leaseTtlMs,sessionIsolation:s.sessionIsolation,installSource:s.installSource,retainMaterializedPaths:s.retainMaterializedPaths,materializedPathRetentionMs:s.materializedPathRetentionMs,materializationId:s.materializationId})});return l.ok||function(e){throw new p(d(e.code),e.message,{...e.details??{},hint:e.hint,diagnosticId:e.diagnosticId,logPath:e.logPath})}(l.error),l.data??{}},i=async(e={})=>{let t=await n(B.sessionList,[],e);return(Array.isArray(t.sessions)?t.sessions:[]).map(eG)},s=async(e,t=[],a={})=>await n(e,t,a),l=(t={})=>{var a,r;return eY((a=e,r=t,{...a,...r}).session)};return{command:(a=async e=>await n(e.command,e.positionals,e.options),r=async e=>await a(e),{wait:async e=>{var t;return await r((t=e,{command:P.wait,positionals:Y.encode(t),options:t}))},alert:async(e={})=>{var t;let a;return await r((a=(t=e).action??"get",{command:P.alert,positionals:[a,...void 0!==t.timeoutMs?[String(t.timeoutMs)]:[]],options:t}))},appState:async(e={})=>await r({command:P.appState,positionals:[],options:e}),back:async(e={})=>await r({command:P.back,positionals:[],options:{...e,backMode:e.mode}}),home:async(e={})=>await r({command:P.home,positionals:[],options:e}),rotate:async e=>await r({command:P.rotate,positionals:[e.orientation],options:e}),appSwitcher:async(e={})=>await r({command:P.appSwitcher,positionals:[],options:e}),keyboard:async(e={})=>await r({command:P.keyboard,positionals:e.action?[e.action]:[],options:e}),clipboard:async e=>{var t;return await r("read"===(t=e).action?{command:P.clipboard,positionals:["read"],options:t}:{command:P.clipboard,positionals:["write",t.text],options:t})},reactNative:async e=>await r({command:P.reactNative,positionals:[e.action],options:e})}),devices:{list:async(e={})=>{let t=await n(P.devices,[],e);return(Array.isArray(t.devices)?t.devices:[]).map(eV)},boot:async(e={})=>await s(P.boot,[],e)},sessions:{list:async(e={})=>await i(e),close:async(e={})=>{let t=l(e),a=(await n(P.close,[],e)).shutdown;return{session:t,shutdown:"object"==typeof a&&null!==a?a:void 0,identifiers:{session:t}}}},apps:{install:async e=>eK(await n(P.install,[e.app,e.appPath],e),l(e)),reinstall:async e=>eK(await n(P.reinstall,[e.app,e.appPath],e),l(e)),installFromSource:async e=>(function(e,t){let a=R(e,"bundleId"),r=R(e,"packageName"),o=a??r??R(e,"appId"),n=R(e,"launchTarget")??r??a??o;if(!n)throw new p("COMMAND_FAILED",'Daemon response is missing "launchTarget".',{response:e});return{appName:R(e,"appName"),appId:o,bundleId:a,packageName:r,launchTarget:n,installablePath:R(e,"installablePath"),archivePath:R(e,"archivePath"),materializationId:R(e,"materializationId"),materializationExpiresAt:R(e,"materializationExpiresAt"),identifiers:E({session:t,bundleId:a,packageName:r,appId:o})}})(await n(B.installSource,[],{...e,installSource:e.source,retainMaterializedPaths:e.retainPaths,materializedPathRetentionMs:e.retentionMs}),l(e)),list:async(e={})=>{let t=await n(P.apps,[],e);return Array.isArray(t.apps)?t.apps.filter(e=>"string"==typeof e):[]},open:async e=>{let t=l(e),a=e.app?e.url?[e.app,e.url]:[e.app]:[],r=await n(P.open,a,e),o=function(e){let t=e.platform,a=R(e,"id"),r=R(e,"device");if("ios"!==t&&"macos"!==t&&"android"!==t&&"linux"!==t||!a||!r)return;let o=K(e,"target"),n=D(t,a,r);return{platform:t,target:o,id:a,name:r,identifiers:n,ios:"ios"===t?{udid:R(e,"device_udid")??a,simulatorSetPath:H(e,"ios_simulator_device_set")}:void 0,android:"android"===t?{serial:R(e,"serial")??a}:void 0}}(r),i=R(r,"appBundleId");return{session:t,appName:R(r,"appName"),appBundleId:i,appId:i,startup:function(e){if(j(e)&&"number"==typeof e.durationMs&&"string"==typeof e.measuredAt&&"string"==typeof e.method)return{durationMs:e.durationMs,measuredAt:e.measuredAt,method:e.method,appTarget:R(e,"appTarget"),appBundleId:R(e,"appBundleId")}}(r.startup),runtime:function(e){if(!j(e))return;let t=e.platform,a=R(e,"metroHost"),r="number"==typeof e.metroPort?e.metroPort:void 0;return{platform:"ios"===t||"android"===t?t:void 0,metroHost:a,metroPort:r,bundleUrl:R(e,"bundleUrl"),launchUrl:R(e,"launchUrl")}}(r.runtime),device:o,identifiers:{session:t,deviceId:o?.id,deviceName:o?.name,udid:o?.ios?.udid,serial:o?.android?.serial,appId:i,appBundleId:i}}},close:async(e={})=>{let t=l(e),a=(await n(P.close,e.app?[e.app]:[],e)).shutdown;return{session:t,closedApp:e.app,shutdown:"object"==typeof a&&null!==a?a:void 0,identifiers:{session:t}}},push:async e=>{var t;return await s(P.push,[e.app,"string"==typeof(t=e.payload)?t:JSON.stringify(t)],e)},triggerEvent:async e=>{var t;return await s(P.triggerAppEvent,[(t=e).event,...t.payload?[JSON.stringify(t.payload)]:[]],e)}},materializations:{release:async e=>{var t;return{released:!0===(t=await n(B.releaseMaterializedPaths,[],{...e,materializationId:e.materializationId})).released,materializationId:$(t,"materializationId"),identifiers:{}}}},leases:{allocate:async e=>e2(await n(B.leaseAllocate,[],{...e,leaseId:void 0,leaseTtlMs:e.ttlMs})),heartbeat:async e=>e2(await n(B.leaseHeartbeat,[],{...e,leaseTtlMs:e.ttlMs})),release:async e=>({released:!0===(await n(B.leaseRelease,[],e)).released})},metro:{prepare:async t=>await ee({projectRoot:t.projectRoot??e.cwd,kind:t.kind,publicBaseUrl:t.publicBaseUrl,proxyBaseUrl:t.proxyBaseUrl,proxyBearerToken:t.bearerToken,bridgeScope:t.bridgeScope,launchUrl:t.launchUrl,companionProfileKey:t.companionProfileKey,companionConsumerKey:t.companionConsumerKey,metroPort:t.port,listenHost:t.listenHost,statusHost:t.statusHost,startupTimeoutMs:t.startupTimeoutMs,probeTimeoutMs:t.probeTimeoutMs,reuseExisting:t.reuseExisting,installDependenciesIfNeeded:t.installDependenciesIfNeeded,runtimeFilePath:t.runtimeFilePath,logPath:t.logPath}),reload:async(t={})=>await Z({metroHost:t.metroHost,metroPort:t.metroPort,bundleUrl:t.bundleUrl,runtime:e.runtime,timeoutMs:t.timeoutMs})},capture:{snapshot:async(e={})=>{var t,a,r,o;let i,s,c,d,p,u=l(e);return t=await n(P.snapshot,[],e),a=u,i=R(t,"appBundleId"),{nodes:Array.isArray(r=t.nodes)?r:[],truncated:!0===t.truncated,appName:R(t,"appName"),appBundleId:i,...(s=eX((o=t).visibility),c=eX(o.androidSnapshot),d=eX(o.unchanged),p=Array.isArray(o.warnings)?o.warnings.filter(e=>"string"==typeof e):void 0,{...s?{visibility:s}:{},...c?{androidSnapshot:c}:{},...d?{unchanged:d}:{},...p?{warnings:p}:{}}),identifiers:{session:a,appId:i,appBundleId:i}}},screenshot:async(e={})=>{let t=l(e),a=await n(P.screenshot,e.path?[e.path]:[],{...e,...G(e)});return{path:$(a,"path"),overlayRefs:function(e){let t=e.overlayRefs;if(!Array.isArray(t))return;let a=[];for(let e of t){if(!j(e))continue;let t=R(e,"ref"),r=A(e,"rect"),o=A(e,"overlayRect"),n=C(e,"center");t&&r&&o&&n&&a.push({ref:t,label:R(e,"label"),rect:r,overlayRect:o,center:n})}return a}(a),identifiers:{session:t}}},diff:async e=>await s(P.diff,[e.kind],{...e,interactiveOnly:e.interactiveOnly,compact:e.compact,depth:e.depth,scope:e.scope,raw:e.raw})},interactions:{click:async e=>await s(P.click,Q.encode(e),{...e,clickButton:e.button}),press:async e=>await s(P.press,Q.encode(e),e),longPress:async e=>await s(P.longPress,M.encode(e),e),swipe:async e=>await s(P.swipe,[String(e.from.x),String(e.from.y),String(e.to.x),String(e.to.y),...e0(e.durationMs)],e),focus:async e=>await s(P.focus,[String(e.x),String(e.y)],e),type:async e=>await s(P.type,z.encode(e),e),fill:async e=>await s(P.fill,q.encode(e),e),scroll:async e=>await s(P.scroll,[e.direction,...e0(e.amount)],e),pinch:async e=>await s(P.pinch,[String(e.scale),...e0(e.x),...e0(e.y)],e),get:async e=>await s(P.get,[e.format,...S.encode(e)],e),is:async e=>await s(P.is,V.encode(e),e),find:async e=>await s(P.find,F.encode(e),{...e,findFirst:e.first,findLast:e.last})},replay:{run:async e=>await s(P.replay,[e.path],{...e,replayUpdate:e.update,replayBackend:e.backend??(!0===e.maestro?"maestro":void 0),replayEnv:e.env,replayShellEnv:e1(process.env)}),test:async e=>await s(P.test,e.paths,{...e,replayUpdate:e.update,replayEnv:e.env,replayShellEnv:e1(process.env)})},batch:{run:async e=>await s(P.batch,[],{...e,batchSteps:e.steps,batchOnError:e.onError,batchMaxSteps:e.maxSteps})},observability:{perf:async(e={})=>await s(P.perf,[],e),logs:async(e={})=>{var t;return await s(P.logs,[(t=e).action??"path",...eZ(t.message)],e)},network:async(e={})=>{var t;return await s(P.network,[...(t=e).action?[t.action]:[],...e0(t.limit)],{...e,networkInclude:e.include})}},recording:{record:async e=>await s(P.record,[e.action,...eZ(e.path)],e),trace:async e=>await s(P.trace,[e.action,...eZ(e.path)],e)},settings:{update:async e=>await s(P.settings,k.encode(e),e)}}}function eX(e){return"object"==typeof e&&null!==e?e:void 0}function eZ(e){return void 0===e?[]:[e]}function e0(e){return void 0===e?[]:[String(e)]}function e1(e){let t={};for(let[a,r]of Object.entries(e))"string"==typeof r&&a.startsWith("AD_VAR_")&&(t[a]=r);return t}function e2(e){let t=e.lease;if(!t||"object"!=typeof t||Array.isArray(t))throw Error("Invalid lease response from daemon");return{leaseId:$(t,"leaseId"),tenantId:$(t,"tenantId"),runId:$(t,"runId"),backend:$(t,"backend"),createdAt:"number"==typeof t.createdAt?t.createdAt:void 0,heartbeatAt:"number"==typeof t.heartbeatAt?t.heartbeatAt:void 0,expiresAt:"number"==typeof t.expiresAt?t.expiresAt:void 0}}export{eQ as createAgentDeviceClient,eh as sendToDaemon};
package/dist/src/989.js CHANGED
@@ -1 +1 @@
1
- import t from"node:dns/promises";import e from"node:net";import{createWriteStream as a,promises as r}from"node:fs";import o from"node:os";import i from"node:path";import{Readable as n}from"node:stream";import{pipeline as s}from"node:stream/promises";import{AppError as l}from"./9152.js";import{runCmd as c}from"./9818.js";import{expandUserHomePath as h}from"./3267.js";import{resolveTimeoutMs as u}from"./4829.js";let d=[".zip",".tar",".tar.gz",".tgz"],p=Object.freeze([...d]),w=u(process.env.AGENT_DEVICE_SOURCE_DOWNLOAD_TIMEOUT_MS,12e4,1e3),f=["1","true","yes","on"];async function m(t){let e=[];try{let a=await _(t.source,{signal:t.signal,downloadTimeoutMs:t.downloadTimeoutMs});e.push(a.cleanup);let r=await S(a.localPath,{archivePath:void 0,isInstallablePath:t.isInstallablePath,installableLabel:t.installableLabel,allowArchiveExtraction:!1!==t.allowArchiveExtraction,registerCleanup:t=>{e.push(t)}});return{archivePath:r.archivePath,installablePath:r.installablePath,cleanup:async()=>{await D(e)}}}catch(t){throw await D(e),t}}async function _(t,e){if("path"===t.kind)return{localPath:h(t.path),cleanup:async()=>{}};let a=await r.mkdtemp(i.join(o.tmpdir(),"agent-device-source-"));try{return{localPath:await A(a,t.url,t.headers,e),cleanup:async()=>{await r.rm(a,{recursive:!0,force:!0})}}}catch(t){throw await r.rm(a,{recursive:!0,force:!0}),t}}async function A(t,e,r,o){let c;try{c=new URL(e)}catch{throw new l("INVALID_ARGS",`Invalid source URL: ${e}`)}await L(c);let h=o?.signal;if(h?.aborted)throw new l("COMMAND_FAILED","request canceled",{reason:"request_canceled"});let u=o?.downloadTimeoutMs??w,d=AbortSignal.timeout(u),p=h?AbortSignal.any([h,d]):d;try{let e=await fetch(c,{headers:r,redirect:"follow",signal:p});if(!e.ok)throw new l("COMMAND_FAILED",`Failed to download app source: ${e.status} ${e.statusText}`,{status:e.status,statusText:e.statusText,url:c.toString()});let o=function(t,e){let a=t.headers.get("content-disposition"),r=a?.match(/filename\*?=(?:UTF-8'')?"?([^";]+)"?/i),o=r?.[1]?.trim();if(o)return i.basename(o);let n=i.basename(e.pathname);return n||"downloaded-artifact.bin"}(e,c),h=i.join(t,o),u=e.body;if(!u)throw new l("COMMAND_FAILED","Download response body was empty",{url:c.toString()});return await s(n.fromWeb(u),a(h)),h}catch(t){if(h?.aborted)throw new l("COMMAND_FAILED","request canceled",{reason:"request_canceled"},t);if(d.aborted)throw new l("COMMAND_FAILED",`App source download timed out after ${u}ms`,{timeoutMs:u,url:c.toString()},t);throw t}}async function L(e){if("http:"!==e.protocol&&"https:"!==e.protocol)throw new l("INVALID_ARGS",`Unsupported source URL protocol: ${e.protocol}`);if(!f.includes((process.env.AGENT_DEVICE_ALLOW_PRIVATE_SOURCE_URLS??"").toLowerCase())){if(I(e.hostname.toLowerCase()))throw new l("INVALID_ARGS",`Source URL host is not allowed: ${e.hostname}`,{hint:"Use a public artifact URL, or set AGENT_DEVICE_ALLOW_PRIVATE_SOURCE_URLS=1 for trusted private-network daemons."});if((await t.lookup(e.hostname,{all:!0,verbatim:!0}).catch(()=>[])).some(t=>v(t.address)))throw new l("INVALID_ARGS",`Source URL host resolved to a private or loopback address: ${e.hostname}`,{hint:"Use a public artifact URL, or set AGENT_DEVICE_ALLOW_PRIVATE_SOURCE_URLS=1 for trusted private-network daemons."})}}function b(t){var e,a,r,o;let i=t instanceof URL?t:new URL(t),n=i.hostname.toLowerCase();if(!n)return!1;let s=i.pathname;return e=n,a=s,("api.github.com"===e?/^\/repos\/[^/]+\/[^/]+\/actions\/artifacts\/\d+\/zip$/i.test(a):"github.com"===e&&/^\/[^/]+\/[^/]+\/(?:actions\/runs\/\d+\/artifacts\/\d+|suites\/\d+\/artifacts\/\d+)$/i.test(a))||(r=n,o=s,("expo.dev"===r||!!r.endsWith(".expo.dev"))&&/^\/(?:artifacts\/eas\/|accounts\/[^/]+\/projects\/[^/]+\/builds\/)/i.test(o))}function I(t){return!!(!t||"localhost"===t||t.endsWith(".localhost"))||v(t)}function v(t){let a,r=e.isIP(t);return 4===r?function(t){let e=t.split(".").map(t=>Number.parseInt(t,10));if(4!==e.length||e.some(t=>Number.isNaN(t)||t<0||t>255))return!1;let[a,r]=e;return 10===a||127===a||169===a&&254===r||172===a&&!!(r>=16)&&!!(r<=31)||192===a&&168===r}(t):6===r&&!!("::1"===(a=t.toLowerCase())||a.startsWith("fc")||a.startsWith("fd")||a.startsWith("fe80:"))}async function S(t,e){let a=await r.stat(t).catch(()=>null);if(!a)throw new l("INVALID_ARGS",`App source not found: ${t}`);if(e.isInstallablePath(t,a))return{archivePath:e.archivePath,installablePath:t};if(a.isFile()&&P(t)){if(!e.allowArchiveExtraction)throw new l("INVALID_ARGS",`URL sources must point directly to a ${e.installableLabel}; archive extraction is not allowed`,{path:t});let a=await C(t);return e.registerCleanup(a.cleanup),await S(a.outputPath,{...e,archivePath:e.archivePath??t})}if(a.isDirectory()){let a=await E(t,e.isInstallablePath);if(1===a.length)return{archivePath:e.archivePath,installablePath:a[0]};if(a.length>1)throw new l("INVALID_ARGS",`Found multiple ${e.installableLabel} candidates under ${t}`,{matches:a});let r=await E(t,(t,e)=>e.isFile()&&P(t));if(1===r.length){if(!e.allowArchiveExtraction)throw new l("INVALID_ARGS",`URL sources must point directly to a ${e.installableLabel}; nested archives are not allowed`,{path:r[0]});let t=await C(r[0]);return e.registerCleanup(t.cleanup),await S(t.outputPath,{...e,archivePath:e.archivePath??r[0]})}if(r.length>1)throw new l("INVALID_ARGS",`Found multiple nested archives under ${t}; expected one ${e.installableLabel} source`,{matches:r})}throw new l("INVALID_ARGS",`Expected ${e.installableLabel} source, but got ${t}`)}async function E(t,e){let a=[],o=[{path:t,depth:0}];for(;o.length>0;){let t=o.shift();if(!t)continue;let n=await r.readdir(t.path,{withFileTypes:!0});for(let r of(n.sort((t,e)=>t.name.localeCompare(e.name)),n)){let n=i.join(t.path,r.name);if(e(n,r)){a.push(n);continue}r.isDirectory()&&t.depth<5&&o.push({path:n,depth:t.depth+1})}}return a}async function C(t){let e=await r.mkdtemp(i.join(o.tmpdir(),"agent-device-archive-"));try{return t.toLowerCase().endsWith(".zip")?await R(t,e):t.toLowerCase().endsWith(".tar.gz")||t.toLowerCase().endsWith(".tgz")?await c("tar",["-xzf",t,"-C",e]):await c("tar",["-xf",t,"-C",e]),{outputPath:e,cleanup:async()=>{await r.rm(e,{recursive:!0,force:!0})}}}catch(t){throw await r.rm(e,{recursive:!0,force:!0}),t}}async function R(t,e){await c("unzip",["-q",t,"-d",e])}function P(t){let e=t.toLowerCase();return d.some(t=>e.endsWith(t))}async function D(t){for(let e=t.length-1;e>=0;e-=1)await t[e]()}export{p as ARCHIVE_EXTENSIONS,v as isBlockedIpAddress,I as isBlockedSourceHostname,b as isTrustedInstallSourceUrl,m as materializeInstallablePath,L as validateDownloadSourceUrl};
1
+ import t from"node:dns/promises";import e from"node:net";import{createWriteStream as a,promises as r}from"node:fs";import o from"node:os";import i from"node:path";import{Readable as n}from"node:stream";import{pipeline as s}from"node:stream/promises";import{AppError as l}from"./9152.js";import{runCmd as c}from"./9818.js";import{expandUserHomePath as h}from"./3267.js";let u=[".zip",".tar",".tar.gz",".tgz"],d=Object.freeze([...u]);async function p(t){let e=[];try{let a=await w(t.source,{signal:t.signal,downloadTimeoutMs:t.downloadTimeoutMs});e.push(a.cleanup);let r=await I(a.localPath,{archivePath:void 0,isInstallablePath:t.isInstallablePath,installableLabel:t.installableLabel,allowArchiveExtraction:!1!==t.allowArchiveExtraction,registerCleanup:t=>{e.push(t)}});return{archivePath:r.archivePath,installablePath:r.installablePath,cleanup:async()=>{await S(e)}}}catch(t){throw await S(e),t}}async function w(t,e){if("path"===t.kind)return{localPath:h(t.path),cleanup:async()=>{}};let a=await r.mkdtemp(i.join(o.tmpdir(),"agent-device-source-"));try{return{localPath:await f(a,t.url,t.headers,e),cleanup:async()=>{await r.rm(a,{recursive:!0,force:!0})}}}catch(t){throw await r.rm(a,{recursive:!0,force:!0}),t}}async function f(t,e,r,o){let c;try{c=new URL(e)}catch{throw new l("INVALID_ARGS",`Invalid source URL: ${e}`)}await m(c);let h=o?.signal;if(h?.aborted)throw new l("COMMAND_FAILED","request canceled",{reason:"request_canceled"});let u=o?.downloadTimeoutMs??12e4,d=AbortSignal.timeout(u),p=h?AbortSignal.any([h,d]):d;try{let e=await fetch(c,{headers:r,redirect:"follow",signal:p});if(!e.ok)throw new l("COMMAND_FAILED",`Failed to download app source: ${e.status} ${e.statusText}`,{status:e.status,statusText:e.statusText,url:c.toString()});let o=function(t,e){let a=t.headers.get("content-disposition"),r=a?.match(/filename\*?=(?:UTF-8'')?"?([^";]+)"?/i),o=r?.[1]?.trim();if(o)return i.basename(o);let n=i.basename(e.pathname);return n||"downloaded-artifact.bin"}(e,c),h=i.join(t,o),u=e.body;if(!u)throw new l("COMMAND_FAILED","Download response body was empty",{url:c.toString()});return await s(n.fromWeb(u),a(h)),h}catch(t){if(h?.aborted)throw new l("COMMAND_FAILED","request canceled",{reason:"request_canceled"},t);if(d.aborted)throw new l("COMMAND_FAILED",`App source download timed out after ${u}ms`,{timeoutMs:u,url:c.toString()},t);throw t}}async function m(e){if("http:"!==e.protocol&&"https:"!==e.protocol)throw new l("INVALID_ARGS",`Unsupported source URL protocol: ${e.protocol}`);if(A(e.hostname.toLowerCase()))throw new l("INVALID_ARGS",`Source URL host is not allowed: ${e.hostname}`,{hint:"Use a public artifact URL."});if((await t.lookup(e.hostname,{all:!0,verbatim:!0}).catch(()=>[])).some(t=>L(t.address)))throw new l("INVALID_ARGS",`Source URL host resolved to a private or loopback address: ${e.hostname}`,{hint:"Use a public artifact URL."})}function b(t){var e,a,r,o;let i=t instanceof URL?t:new URL(t),n=i.hostname.toLowerCase();if(!n)return!1;let s=i.pathname;return e=n,a=s,("api.github.com"===e?/^\/repos\/[^/]+\/[^/]+\/actions\/artifacts\/\d+\/zip$/i.test(a):"github.com"===e&&/^\/[^/]+\/[^/]+\/(?:actions\/runs\/\d+\/artifacts\/\d+|suites\/\d+\/artifacts\/\d+)$/i.test(a))||(r=n,o=s,("expo.dev"===r||!!r.endsWith(".expo.dev"))&&/^\/(?:artifacts\/eas\/|accounts\/[^/]+\/projects\/[^/]+\/builds\/)/i.test(o))}function A(t){return!!(!t||"localhost"===t||t.endsWith(".localhost"))||L(t)}function L(t){let a,r=e.isIP(t);return 4===r?function(t){let e=t.split(".").map(t=>Number.parseInt(t,10));if(4!==e.length||e.some(t=>Number.isNaN(t)||t<0||t>255))return!1;let[a,r]=e;return 10===a||127===a||169===a&&254===r||172===a&&!!(r>=16)&&!!(r<=31)||192===a&&168===r}(t):6===r&&!!("::1"===(a=t.toLowerCase())||a.startsWith("fc")||a.startsWith("fd")||a.startsWith("fe80:"))}async function I(t,e){let a=await r.stat(t).catch(()=>null);if(!a)throw new l("INVALID_ARGS",`App source not found: ${t}`);if(e.isInstallablePath(t,a))return{archivePath:e.archivePath,installablePath:t};if(a.isFile()&&g(t)){if(!e.allowArchiveExtraction)throw new l("INVALID_ARGS",`URL sources must point directly to a ${e.installableLabel}; archive extraction is not allowed`,{path:t});let a=await _(t);return e.registerCleanup(a.cleanup),await I(a.outputPath,{...e,archivePath:e.archivePath??t})}if(a.isDirectory()){let a=await v(t,e.isInstallablePath);if(1===a.length)return{archivePath:e.archivePath,installablePath:a[0]};if(a.length>1)throw new l("INVALID_ARGS",`Found multiple ${e.installableLabel} candidates under ${t}`,{matches:a});let r=await v(t,(t,e)=>e.isFile()&&g(t));if(1===r.length){if(!e.allowArchiveExtraction)throw new l("INVALID_ARGS",`URL sources must point directly to a ${e.installableLabel}; nested archives are not allowed`,{path:r[0]});let t=await _(r[0]);return e.registerCleanup(t.cleanup),await I(t.outputPath,{...e,archivePath:e.archivePath??r[0]})}if(r.length>1)throw new l("INVALID_ARGS",`Found multiple nested archives under ${t}; expected one ${e.installableLabel} source`,{matches:r})}throw new l("INVALID_ARGS",`Expected ${e.installableLabel} source, but got ${t}`)}async function v(t,e){let a=[],o=[{path:t,depth:0}];for(;o.length>0;){let t=o.shift();if(!t)continue;let n=await r.readdir(t.path,{withFileTypes:!0});for(let r of(n.sort((t,e)=>t.name.localeCompare(e.name)),n)){let n=i.join(t.path,r.name);if(e(n,r)){a.push(n);continue}r.isDirectory()&&t.depth<5&&o.push({path:n,depth:t.depth+1})}}return a}async function _(t){let e=await r.mkdtemp(i.join(o.tmpdir(),"agent-device-archive-"));try{return t.toLowerCase().endsWith(".zip")?await P(t,e):t.toLowerCase().endsWith(".tar.gz")||t.toLowerCase().endsWith(".tgz")?await c("tar",["-xzf",t,"-C",e]):await c("tar",["-xf",t,"-C",e]),{outputPath:e,cleanup:async()=>{await r.rm(e,{recursive:!0,force:!0})}}}catch(t){throw await r.rm(e,{recursive:!0,force:!0}),t}}async function P(t,e){await c("unzip",["-q",t,"-d",e])}function g(t){let e=t.toLowerCase();return u.some(t=>e.endsWith(t))}async function S(t){for(let e=t.length-1;e>=0;e-=1)await t[e]()}export{d as ARCHIVE_EXTENSIONS,L as isBlockedIpAddress,A as isBlockedSourceHostname,b as isTrustedInstallSourceUrl,p as materializeInstallablePath,m as validateDownloadSourceUrl};
@@ -1 +1 @@
1
- import{AppError as t}from"./9152.js";import{resolveAppsFilter as e}from"./840.js";import{parseAndroidForegroundApp as i,parseAndroidLaunchablePackages as a,parseAndroidUserInstalledPackages as r,inferAndroidAppName as o,parseAndroidLaunchComponent as n,isAmStartError as d}from"./1769.js";let l="android.intent.category.LAUNCHER",s="android.intent.category.LEANBACK_LAUNCHER";async function c(t,i={}){let a=await A(t,i.target??"auto");return("user-installed"===e(i.filter)?(await p(t)).filter(t=>a.has(t)):Array.from(a)).map(t=>({package:t,name:o(t)})).sort((t,e)=>t.package.localeCompare(e.package))}async function u(t){let e=await f(t,[["shell","dumpsys","window","windows"],["shell","dumpsys","window"]]);if(e)return e;let i=await f(t,[["shell","dumpsys","activity","activities"],["shell","dumpsys","activity"]]);return i||{}}async function A(t,e){return new Set((await Promise.all((function(t){switch(t){case"mobile":return[l];case"tv":return[s];default:return[l,s]}})(e).map(async e=>{var i;let r=await t(["shell","cmd","package","query-activities","--brief","-a","android.intent.action.MAIN","-c",e],{allowFailure:!0});return 0===r.exitCode?0===(i=r.stdout).trim().length?[]:a(i):[]}))).flat())}async function p(e){let i=await e(["shell","pm","list","packages","-3"],{allowFailure:!0});if(0!==i.exitCode)throw new t("COMMAND_FAILED","Failed to list Android user-installed apps",{stdout:i.stdout,stderr:i.stderr,exitCode:i.exitCode});return r(i.stdout)}async function f(t,e){for(let a of e){let e=i((await t(a,{allowFailure:!0})).stdout??"");if(e)return e}return null}let w="android.intent.category.LAUNCHER",m="android.intent.category.DEFAULT";async function h(t,e){await t(["shell","am","force-stop",e])}async function C(t,e,i=[w]){for(let a of i){let i=await t(["shell","cmd","package","resolve-activity","--brief","-a","android.intent.action.MAIN","-c",a,e],{allowFailure:!0});if(0!==i.exitCode)continue;let r=n(i.stdout);if(r)return r}return null}async function _(e,i,a={}){let r=a.category??w;if(a.activity){var o,n;return void await e(["shell","am","start","-W","-a","android.intent.action.MAIN","-c",m,"-c",r,"-n",(o=i,(n=a.activity).includes("/")?n:`${o}/${n.startsWith(".")?n:`.${n}`}`)])}let l=await e(["shell","am","start","-W","-a","android.intent.action.MAIN","-c",m,"-c",r,"-p",i],{allowFailure:!0});if(0===l.exitCode&&!d(l.stdout,l.stderr))return;let s=await C(e,i,[r]);if(!s)throw new t("COMMAND_FAILED",`Failed to resolve Android launch component for ${i}`,{stdout:l.stdout,stderr:l.stderr,exitCode:l.exitCode});await e(["shell","am","start","-W","-a","android.intent.action.MAIN","-c",m,"-c",r,"-n",s])}export{captureAndroidLogcatWithAdb,dismissAndroidKeyboardWithAdb,getAndroidKeyboardStatusWithAdb,readAndroidClipboardWithAdb,streamAndroidLogcatWithAdb,writeAndroidClipboardWithAdb}from"./1769.js";export{createAndroidPortReverseManager,createLocalAndroidAdbProvider}from"./9639.js";export{h as forceStopAndroidAppWithAdb,u as getAndroidAppStateWithAdb,c as listAndroidAppsWithAdb,_ as openAndroidAppWithAdb,C as resolveAndroidLaunchComponentWithAdb};
1
+ import{AppError as t}from"./9152.js";import{resolveAppsFilter as e}from"./1393.js";import{parseAndroidForegroundApp as i,parseAndroidLaunchablePackages as a,parseAndroidUserInstalledPackages as r,inferAndroidAppName as o,parseAndroidLaunchComponent as n,isAmStartError as d}from"./1769.js";let l="android.intent.category.LAUNCHER",s="android.intent.category.LEANBACK_LAUNCHER";async function c(t,i={}){let a=await A(t,i.target??"auto");return("user-installed"===e(i.filter)?(await p(t)).filter(t=>a.has(t)):Array.from(a)).map(t=>({package:t,name:o(t)})).sort((t,e)=>t.package.localeCompare(e.package))}async function u(t){let e=await f(t,[["shell","dumpsys","window","windows"],["shell","dumpsys","window"]]);if(e)return e;let i=await f(t,[["shell","dumpsys","activity","activities"],["shell","dumpsys","activity"]]);return i||{}}async function A(t,e){return new Set((await Promise.all((function(t){switch(t){case"mobile":return[l];case"tv":return[s];default:return[l,s]}})(e).map(async e=>{var i;let r=await t(["shell","cmd","package","query-activities","--brief","-a","android.intent.action.MAIN","-c",e],{allowFailure:!0});return 0===r.exitCode?0===(i=r.stdout).trim().length?[]:a(i):[]}))).flat())}async function p(e){let i=await e(["shell","pm","list","packages","-3"],{allowFailure:!0});if(0!==i.exitCode)throw new t("COMMAND_FAILED","Failed to list Android user-installed apps",{stdout:i.stdout,stderr:i.stderr,exitCode:i.exitCode});return r(i.stdout)}async function f(t,e){for(let a of e){let e=i((await t(a,{allowFailure:!0})).stdout??"");if(e)return e}return null}let w="android.intent.category.LAUNCHER",m="android.intent.category.DEFAULT";async function h(t,e){await t(["shell","am","force-stop",e])}async function C(t,e,i=[w]){for(let a of i){let i=await t(["shell","cmd","package","resolve-activity","--brief","-a","android.intent.action.MAIN","-c",a,e],{allowFailure:!0});if(0!==i.exitCode)continue;let r=n(i.stdout);if(r)return r}return null}async function _(e,i,a={}){let r=a.category??w;if(a.activity){var o,n;return void await e(["shell","am","start","-W","-a","android.intent.action.MAIN","-c",m,"-c",r,"-n",(o=i,(n=a.activity).includes("/")?n:`${o}/${n.startsWith(".")?n:`.${n}`}`)])}let l=await e(["shell","am","start","-W","-a","android.intent.action.MAIN","-c",m,"-c",r,"-p",i],{allowFailure:!0});if(0===l.exitCode&&!d(l.stdout,l.stderr))return;let s=await C(e,i,[r]);if(!s)throw new t("COMMAND_FAILED",`Failed to resolve Android launch component for ${i}`,{stdout:l.stdout,stderr:l.stderr,exitCode:l.exitCode});await e(["shell","am","start","-W","-a","android.intent.action.MAIN","-c",m,"-c",r,"-n",s])}export{captureAndroidLogcatWithAdb,dismissAndroidKeyboardWithAdb,getAndroidKeyboardStatusWithAdb,readAndroidClipboardWithAdb,streamAndroidLogcatWithAdb,writeAndroidClipboardWithAdb}from"./1769.js";export{createAndroidPortReverseManager,createLocalAndroidAdbProvider}from"./9639.js";export{h as forceStopAndroidAppWithAdb,u as getAndroidAppStateWithAdb,c as listAndroidAppsWithAdb,_ as openAndroidAppWithAdb,C as resolveAndroidLaunchComponentWithAdb};