// Ios Team Id tests cover ios team id script behavior. import { execFileSync } from "node:child_process"; import { chmodSync } from "node:fs"; import { mkdir, writeFile } from "node:fs/promises"; import path from "node:path"; import { afterAll, beforeAll, describe, expect, it } from "vitest"; import { cleanupTempDirs, makeTempDir } from "../helpers/temp-dir.js"; const SCRIPT = path.join(process.cwd(), "scripts", "ios-team-id.sh"); const BASH_BIN = process.platform === "win32" ? "bash" : "/bin/bash"; const BASH_ARGS = process.platform === "win32" ? [SCRIPT] : ["--noprofile", "--norc", SCRIPT]; const BASE_PATH = process.env.PATH ?? "/usr/bin:/bin"; const BASE_LANG = process.env.LANG ?? "C"; const CANONICAL_TEAM_ID = "FWJYW4S8P8"; let fixtureRoot = ""; let sharedBinDir = ""; let sharedHomeDir = ""; let sharedHomeBinDir = ""; let sharedFakePythonPath = ""; const tempDirs: string[] = []; const runScriptCache = new Map(); type TeamCandidate = { teamId: string; isFree: boolean; teamName: string; }; function parseTeamCandidateRows(raw: string): TeamCandidate[] { const candidates: TeamCandidate[] = []; for (const rawLine of raw.split("\n")) { const line = rawLine.replace(/\r/g, "").trim(); if (!line) { continue; } const parts = line.split("\t"); if (parts.length < 3) { continue; } const teamId = parts[0] ?? ""; if (!teamId) { continue; } candidates.push({ teamId, isFree: (parts[1] ?? "0") === "1", teamName: parts[2] ?? "", }); } return candidates; } function pickTeamIdFromCandidates(params: { candidates: TeamCandidate[]; canonicalTeamId?: string; preferredTeamId?: string; preferredTeamName?: string; preferNonFreeTeam?: boolean; requireCanonical?: boolean; }): string | undefined { const canonicalTeamId = (params.canonicalTeamId ?? CANONICAL_TEAM_ID).trim(); if (canonicalTeamId) { const canonical = params.candidates.find((candidate) => candidate.teamId === canonicalTeamId); if (canonical || params.requireCanonical) { return canonical?.teamId; } } const preferredTeamId = (params.preferredTeamId ?? "").trim(); if (preferredTeamId) { const preferred = params.candidates.find((candidate) => candidate.teamId === preferredTeamId); if (preferred) { return preferred.teamId; } } const preferredTeamName = (params.preferredTeamName ?? "").trim().toLowerCase(); if (preferredTeamName) { const preferredByName = params.candidates.find( (candidate) => candidate.teamName.trim().toLowerCase() === preferredTeamName, ); if (preferredByName) { return preferredByName.teamId; } } if (params.preferNonFreeTeam !== false) { const paid = params.candidates.find((candidate) => !candidate.isFree); if (paid) { return paid.teamId; } } return params.candidates[0]?.teamId; } async function writeExecutable(filePath: string, body: string): Promise { await writeFile(filePath, body, "utf8"); chmodSync(filePath, 0o755); } function runScript( homeDir: string, extraEnv: Record = {}, scriptArgs: string[] = [], ): { ok: boolean; stdout: string; stderr: string; } { const extraEnvKey = Object.keys(extraEnv) .toSorted((a, b) => a.localeCompare(b)) .map((key) => `${key}=${extraEnv[key] ?? ""}`) .join("\u0001"); const cacheKey = `${homeDir}\u0000${extraEnvKey}\u0000${scriptArgs.join("\u0001")}`; const cached = runScriptCache.get(cacheKey); if (cached) { return cached; } const binDir = path.join(homeDir, "bin"); const env = { HOME: homeDir, PATH: `${binDir}${path.delimiter}${sharedBinDir}${path.delimiter}${BASE_PATH}`, LANG: BASE_LANG, ...extraEnv, }; try { const stdout = execFileSync(BASH_BIN, [...BASH_ARGS, ...scriptArgs], { env, encoding: "utf8", stdio: ["ignore", "pipe", "pipe"], }); const result = { ok: true, stdout: stdout.trim(), stderr: "" }; runScriptCache.set(cacheKey, result); return result; } catch (error) { const e = error as { stdout?: unknown; stderr?: unknown }; const stdout = typeof e.stdout === "string" ? e.stdout : Buffer.isBuffer(e.stdout) ? e.stdout.toString("utf8") : ""; const stderr = typeof e.stderr === "string" ? e.stderr : Buffer.isBuffer(e.stderr) ? e.stderr.toString("utf8") : ""; const result = { ok: false, stdout: stdout.trim(), stderr: stderr.trim() }; runScriptCache.set(cacheKey, result); return result; } } describe("scripts/ios-team-id.sh", () => { beforeAll(async () => { fixtureRoot = makeTempDir(tempDirs, "openclaw-ios-team-id-"); sharedBinDir = path.join(fixtureRoot, "shared-bin"); await mkdir(sharedBinDir, { recursive: true }); sharedHomeDir = path.join(fixtureRoot, "home"); sharedHomeBinDir = path.join(sharedHomeDir, "bin"); await mkdir(sharedHomeBinDir, { recursive: true }); await mkdir(path.join(sharedHomeDir, "Library", "Preferences"), { recursive: true }); await writeFile( path.join(sharedHomeDir, "Library", "Preferences", "com.apple.dt.Xcode.plist"), "", ); await writeExecutable( path.join(sharedBinDir, "plutil"), `#!/usr/bin/env bash echo '{}'`, ); await writeExecutable( path.join(sharedBinDir, "defaults"), `#!/usr/bin/env bash if [[ "$3" == "DVTDeveloperAccountManagerAppleIDLists" ]]; then echo '(identifier = "dev@example.com";)' exit 0 fi exit 0`, ); await writeExecutable( path.join(sharedBinDir, "security"), `#!/usr/bin/env bash if [[ "$1" == "cms" && "$2" == "-D" ]]; then if [[ "$4" == *"one.mobileprovision" ]]; then cat <<'PLIST' TeamIdentifierAAAAA11111 PLIST exit 0 fi if [[ "$4" == *"two.mobileprovision" ]]; then cat <<'PLIST' TeamIdentifierBBBBB22222 PLIST exit 0 fi fi exit 1`, ); sharedFakePythonPath = path.join(sharedHomeBinDir, "fake-python"); await writeExecutable( sharedFakePythonPath, `#!/usr/bin/env bash printf 'AAAAA11111\\t0\\tAlpha Team\\r\\n' printf 'BBBBB22222\\t0\\tBeta Team\\r\\n'`, ); }); afterAll(async () => { cleanupTempDirs(tempDirs); }); it("parses team listings and prioritizes preferred IDs without shelling out", () => { const rows = parseTeamCandidateRows( "AAAAA11111\t1\tAlpha Team\r\nBBBBB22222\t0\tBeta Team\r\n", ); expect(rows).toStrictEqual([ { teamId: "AAAAA11111", isFree: true, teamName: "Alpha Team" }, { teamId: "BBBBB22222", isFree: false, teamName: "Beta Team" }, ]); const preferred = pickTeamIdFromCandidates({ candidates: rows, preferredTeamId: "BBBBB22222", }); expect(preferred).toBe("BBBBB22222"); const missingCanonical = pickTeamIdFromCandidates({ candidates: rows, requireCanonical: true, }); expect(missingCanonical).toBeUndefined(); const fallback = pickTeamIdFromCandidates({ candidates: rows, preferredTeamId: "CCCCCC3333", }); expect(fallback).toBe("BBBBB22222"); }); it("prefers the canonical OpenClaw iOS team when it is present", async () => { const homeDir = makeTempDir(tempDirs, "openclaw-ios-team-id-canonical-"); const binDir = path.join(homeDir, "bin"); await mkdir(path.join(homeDir, "Library", "Preferences"), { recursive: true }); await mkdir(binDir, { recursive: true }); await writeFile( path.join(homeDir, "Library", "Preferences", "com.apple.dt.Xcode.plist"), "", "utf8", ); const fakePythonPath = path.join(binDir, "fake-python"); await writeExecutable( fakePythonPath, `#!/usr/bin/env bash printf 'AAAAA11111\\t0\\tAlpha Team\\r\\n' printf '${CANONICAL_TEAM_ID}\\t0\\tOpenClaw\\r\\n'`, ); const result = runScript(homeDir, { IOS_PYTHON_BIN: fakePythonPath }); expect(result.ok).toBe(true); expect(result.stdout).toBe(CANONICAL_TEAM_ID); }); it("loads teams from Xcode account identifier team metadata", async () => { const homeDir = makeTempDir(tempDirs, "openclaw-ios-team-id-by-identifier-"); const binDir = path.join(homeDir, "bin"); await mkdir(path.join(homeDir, "Library", "Preferences"), { recursive: true }); await mkdir(binDir, { recursive: true }); await writeFile( path.join(homeDir, "Library", "Preferences", "com.apple.dt.Xcode.plist"), "", "utf8", ); await writeExecutable( path.join(binDir, "plutil"), `#!/usr/bin/env bash if [[ "$1" == "-extract" && "$2" == "IDEProvisioningTeamByIdentifier" ]]; then cat <<'JSON' {"account-id":[{"teamID":"FWJYW4S8P8","teamName":"OpenClaw Foundation","isFreeProvisioningTeam":false,"teamType":"Company"}]} JSON exit 0 fi echo '{}'`, ); const result = runScript(homeDir, { IOS_PYTHON_BIN: "python3" }, ["--require-canonical"]); expect(result.ok).toBe(true); expect(result.stdout).toBe(CANONICAL_TEAM_ID); }); it("resolves a fallback team ID from Xcode team listings (smoke)", () => { const fallbackResult = runScript(sharedHomeDir, { IOS_PYTHON_BIN: sharedFakePythonPath }); expect(fallbackResult.ok).toBe(true); expect(fallbackResult.stdout).toBe("AAAAA11111"); }); it("fails canonical-only resolution when only fallback teams are available", () => { const result = runScript(sharedHomeDir, { IOS_PYTHON_BIN: sharedFakePythonPath }, [ "--require-canonical", ]); expect(result.ok).toBe(false); expect(result.stderr).toContain( `Canonical OpenClaw iOS Team ID '${CANONICAL_TEAM_ID}' is not available`, ); }); it("rejects explicit non-canonical teams in canonical-only mode", () => { const result = runScript(sharedHomeDir, { IOS_DEVELOPMENT_TEAM: "BBBBB22222" }, [ "--require-canonical", ]); expect(result.ok).toBe(false); expect(result.stderr).toContain("is not the canonical OpenClaw iOS team"); }); it("prints actionable guidance when Xcode account exists but no Team ID is resolvable", () => { const result = runScript(sharedHomeDir); expect(result.ok).toBe(false); expect( result.stderr.includes("An Apple account is signed in to Xcode") || result.stderr.includes("No Apple Team ID found in Xcode accounts"), ).toBe(true); expect( result.stderr.includes("IOS_DEVELOPMENT_TEAM") || result.stderr.includes("IOS_ALLOW_KEYCHAIN_TEAM_FALLBACK"), ).toBe(true); }); });