@10play/expo-air 0.1.0

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 (98) hide show
  1. package/README.md +156 -0
  2. package/android/build.gradle +43 -0
  3. package/android/src/main/AndroidManifest.xml +2 -0
  4. package/android/src/main/java/expo/modules/expoair/ExpoAirModule.kt +50 -0
  5. package/android/src/main/java/expo/modules/expoair/ExpoAirView.kt +30 -0
  6. package/build/ExpoAir.types.d.ts +25 -0
  7. package/build/ExpoAir.types.d.ts.map +1 -0
  8. package/build/ExpoAir.types.js +2 -0
  9. package/build/ExpoAir.types.js.map +1 -0
  10. package/build/ExpoAirModule.d.ts +17 -0
  11. package/build/ExpoAirModule.d.ts.map +1 -0
  12. package/build/ExpoAirModule.js +4 -0
  13. package/build/ExpoAirModule.js.map +1 -0
  14. package/build/ExpoAirModule.web.d.ts +10 -0
  15. package/build/ExpoAirModule.web.d.ts.map +1 -0
  16. package/build/ExpoAirModule.web.js +12 -0
  17. package/build/ExpoAirModule.web.js.map +1 -0
  18. package/build/ExpoAirView.d.ts +4 -0
  19. package/build/ExpoAirView.d.ts.map +1 -0
  20. package/build/ExpoAirView.js +7 -0
  21. package/build/ExpoAirView.js.map +1 -0
  22. package/build/ExpoAirView.web.d.ts +4 -0
  23. package/build/ExpoAirView.web.d.ts.map +1 -0
  24. package/build/ExpoAirView.web.js +7 -0
  25. package/build/ExpoAirView.web.js.map +1 -0
  26. package/build/index.d.ts +4 -0
  27. package/build/index.d.ts.map +1 -0
  28. package/build/index.js +6 -0
  29. package/build/index.js.map +1 -0
  30. package/cli/dist/bin/expo-air.d.ts +3 -0
  31. package/cli/dist/bin/expo-air.d.ts.map +1 -0
  32. package/cli/dist/bin/expo-air.js +51 -0
  33. package/cli/dist/bin/expo-air.js.map +1 -0
  34. package/cli/dist/commands/fly.d.ts +11 -0
  35. package/cli/dist/commands/fly.d.ts.map +1 -0
  36. package/cli/dist/commands/fly.js +324 -0
  37. package/cli/dist/commands/fly.js.map +1 -0
  38. package/cli/dist/commands/init.d.ts +7 -0
  39. package/cli/dist/commands/init.d.ts.map +1 -0
  40. package/cli/dist/commands/init.js +126 -0
  41. package/cli/dist/commands/init.js.map +1 -0
  42. package/cli/dist/commands/server.d.ts +7 -0
  43. package/cli/dist/commands/server.d.ts.map +1 -0
  44. package/cli/dist/commands/server.js +40 -0
  45. package/cli/dist/commands/server.js.map +1 -0
  46. package/cli/dist/commands/start.d.ts +12 -0
  47. package/cli/dist/commands/start.d.ts.map +1 -0
  48. package/cli/dist/commands/start.js +266 -0
  49. package/cli/dist/commands/start.js.map +1 -0
  50. package/cli/dist/server/promptServer.d.ts +38 -0
  51. package/cli/dist/server/promptServer.d.ts.map +1 -0
  52. package/cli/dist/server/promptServer.js +591 -0
  53. package/cli/dist/server/promptServer.js.map +1 -0
  54. package/cli/dist/tsconfig.tsbuildinfo +1 -0
  55. package/cli/dist/tunnel/bore.d.ts +24 -0
  56. package/cli/dist/tunnel/bore.d.ts.map +1 -0
  57. package/cli/dist/tunnel/bore.js +157 -0
  58. package/cli/dist/tunnel/bore.js.map +1 -0
  59. package/cli/dist/tunnel/cloudflare.d.ts +17 -0
  60. package/cli/dist/tunnel/cloudflare.d.ts.map +1 -0
  61. package/cli/dist/tunnel/cloudflare.js +159 -0
  62. package/cli/dist/tunnel/cloudflare.js.map +1 -0
  63. package/cli/dist/tunnel/localhost-run.d.ts +13 -0
  64. package/cli/dist/tunnel/localhost-run.d.ts.map +1 -0
  65. package/cli/dist/tunnel/localhost-run.js +65 -0
  66. package/cli/dist/tunnel/localhost-run.js.map +1 -0
  67. package/cli/dist/tunnel/localtunnel.d.ts +12 -0
  68. package/cli/dist/tunnel/localtunnel.d.ts.map +1 -0
  69. package/cli/dist/tunnel/localtunnel.js +23 -0
  70. package/cli/dist/tunnel/localtunnel.js.map +1 -0
  71. package/cli/dist/types/messages.d.ts +90 -0
  72. package/cli/dist/types/messages.d.ts.map +1 -0
  73. package/cli/dist/types/messages.js +8 -0
  74. package/cli/dist/types/messages.js.map +1 -0
  75. package/expo-module.config.json +10 -0
  76. package/ios/ExpoAir.podspec +34 -0
  77. package/ios/ExpoAirAppDelegateSubscriber.swift +85 -0
  78. package/ios/ExpoAirModule.swift +88 -0
  79. package/ios/ExpoAirView.swift +38 -0
  80. package/ios/FloatingBubbleManager.swift +483 -0
  81. package/ios/WidgetBridge.h +4 -0
  82. package/ios/WidgetBridge.mm +64 -0
  83. package/ios/WidgetRuntime.h +11 -0
  84. package/ios/WidgetRuntime.mm +153 -0
  85. package/package.json +90 -0
  86. package/plugin/build/index.d.ts +3 -0
  87. package/plugin/build/index.js +152 -0
  88. package/widget/BubbleContent.tsx +474 -0
  89. package/widget/app.json +8 -0
  90. package/widget/babel.config.js +6 -0
  91. package/widget/components/GitChangesTab.tsx +148 -0
  92. package/widget/components/PromptInput.tsx +150 -0
  93. package/widget/components/ResponseArea.tsx +253 -0
  94. package/widget/index.ts +5 -0
  95. package/widget/metro.config.js +28 -0
  96. package/widget/package.json +16 -0
  97. package/widget/services/websocket.ts +316 -0
  98. package/widget/tsconfig.json +4 -0
@@ -0,0 +1,157 @@
1
+ import { spawn, execSync } from "child_process";
2
+ import { createWriteStream, existsSync, mkdirSync, chmodSync } from "fs";
3
+ import { join } from "path";
4
+ import { get } from "https";
5
+ import { homedir, platform, arch } from "os";
6
+ const BORE_VERSION = "0.6.0";
7
+ const BORE_DEFAULT_SERVER = process.env.BORE_SERVER || "bore.pub";
8
+ export class BoreTunnel {
9
+ process = null;
10
+ tunnelUrl = null;
11
+ binPath;
12
+ constructor() {
13
+ const cacheDir = join(homedir(), ".cache", "expo-air");
14
+ if (!existsSync(cacheDir)) {
15
+ mkdirSync(cacheDir, { recursive: true });
16
+ }
17
+ this.binPath = join(cacheDir, "bore");
18
+ }
19
+ async ensureBinary() {
20
+ // Check if binary exists and is correct version
21
+ if (existsSync(this.binPath)) {
22
+ try {
23
+ const version = execSync(`"${this.binPath}" --version`, { encoding: "utf-8" });
24
+ if (version.includes(BORE_VERSION)) {
25
+ return;
26
+ }
27
+ console.log(` Updating bore to v${BORE_VERSION}...`);
28
+ }
29
+ catch {
30
+ // Binary exists but can't run, re-download
31
+ }
32
+ }
33
+ const os = platform();
34
+ const architecture = arch();
35
+ let target;
36
+ if (os === "darwin") {
37
+ target = architecture === "arm64"
38
+ ? "aarch64-apple-darwin"
39
+ : "x86_64-apple-darwin";
40
+ }
41
+ else if (os === "linux") {
42
+ target = architecture === "arm64"
43
+ ? "aarch64-unknown-linux-musl"
44
+ : "x86_64-unknown-linux-musl";
45
+ }
46
+ else {
47
+ throw new Error(`Unsupported platform: ${os}-${architecture}`);
48
+ }
49
+ const url = `https://github.com/ekzhang/bore/releases/download/v${BORE_VERSION}/bore-v${BORE_VERSION}-${target}.tar.gz`;
50
+ console.log(` Downloading bore v${BORE_VERSION}...`);
51
+ const tarPath = `${this.binPath}.tar.gz`;
52
+ await this.downloadFile(url, tarPath);
53
+ // Extract binary
54
+ const cacheDir = join(homedir(), ".cache", "expo-air");
55
+ execSync(`tar -xzf "${tarPath}" -C "${cacheDir}"`, { stdio: "ignore" });
56
+ execSync(`rm "${tarPath}"`, { stdio: "ignore" });
57
+ // Make executable
58
+ chmodSync(this.binPath, 0o755);
59
+ console.log(` ✓ bore installed to ${this.binPath}`);
60
+ }
61
+ downloadFile(url, dest) {
62
+ return new Promise((resolve, reject) => {
63
+ const file = createWriteStream(dest);
64
+ const request = (currentUrl) => {
65
+ get(currentUrl, (response) => {
66
+ // Handle redirects
67
+ if (response.statusCode === 302 || response.statusCode === 301) {
68
+ const redirectUrl = response.headers.location;
69
+ if (redirectUrl) {
70
+ request(redirectUrl);
71
+ return;
72
+ }
73
+ }
74
+ if (response.statusCode !== 200) {
75
+ reject(new Error(`Download failed: ${response.statusCode}`));
76
+ return;
77
+ }
78
+ response.pipe(file);
79
+ file.on("finish", () => {
80
+ file.close();
81
+ resolve();
82
+ });
83
+ }).on("error", (err) => {
84
+ reject(err);
85
+ });
86
+ };
87
+ request(url);
88
+ });
89
+ }
90
+ async start(config) {
91
+ await this.ensureBinary();
92
+ const server = config.server || BORE_DEFAULT_SERVER;
93
+ const secret = config.secret || process.env.BORE_SECRET;
94
+ return new Promise((resolve, reject) => {
95
+ const args = ["local", config.localPort.toString(), "--to", server];
96
+ // Add fixed remote port if specified
97
+ if (config.remotePort) {
98
+ args.push("-p", config.remotePort.toString());
99
+ }
100
+ // Add secret for HMAC authentication
101
+ if (secret) {
102
+ args.push("-s", secret);
103
+ }
104
+ this.process = spawn(this.binPath, args, {
105
+ stdio: ["ignore", "pipe", "pipe"],
106
+ });
107
+ let output = "";
108
+ let resolved = false;
109
+ const parseUrl = (data) => {
110
+ output += data;
111
+ // bore outputs: "listening at bore.pub:XXXXX"
112
+ const match = output.match(/listening at ([^\s]+):(\d+)/);
113
+ if (match && !resolved) {
114
+ resolved = true;
115
+ const [, host, portStr] = match;
116
+ const port = parseInt(portStr, 10);
117
+ this.tunnelUrl = `ws://${host}:${port}`;
118
+ resolve({
119
+ url: this.tunnelUrl,
120
+ host: `${host}:${port}`,
121
+ port,
122
+ });
123
+ }
124
+ };
125
+ this.process.stdout?.on("data", (data) => parseUrl(data.toString()));
126
+ this.process.stderr?.on("data", (data) => parseUrl(data.toString()));
127
+ this.process.on("error", (err) => {
128
+ if (!resolved) {
129
+ reject(new Error(`Failed to start bore: ${err.message}`));
130
+ }
131
+ });
132
+ this.process.on("exit", (code) => {
133
+ if (!resolved) {
134
+ reject(new Error(`bore exited with code ${code}\nOutput: ${output}`));
135
+ }
136
+ });
137
+ // Timeout after 30 seconds
138
+ setTimeout(() => {
139
+ if (!resolved) {
140
+ this.stop();
141
+ reject(new Error(`Tunnel connection timeout.\nOutput: ${output}`));
142
+ }
143
+ }, 30000);
144
+ });
145
+ }
146
+ async stop() {
147
+ if (this.process) {
148
+ this.process.kill("SIGTERM");
149
+ this.process = null;
150
+ this.tunnelUrl = null;
151
+ }
152
+ }
153
+ getUrl() {
154
+ return this.tunnelUrl;
155
+ }
156
+ }
157
+ //# sourceMappingURL=bore.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bore.js","sourceRoot":"","sources":["../../tunnel/bore.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,17 @@
1
+ interface TunnelInfo {
2
+ url: string;
3
+ host: string;
4
+ }
5
+ export declare class CloudflareTunnel {
6
+ private process;
7
+ private tunnelUrl;
8
+ private binPath;
9
+ constructor();
10
+ ensureBinary(): Promise<void>;
11
+ private downloadFile;
12
+ start(port: number): Promise<TunnelInfo>;
13
+ stop(): Promise<void>;
14
+ getUrl(): string | null;
15
+ }
16
+ export {};
17
+ //# sourceMappingURL=cloudflare.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cloudflare.d.ts","sourceRoot":"","sources":["../../tunnel/cloudflare.ts"],"names":[],"mappings":"AAMA,UAAU,UAAU;IAClB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;CACd;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,OAAO,CAA6B;IAC5C,OAAO,CAAC,SAAS,CAAuB;IACxC,OAAO,CAAC,OAAO,CAAS;;IAUlB,YAAY,IAAI,OAAO,CAAC,IAAI,CAAC;IAwDnC,OAAO,CAAC,YAAY;IAkCd,KAAK,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,CAAC;IAwDxC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAQ3B,MAAM,IAAI,MAAM,GAAG,IAAI;CAGxB"}
@@ -0,0 +1,159 @@
1
+ import { spawn, execSync } from "child_process";
2
+ import { createWriteStream, existsSync, mkdirSync, chmodSync, unlinkSync } from "fs";
3
+ import { join } from "path";
4
+ import { get } from "https";
5
+ import { homedir, platform, arch } from "os";
6
+ export class CloudflareTunnel {
7
+ process = null;
8
+ tunnelUrl = null;
9
+ binPath;
10
+ constructor() {
11
+ const cacheDir = join(homedir(), ".cache", "expo-air");
12
+ if (!existsSync(cacheDir)) {
13
+ mkdirSync(cacheDir, { recursive: true });
14
+ }
15
+ this.binPath = join(cacheDir, "cloudflared");
16
+ }
17
+ async ensureBinary() {
18
+ // Check if binary exists
19
+ if (existsSync(this.binPath)) {
20
+ try {
21
+ execSync(`"${this.binPath}" --version`, { encoding: "utf-8", stdio: "pipe" });
22
+ return;
23
+ }
24
+ catch {
25
+ // Binary exists but can't run, re-download
26
+ }
27
+ }
28
+ const os = platform();
29
+ const architecture = arch();
30
+ let downloadUrl;
31
+ let isZip = false;
32
+ if (os === "darwin") {
33
+ // macOS - use .tgz
34
+ if (architecture === "arm64") {
35
+ downloadUrl = "https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-darwin-arm64.tgz";
36
+ }
37
+ else {
38
+ downloadUrl = "https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-darwin-amd64.tgz";
39
+ }
40
+ }
41
+ else if (os === "linux") {
42
+ // Linux - direct binary
43
+ if (architecture === "arm64") {
44
+ downloadUrl = "https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-linux-arm64";
45
+ }
46
+ else {
47
+ downloadUrl = "https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-linux-amd64";
48
+ }
49
+ }
50
+ else if (os === "win32") {
51
+ downloadUrl = "https://github.com/cloudflare/cloudflared/releases/latest/download/cloudflared-windows-amd64.exe";
52
+ }
53
+ else {
54
+ throw new Error(`Unsupported platform: ${os}-${architecture}`);
55
+ }
56
+ console.log(` Downloading cloudflared...`);
57
+ const isTgz = downloadUrl.endsWith(".tgz");
58
+ const cacheDir = join(homedir(), ".cache", "expo-air");
59
+ if (isTgz) {
60
+ const tgzPath = `${this.binPath}.tgz`;
61
+ await this.downloadFile(downloadUrl, tgzPath);
62
+ execSync(`tar -xzf "${tgzPath}" -C "${cacheDir}"`, { stdio: "ignore" });
63
+ unlinkSync(tgzPath);
64
+ }
65
+ else {
66
+ await this.downloadFile(downloadUrl, this.binPath);
67
+ }
68
+ // Make executable
69
+ chmodSync(this.binPath, 0o755);
70
+ console.log(` ✓ cloudflared installed`);
71
+ }
72
+ downloadFile(url, dest) {
73
+ return new Promise((resolve, reject) => {
74
+ const file = createWriteStream(dest);
75
+ const request = (currentUrl) => {
76
+ get(currentUrl, (response) => {
77
+ // Handle redirects
78
+ if (response.statusCode === 302 || response.statusCode === 301) {
79
+ const redirectUrl = response.headers.location;
80
+ if (redirectUrl) {
81
+ request(redirectUrl);
82
+ return;
83
+ }
84
+ }
85
+ if (response.statusCode !== 200) {
86
+ reject(new Error(`Download failed: ${response.statusCode}`));
87
+ return;
88
+ }
89
+ response.pipe(file);
90
+ file.on("finish", () => {
91
+ file.close();
92
+ resolve();
93
+ });
94
+ }).on("error", (err) => {
95
+ reject(err);
96
+ });
97
+ };
98
+ request(url);
99
+ });
100
+ }
101
+ async start(port) {
102
+ await this.ensureBinary();
103
+ return new Promise((resolve, reject) => {
104
+ // cloudflared tunnel --url http://localhost:PORT
105
+ this.process = spawn(this.binPath, [
106
+ "tunnel",
107
+ "--url", `http://localhost:${port}`,
108
+ ], {
109
+ stdio: ["ignore", "pipe", "pipe"],
110
+ });
111
+ let output = "";
112
+ let resolved = false;
113
+ const parseUrl = (data) => {
114
+ output += data;
115
+ // cloudflared outputs something like:
116
+ // "https://random-words.trycloudflare.com"
117
+ const match = output.match(/https:\/\/[a-z0-9-]+\.trycloudflare\.com/i);
118
+ if (match && !resolved) {
119
+ resolved = true;
120
+ this.tunnelUrl = match[0];
121
+ resolve({
122
+ url: this.tunnelUrl,
123
+ host: this.tunnelUrl.replace("https://", ""),
124
+ });
125
+ }
126
+ };
127
+ this.process.stdout?.on("data", (data) => parseUrl(data.toString()));
128
+ this.process.stderr?.on("data", (data) => parseUrl(data.toString()));
129
+ this.process.on("error", (err) => {
130
+ if (!resolved) {
131
+ reject(new Error(`Failed to start cloudflared: ${err.message}`));
132
+ }
133
+ });
134
+ this.process.on("exit", (code) => {
135
+ if (!resolved) {
136
+ reject(new Error(`cloudflared exited with code ${code}\nOutput: ${output}`));
137
+ }
138
+ });
139
+ // Timeout after 30 seconds
140
+ setTimeout(() => {
141
+ if (!resolved) {
142
+ this.stop();
143
+ reject(new Error(`Tunnel connection timeout.\nOutput: ${output}`));
144
+ }
145
+ }, 30000);
146
+ });
147
+ }
148
+ async stop() {
149
+ if (this.process) {
150
+ this.process.kill("SIGTERM");
151
+ this.process = null;
152
+ this.tunnelUrl = null;
153
+ }
154
+ }
155
+ getUrl() {
156
+ return this.tunnelUrl;
157
+ }
158
+ }
159
+ //# sourceMappingURL=cloudflare.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cloudflare.js","sourceRoot":"","sources":["../../tunnel/cloudflare.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAgB,QAAQ,EAAE,MAAM,eAAe,CAAC;AAC9D,OAAO,EAAE,iBAAiB,EAAE,UAAU,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,IAAI,CAAC;AACrF,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,GAAG,EAAE,MAAM,OAAO,CAAC;AAC5B,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,IAAI,EAAE,MAAM,IAAI,CAAC;AAO7C,MAAM,OAAO,gBAAgB;IACnB,OAAO,GAAwB,IAAI,CAAC;IACpC,SAAS,GAAkB,IAAI,CAAC;IAChC,OAAO,CAAS;IAExB;QACE,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;QACvD,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC1B,SAAS,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3C,CAAC;QACD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,QAAQ,EAAE,aAAa,CAAC,CAAC;IAC/C,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,yBAAyB;QACzB,IAAI,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7B,IAAI,CAAC;gBACH,QAAQ,CAAC,IAAI,IAAI,CAAC,OAAO,aAAa,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;gBAC9E,OAAO;YACT,CAAC;YAAC,MAAM,CAAC;gBACP,2CAA2C;YAC7C,CAAC;QACH,CAAC;QAED,MAAM,EAAE,GAAG,QAAQ,EAAE,CAAC;QACtB,MAAM,YAAY,GAAG,IAAI,EAAE,CAAC;QAE5B,IAAI,WAAmB,CAAC;QACxB,IAAI,KAAK,GAAG,KAAK,CAAC;QAElB,IAAI,EAAE,KAAK,QAAQ,EAAE,CAAC;YACpB,mBAAmB;YACnB,IAAI,YAAY,KAAK,OAAO,EAAE,CAAC;gBAC7B,WAAW,GAAG,iGAAiG,CAAC;YAClH,CAAC;iBAAM,CAAC;gBACN,WAAW,GAAG,iGAAiG,CAAC;YAClH,CAAC;QACH,CAAC;aAAM,IAAI,EAAE,KAAK,OAAO,EAAE,CAAC;YAC1B,wBAAwB;YACxB,IAAI,YAAY,KAAK,OAAO,EAAE,CAAC;gBAC7B,WAAW,GAAG,4FAA4F,CAAC;YAC7G,CAAC;iBAAM,CAAC;gBACN,WAAW,GAAG,4FAA4F,CAAC;YAC7G,CAAC;QACH,CAAC;aAAM,IAAI,EAAE,KAAK,OAAO,EAAE,CAAC;YAC1B,WAAW,GAAG,kGAAkG,CAAC;QACnH,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,yBAAyB,EAAE,IAAI,YAAY,EAAE,CAAC,CAAC;QACjE,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,8BAA8B,CAAC,CAAC;QAE5C,MAAM,KAAK,GAAG,WAAW,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,EAAE,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;QAEvD,IAAI,KAAK,EAAE,CAAC;YACV,MAAM,OAAO,GAAG,GAAG,IAAI,CAAC,OAAO,MAAM,CAAC;YACtC,MAAM,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;YAC9C,QAAQ,CAAC,aAAa,OAAO,SAAS,QAAQ,GAAG,EAAE,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC;YACxE,UAAU,CAAC,OAAO,CAAC,CAAC;QACtB,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,CAAC,YAAY,CAAC,WAAW,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QACrD,CAAC;QAED,kBAAkB;QAClB,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAC/B,OAAO,CAAC,GAAG,CAAC,2BAA2B,CAAC,CAAC;IAC3C,CAAC;IAEO,YAAY,CAAC,GAAW,EAAE,IAAY;QAC5C,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,IAAI,GAAG,iBAAiB,CAAC,IAAI,CAAC,CAAC;YAErC,MAAM,OAAO,GAAG,CAAC,UAAkB,EAAE,EAAE;gBACrC,GAAG,CAAC,UAAU,EAAE,CAAC,QAAQ,EAAE,EAAE;oBAC3B,mBAAmB;oBACnB,IAAI,QAAQ,CAAC,UAAU,KAAK,GAAG,IAAI,QAAQ,CAAC,UAAU,KAAK,GAAG,EAAE,CAAC;wBAC/D,MAAM,WAAW,GAAG,QAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC;wBAC9C,IAAI,WAAW,EAAE,CAAC;4BAChB,OAAO,CAAC,WAAW,CAAC,CAAC;4BACrB,OAAO;wBACT,CAAC;oBACH,CAAC;oBAED,IAAI,QAAQ,CAAC,UAAU,KAAK,GAAG,EAAE,CAAC;wBAChC,MAAM,CAAC,IAAI,KAAK,CAAC,oBAAoB,QAAQ,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC;wBAC7D,OAAO;oBACT,CAAC;oBAED,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBACpB,IAAI,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;wBACrB,IAAI,CAAC,KAAK,EAAE,CAAC;wBACb,OAAO,EAAE,CAAC;oBACZ,CAAC,CAAC,CAAC;gBACL,CAAC,CAAC,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;oBACrB,MAAM,CAAC,GAAG,CAAC,CAAC;gBACd,CAAC,CAAC,CAAC;YACL,CAAC,CAAC;YAEF,OAAO,CAAC,GAAG,CAAC,CAAC;QACf,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,IAAY;QACtB,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QAE1B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,iDAAiD;YACjD,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE;gBACjC,QAAQ;gBACR,OAAO,EAAE,oBAAoB,IAAI,EAAE;aACpC,EAAE;gBACD,KAAK,EAAE,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,CAAC;aAClC,CAAC,CAAC;YAEH,IAAI,MAAM,GAAG,EAAE,CAAC;YAChB,IAAI,QAAQ,GAAG,KAAK,CAAC;YAErB,MAAM,QAAQ,GAAG,CAAC,IAAY,EAAE,EAAE;gBAChC,MAAM,IAAI,IAAI,CAAC;gBAEf,sCAAsC;gBACtC,2CAA2C;gBAC3C,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,2CAA2C,CAAC,CAAC;gBACxE,IAAI,KAAK,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACvB,QAAQ,GAAG,IAAI,CAAC;oBAChB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;oBAC1B,OAAO,CAAC;wBACN,GAAG,EAAE,IAAI,CAAC,SAAS;wBACnB,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,UAAU,EAAE,EAAE,CAAC;qBAC7C,CAAC,CAAC;gBACL,CAAC;YACH,CAAC,CAAC;YAEF,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;YACrE,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;YAErE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBAC/B,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACd,MAAM,CAAC,IAAI,KAAK,CAAC,gCAAgC,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBACnE,CAAC;YACH,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;gBAC/B,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACd,MAAM,CAAC,IAAI,KAAK,CAAC,gCAAgC,IAAI,aAAa,MAAM,EAAE,CAAC,CAAC,CAAC;gBAC/E,CAAC;YACH,CAAC,CAAC,CAAC;YAEH,2BAA2B;YAC3B,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACd,IAAI,CAAC,IAAI,EAAE,CAAC;oBACZ,MAAM,CAAC,IAAI,KAAK,CAAC,uCAAuC,MAAM,EAAE,CAAC,CAAC,CAAC;gBACrE,CAAC;YACH,CAAC,EAAE,KAAK,CAAC,CAAC;QACZ,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,IAAI;QACR,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAC7B,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;YACpB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACxB,CAAC;IACH,CAAC;IAED,MAAM;QACJ,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;CACF"}
@@ -0,0 +1,13 @@
1
+ interface TunnelInfo {
2
+ url: string;
3
+ host: string;
4
+ }
5
+ export declare class LocalhostRunTunnel {
6
+ private process;
7
+ private tunnelUrl;
8
+ start(port: number): Promise<TunnelInfo>;
9
+ stop(): Promise<void>;
10
+ getUrl(): string | null;
11
+ }
12
+ export {};
13
+ //# sourceMappingURL=localhost-run.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"localhost-run.d.ts","sourceRoot":"","sources":["../../tunnel/localhost-run.ts"],"names":[],"mappings":"AAEA,UAAU,UAAU;IAClB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;CACd;AAED,qBAAa,kBAAkB;IAC7B,OAAO,CAAC,OAAO,CAA6B;IAC5C,OAAO,CAAC,SAAS,CAAuB;IAElC,KAAK,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,CAAC;IAyDxC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAQ3B,MAAM,IAAI,MAAM,GAAG,IAAI;CAGxB"}
@@ -0,0 +1,65 @@
1
+ import { spawn } from "child_process";
2
+ export class LocalhostRunTunnel {
3
+ process = null;
4
+ tunnelUrl = null;
5
+ async start(port) {
6
+ return new Promise((resolve, reject) => {
7
+ // Use SSH to create tunnel via localhost.run
8
+ // ssh -R 80:localhost:PORT localhost.run
9
+ this.process = spawn("ssh", [
10
+ "-o", "StrictHostKeyChecking=no",
11
+ "-o", "ServerAliveInterval=60",
12
+ "-R", `80:localhost:${port}`,
13
+ "localhost.run"
14
+ ], {
15
+ stdio: ["ignore", "pipe", "pipe"],
16
+ });
17
+ let output = "";
18
+ let resolved = false;
19
+ const parseUrl = (data) => {
20
+ output += data;
21
+ // localhost.run outputs something like:
22
+ // "https://abc123.localhost.run tunneled with tls termination"
23
+ const match = output.match(/https:\/\/[^\s]+\.localhost\.run/);
24
+ if (match && !resolved) {
25
+ resolved = true;
26
+ this.tunnelUrl = match[0];
27
+ resolve({
28
+ url: this.tunnelUrl,
29
+ host: this.tunnelUrl.replace("https://", ""),
30
+ });
31
+ }
32
+ };
33
+ this.process.stdout?.on("data", (data) => parseUrl(data.toString()));
34
+ this.process.stderr?.on("data", (data) => parseUrl(data.toString()));
35
+ this.process.on("error", (err) => {
36
+ if (!resolved) {
37
+ reject(new Error(`Failed to start localhost.run tunnel: ${err.message}`));
38
+ }
39
+ });
40
+ this.process.on("exit", (code) => {
41
+ if (!resolved) {
42
+ reject(new Error(`localhost.run exited with code ${code}\nOutput: ${output}`));
43
+ }
44
+ });
45
+ // Timeout after 30 seconds
46
+ setTimeout(() => {
47
+ if (!resolved) {
48
+ this.stop();
49
+ reject(new Error(`Tunnel connection timeout.\nOutput: ${output}`));
50
+ }
51
+ }, 30000);
52
+ });
53
+ }
54
+ async stop() {
55
+ if (this.process) {
56
+ this.process.kill("SIGTERM");
57
+ this.process = null;
58
+ this.tunnelUrl = null;
59
+ }
60
+ }
61
+ getUrl() {
62
+ return this.tunnelUrl;
63
+ }
64
+ }
65
+ //# sourceMappingURL=localhost-run.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"localhost-run.js","sourceRoot":"","sources":["../../tunnel/localhost-run.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,12 @@
1
+ interface TunnelInfo {
2
+ url: string;
3
+ host: string;
4
+ }
5
+ export declare class LocalTunnel {
6
+ private tunnel;
7
+ start(port: number): Promise<TunnelInfo>;
8
+ stop(): Promise<void>;
9
+ getUrl(): string | null;
10
+ }
11
+ export {};
12
+ //# sourceMappingURL=localtunnel.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"localtunnel.d.ts","sourceRoot":"","sources":["../../tunnel/localtunnel.ts"],"names":[],"mappings":"AAEA,UAAU,UAAU;IAClB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;CACd;AAOD,qBAAa,WAAW;IACtB,OAAO,CAAC,MAAM,CAAoC;IAE5C,KAAK,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,UAAU,CAAC;IAYxC,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAO3B,MAAM,IAAI,MAAM,GAAG,IAAI;CAGxB"}
@@ -0,0 +1,23 @@
1
+ import localtunnel from "localtunnel";
2
+ export class LocalTunnel {
3
+ tunnel = null;
4
+ async start(port) {
5
+ this.tunnel = await localtunnel({ port });
6
+ // localtunnel returns https URLs like https://random-name.loca.lt
7
+ const url = this.tunnel.url;
8
+ return {
9
+ url,
10
+ host: url.replace(/^https?:\/\//, ""),
11
+ };
12
+ }
13
+ async stop() {
14
+ if (this.tunnel) {
15
+ this.tunnel.close();
16
+ this.tunnel = null;
17
+ }
18
+ }
19
+ getUrl() {
20
+ return this.tunnel?.url ?? null;
21
+ }
22
+ }
23
+ //# sourceMappingURL=localtunnel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"localtunnel.js","sourceRoot":"","sources":["../../tunnel/localtunnel.ts"],"names":[],"mappings":"AAAA,OAAO,WAAW,MAAM,aAAa,CAAC;AAYtC,MAAM,OAAO,WAAW;IACd,MAAM,GAA+B,IAAI,CAAC;IAElD,KAAK,CAAC,KAAK,CAAC,IAAY;QACtB,IAAI,CAAC,MAAM,GAAG,MAAM,WAAW,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;QAE1C,kEAAkE;QAClE,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC;QAE5B,OAAO;YACL,GAAG;YACH,IAAI,EAAE,GAAG,CAAC,OAAO,CAAC,cAAc,EAAE,EAAE,CAAC;SACtC,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,IAAI;QACR,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;YACpB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACrB,CAAC;IACH,CAAC;IAED,MAAM;QACJ,OAAO,IAAI,CAAC,MAAM,EAAE,GAAG,IAAI,IAAI,CAAC;IAClC,CAAC;CACF"}
@@ -0,0 +1,90 @@
1
+ /**
2
+ * Message types for communication between the prompt server and widget.
3
+ * These types are shared between:
4
+ * - cli/server/promptServer.ts (server side)
5
+ * - widget/services/websocket.ts (client side - Phase 4)
6
+ */
7
+ export interface PromptMessage {
8
+ type: "prompt";
9
+ id?: string;
10
+ content: string;
11
+ }
12
+ export interface NewSessionMessage {
13
+ type: "new_session";
14
+ }
15
+ export interface StopMessage {
16
+ type: "stop";
17
+ }
18
+ export interface DiscardChangesMessage {
19
+ type: "discard_changes";
20
+ }
21
+ export interface StreamMessage {
22
+ type: "stream";
23
+ promptId: string;
24
+ chunk: string;
25
+ done: boolean;
26
+ timestamp: number;
27
+ }
28
+ export interface ToolMessage {
29
+ type: "tool";
30
+ promptId: string;
31
+ toolName: string;
32
+ status: "started" | "completed" | "failed";
33
+ input?: unknown;
34
+ output?: unknown;
35
+ timestamp: number;
36
+ }
37
+ export interface StatusMessage {
38
+ type: "status";
39
+ status: "idle" | "processing" | "connected";
40
+ promptId?: string;
41
+ branchName?: string;
42
+ timestamp: number;
43
+ }
44
+ export interface GitChange {
45
+ file: string;
46
+ status: "added" | "modified" | "deleted" | "renamed" | "untracked";
47
+ }
48
+ export interface GitStatusMessage {
49
+ type: "git_status";
50
+ branchName: string;
51
+ changes: GitChange[];
52
+ timestamp: number;
53
+ }
54
+ export interface ResultMessage {
55
+ type: "result";
56
+ promptId: string;
57
+ success: boolean;
58
+ result?: string;
59
+ error?: string;
60
+ costUsd?: number;
61
+ durationMs?: number;
62
+ timestamp: number;
63
+ }
64
+ export interface ErrorMessage {
65
+ type: "error";
66
+ promptId?: string;
67
+ message: string;
68
+ timestamp: number;
69
+ }
70
+ export interface SessionClearedMessage {
71
+ type: "session_cleared";
72
+ timestamp: number;
73
+ }
74
+ export interface StoppedMessage {
75
+ type: "stopped";
76
+ timestamp: number;
77
+ }
78
+ export interface ConversationEntry {
79
+ role: "user" | "assistant";
80
+ content: string;
81
+ timestamp: number;
82
+ }
83
+ export interface HistoryMessage {
84
+ type: "history";
85
+ entries: ConversationEntry[];
86
+ timestamp: number;
87
+ }
88
+ export type OutgoingMessage = StreamMessage | ToolMessage | StatusMessage | ResultMessage | ErrorMessage | SessionClearedMessage | StoppedMessage | HistoryMessage | GitStatusMessage;
89
+ export type IncomingMessage = PromptMessage | NewSessionMessage | StopMessage | DiscardChangesMessage;
90
+ //# sourceMappingURL=messages.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"messages.d.ts","sourceRoot":"","sources":["../../types/messages.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Message types for communication between the prompt server and widget.
3
+ * These types are shared between:
4
+ * - cli/server/promptServer.ts (server side)
5
+ * - widget/services/websocket.ts (client side - Phase 4)
6
+ */
7
+ export {};
8
+ //# sourceMappingURL=messages.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"messages.js","sourceRoot":"","sources":["../../types/messages.ts"],"names":[],"mappings":"AAAA;;;;;GAKG"}
@@ -0,0 +1,10 @@
1
+ {
2
+ "platforms": ["apple", "android", "web"],
3
+ "apple": {
4
+ "modules": ["ExpoAirModule"],
5
+ "appDelegateSubscribers": ["ExpoAirAppDelegateSubscriber"]
6
+ },
7
+ "android": {
8
+ "modules": ["expo.modules.expoair.ExpoAirModule"]
9
+ }
10
+ }
@@ -0,0 +1,34 @@
1
+ require 'json'
2
+
3
+ package = JSON.parse(File.read(File.join(__dir__, '..', 'package.json')))
4
+
5
+ Pod::Spec.new do |s|
6
+ s.name = 'ExpoAir'
7
+ s.version = package['version']
8
+ s.summary = package['description']
9
+ s.description = package['description']
10
+ s.license = package['license']
11
+ s.author = package['author']
12
+ s.homepage = package['homepage']
13
+ s.platforms = {
14
+ :ios => '15.1',
15
+ :tvos => '15.1'
16
+ }
17
+ s.swift_version = '5.9'
18
+ s.source = { git: 'https://github.com/10play/expo-air' }
19
+ s.static_framework = true
20
+
21
+ s.dependency 'ExpoModulesCore'
22
+ s.dependency 'React-RCTAppDelegate'
23
+
24
+ s.resource = 'widget.jsbundle'
25
+
26
+ # Swift/Objective-C compatibility
27
+ s.pod_target_xcconfig = {
28
+ 'DEFINES_MODULE' => 'YES',
29
+ 'CLANG_CXX_LANGUAGE_STANDARD' => 'c++20',
30
+ 'HEADER_SEARCH_PATHS' => '"$(PODS_ROOT)/Headers/Public/React-RCTAppDelegate" "$(PODS_ROOT)/Headers/Public/ReactAppDependencyProvider" "$(PODS_ROOT)/Headers/Public/React-defaultsnativemodule" "$(PODS_ROOT)/Headers/Public/ReactCommon" "$(PODS_ROOT)/Headers/Public/React-NativeModulesApple" "$(PODS_ROOT)/Headers/Public/React-callinvoker"',
31
+ }
32
+
33
+ s.source_files = "**/*.{h,m,mm,swift,hpp,cpp}"
34
+ end