runway-cli 0.8.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.
- package/README.md +376 -0
- package/dist/commands/deploy.d.ts +12 -0
- package/dist/commands/deploy.js +334 -0
- package/dist/commands/init.d.ts +5 -0
- package/dist/commands/init.js +196 -0
- package/dist/commands/list.d.ts +1 -0
- package/dist/commands/list.js +84 -0
- package/dist/commands/status.d.ts +1 -0
- package/dist/commands/status.js +117 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +44 -0
- package/dist/services/authService.d.ts +64 -0
- package/dist/services/authService.js +162 -0
- package/dist/services/buildService.d.ts +21 -0
- package/dist/services/buildService.js +180 -0
- package/dist/services/packageService.d.ts +20 -0
- package/dist/services/packageService.js +153 -0
- package/dist/services/projectDetector.d.ts +21 -0
- package/dist/services/projectDetector.js +165 -0
- package/dist/services/uploadService.d.ts +78 -0
- package/dist/services/uploadService.js +222 -0
- package/dist/types.d.ts +3 -0
- package/dist/types.js +4 -0
- package/dist/utils/config.d.ts +19 -0
- package/dist/utils/config.js +102 -0
- package/dist/utils/logger.d.ts +10 -0
- package/dist/utils/logger.js +38 -0
- package/package.json +51 -0
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { ProjectType, BuildMode } from '../types';
|
|
2
|
+
export interface UploadOptions {
|
|
3
|
+
zipPath: string;
|
|
4
|
+
projectName: string;
|
|
5
|
+
projectType: ProjectType;
|
|
6
|
+
version?: string;
|
|
7
|
+
buildMode: BuildMode;
|
|
8
|
+
confirmServerBuild?: boolean;
|
|
9
|
+
deploymentSource?: 'ui' | 'cli';
|
|
10
|
+
envInjected?: boolean;
|
|
11
|
+
}
|
|
12
|
+
export interface UploadResult {
|
|
13
|
+
success: boolean;
|
|
14
|
+
projectId?: string;
|
|
15
|
+
deploymentId?: string;
|
|
16
|
+
error?: string;
|
|
17
|
+
}
|
|
18
|
+
export interface DeploymentStatus {
|
|
19
|
+
status: 'queued' | 'building' | 'deploying' | 'success' | 'failed';
|
|
20
|
+
progress?: number;
|
|
21
|
+
logs?: string;
|
|
22
|
+
error?: string;
|
|
23
|
+
}
|
|
24
|
+
export interface DeployWarning {
|
|
25
|
+
level: 'info' | 'warning' | 'critical';
|
|
26
|
+
message: string;
|
|
27
|
+
code: string;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Analysis result from backend
|
|
31
|
+
* Backend trusts user-declared type - no auto-detection
|
|
32
|
+
*/
|
|
33
|
+
export interface DeployAnalysis {
|
|
34
|
+
declaredType: ProjectType;
|
|
35
|
+
hasPackageJson: boolean;
|
|
36
|
+
hasBuildScript: boolean;
|
|
37
|
+
hasStartScript: boolean;
|
|
38
|
+
hasBuildOutput: boolean;
|
|
39
|
+
buildOutputDir: string | null;
|
|
40
|
+
requiresBuild: boolean;
|
|
41
|
+
isPrebuiltProject: boolean;
|
|
42
|
+
isStaticSite: boolean;
|
|
43
|
+
strategy: 'static' | 'build-and-serve' | 'serve-prebuilt';
|
|
44
|
+
serveMethod: 'caddy-static' | 'pm2-proxy';
|
|
45
|
+
warnings: DeployWarning[];
|
|
46
|
+
requiresConfirmation: boolean;
|
|
47
|
+
confirmationReason?: string;
|
|
48
|
+
}
|
|
49
|
+
export interface AnalyzeResult {
|
|
50
|
+
success: boolean;
|
|
51
|
+
analysis?: DeployAnalysis;
|
|
52
|
+
error?: string;
|
|
53
|
+
}
|
|
54
|
+
export declare class UploadService {
|
|
55
|
+
private serverUrl;
|
|
56
|
+
private token;
|
|
57
|
+
constructor();
|
|
58
|
+
upload(options: UploadOptions): Promise<UploadResult>;
|
|
59
|
+
getDeploymentStatus(deploymentId: string): Promise<DeploymentStatus>;
|
|
60
|
+
pollDeploymentStatus(deploymentId: string, onUpdate: (status: DeploymentStatus) => void, timeoutMs?: number): Promise<DeploymentStatus>;
|
|
61
|
+
login(username: string, password: string): Promise<string>;
|
|
62
|
+
listProjects(): Promise<Array<{
|
|
63
|
+
id: string;
|
|
64
|
+
name: string;
|
|
65
|
+
type: ProjectType;
|
|
66
|
+
status: string;
|
|
67
|
+
}>>;
|
|
68
|
+
/**
|
|
69
|
+
* Analyze a package before deployment to get server warnings and recommendations
|
|
70
|
+
*/
|
|
71
|
+
/**
|
|
72
|
+
* Analyze a package before deployment
|
|
73
|
+
* @param zipPath - Path to the zip file
|
|
74
|
+
* @param declaredType - REQUIRED - User-selected project type (backend trusts this)
|
|
75
|
+
*/
|
|
76
|
+
analyzePackage(zipPath: string, declaredType: ProjectType): Promise<AnalyzeResult>;
|
|
77
|
+
}
|
|
78
|
+
export declare const createUploadService: () => UploadService;
|
|
@@ -0,0 +1,222 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.createUploadService = exports.UploadService = void 0;
|
|
7
|
+
const axios_1 = __importDefault(require("axios"));
|
|
8
|
+
const form_data_1 = __importDefault(require("form-data"));
|
|
9
|
+
const fs_1 = __importDefault(require("fs"));
|
|
10
|
+
const config_1 = require("../utils/config");
|
|
11
|
+
const logger_1 = require("../utils/logger");
|
|
12
|
+
class UploadService {
|
|
13
|
+
constructor() {
|
|
14
|
+
const config = (0, config_1.getConfig)();
|
|
15
|
+
if (!config.serverUrl || !config.token) {
|
|
16
|
+
throw new Error('CLI not configured. Run "runway init" first.');
|
|
17
|
+
}
|
|
18
|
+
this.serverUrl = config.serverUrl;
|
|
19
|
+
this.token = config.token;
|
|
20
|
+
}
|
|
21
|
+
async upload(options) {
|
|
22
|
+
const { zipPath, projectName, projectType, version, buildMode, confirmServerBuild, deploymentSource, envInjected } = options;
|
|
23
|
+
if (!fs_1.default.existsSync(zipPath)) {
|
|
24
|
+
return {
|
|
25
|
+
success: false,
|
|
26
|
+
error: `Zip file not found: ${zipPath}`,
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
const formData = new form_data_1.default();
|
|
30
|
+
formData.append('file', fs_1.default.createReadStream(zipPath));
|
|
31
|
+
formData.append('name', projectName);
|
|
32
|
+
formData.append('type', projectType);
|
|
33
|
+
formData.append('buildMode', buildMode);
|
|
34
|
+
if (version) {
|
|
35
|
+
formData.append('version', version);
|
|
36
|
+
}
|
|
37
|
+
if (confirmServerBuild) {
|
|
38
|
+
formData.append('confirmServerBuild', 'true');
|
|
39
|
+
}
|
|
40
|
+
// ENV mutability tracking
|
|
41
|
+
if (deploymentSource) {
|
|
42
|
+
formData.append('deploymentSource', deploymentSource);
|
|
43
|
+
}
|
|
44
|
+
if (envInjected !== undefined) {
|
|
45
|
+
formData.append('envInjected', envInjected ? 'true' : 'false');
|
|
46
|
+
}
|
|
47
|
+
// Choose endpoint based on build mode
|
|
48
|
+
const endpoint = buildMode === 'local'
|
|
49
|
+
? '/api/project/deploy-prebuilt'
|
|
50
|
+
: '/api/project/deploy';
|
|
51
|
+
logger_1.logger.info(`Uploading to ${this.serverUrl}${endpoint}...`);
|
|
52
|
+
try {
|
|
53
|
+
const response = await axios_1.default.post(`${this.serverUrl}${endpoint}`, formData, {
|
|
54
|
+
headers: {
|
|
55
|
+
...formData.getHeaders(),
|
|
56
|
+
Authorization: `Bearer ${this.token}`,
|
|
57
|
+
},
|
|
58
|
+
maxBodyLength: Infinity,
|
|
59
|
+
maxContentLength: Infinity,
|
|
60
|
+
timeout: 300000, // 5 minutes
|
|
61
|
+
onUploadProgress: (progressEvent) => {
|
|
62
|
+
if (progressEvent.total) {
|
|
63
|
+
const percent = Math.round((progressEvent.loaded * 100) / progressEvent.total);
|
|
64
|
+
process.stdout.write(`\r Uploading: ${percent}%`);
|
|
65
|
+
}
|
|
66
|
+
},
|
|
67
|
+
});
|
|
68
|
+
process.stdout.write('\n');
|
|
69
|
+
if (response.data.success) {
|
|
70
|
+
return {
|
|
71
|
+
success: true,
|
|
72
|
+
projectId: response.data.data?.projectId,
|
|
73
|
+
deploymentId: response.data.data?.deploymentId,
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
return {
|
|
78
|
+
success: false,
|
|
79
|
+
error: response.data.error || 'Unknown error',
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
catch (error) {
|
|
84
|
+
process.stdout.write('\n');
|
|
85
|
+
if (axios_1.default.isAxiosError(error)) {
|
|
86
|
+
const message = error.response?.data?.error || error.message;
|
|
87
|
+
return {
|
|
88
|
+
success: false,
|
|
89
|
+
error: `Upload failed: ${message}`,
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
return {
|
|
93
|
+
success: false,
|
|
94
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
async getDeploymentStatus(deploymentId) {
|
|
99
|
+
try {
|
|
100
|
+
const response = await axios_1.default.get(`${this.serverUrl}/api/project/status/${deploymentId}`, {
|
|
101
|
+
headers: {
|
|
102
|
+
Authorization: `Bearer ${this.token}`,
|
|
103
|
+
},
|
|
104
|
+
});
|
|
105
|
+
return response.data.data;
|
|
106
|
+
}
|
|
107
|
+
catch (error) {
|
|
108
|
+
if (axios_1.default.isAxiosError(error)) {
|
|
109
|
+
throw new Error(error.response?.data?.error || error.message);
|
|
110
|
+
}
|
|
111
|
+
throw error;
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
async pollDeploymentStatus(deploymentId, onUpdate, timeoutMs = 300000 // 5 minutes
|
|
115
|
+
) {
|
|
116
|
+
const startTime = Date.now();
|
|
117
|
+
const pollInterval = 2000; // 2 seconds
|
|
118
|
+
while (Date.now() - startTime < timeoutMs) {
|
|
119
|
+
const status = await this.getDeploymentStatus(deploymentId);
|
|
120
|
+
onUpdate(status);
|
|
121
|
+
if (status.status === 'success' || status.status === 'failed') {
|
|
122
|
+
return status;
|
|
123
|
+
}
|
|
124
|
+
await new Promise(resolve => setTimeout(resolve, pollInterval));
|
|
125
|
+
}
|
|
126
|
+
throw new Error('Deployment timed out');
|
|
127
|
+
}
|
|
128
|
+
async login(username, password) {
|
|
129
|
+
try {
|
|
130
|
+
const response = await axios_1.default.post(`${this.serverUrl}/api/auth/login`, { username, password }, {
|
|
131
|
+
timeout: 10000,
|
|
132
|
+
});
|
|
133
|
+
if (response.data.success && response.data.data?.token) {
|
|
134
|
+
return response.data.data.token;
|
|
135
|
+
}
|
|
136
|
+
throw new Error(response.data.error || 'Login failed');
|
|
137
|
+
}
|
|
138
|
+
catch (error) {
|
|
139
|
+
if (axios_1.default.isAxiosError(error)) {
|
|
140
|
+
throw new Error(error.response?.data?.error || error.message);
|
|
141
|
+
}
|
|
142
|
+
throw error;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
async listProjects() {
|
|
146
|
+
try {
|
|
147
|
+
const response = await axios_1.default.get(`${this.serverUrl}/api/project`, {
|
|
148
|
+
headers: {
|
|
149
|
+
Authorization: `Bearer ${this.token}`,
|
|
150
|
+
},
|
|
151
|
+
});
|
|
152
|
+
return response.data.data || [];
|
|
153
|
+
}
|
|
154
|
+
catch (error) {
|
|
155
|
+
if (axios_1.default.isAxiosError(error)) {
|
|
156
|
+
throw new Error(error.response?.data?.error || error.message);
|
|
157
|
+
}
|
|
158
|
+
throw error;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Analyze a package before deployment to get server warnings and recommendations
|
|
163
|
+
*/
|
|
164
|
+
/**
|
|
165
|
+
* Analyze a package before deployment
|
|
166
|
+
* @param zipPath - Path to the zip file
|
|
167
|
+
* @param declaredType - REQUIRED - User-selected project type (backend trusts this)
|
|
168
|
+
*/
|
|
169
|
+
async analyzePackage(zipPath, declaredType) {
|
|
170
|
+
if (!fs_1.default.existsSync(zipPath)) {
|
|
171
|
+
return {
|
|
172
|
+
success: false,
|
|
173
|
+
error: `Zip file not found: ${zipPath}`,
|
|
174
|
+
};
|
|
175
|
+
}
|
|
176
|
+
const formData = new form_data_1.default();
|
|
177
|
+
formData.append('file', fs_1.default.createReadStream(zipPath));
|
|
178
|
+
formData.append('type', declaredType);
|
|
179
|
+
try {
|
|
180
|
+
const response = await axios_1.default.post(`${this.serverUrl}/api/project/analyze`, formData, {
|
|
181
|
+
headers: {
|
|
182
|
+
...formData.getHeaders(),
|
|
183
|
+
Authorization: `Bearer ${this.token}`,
|
|
184
|
+
},
|
|
185
|
+
maxBodyLength: Infinity,
|
|
186
|
+
maxContentLength: Infinity,
|
|
187
|
+
timeout: 60000, // 1 minute for analysis
|
|
188
|
+
});
|
|
189
|
+
if (response.data.success) {
|
|
190
|
+
return {
|
|
191
|
+
success: true,
|
|
192
|
+
analysis: response.data.data,
|
|
193
|
+
};
|
|
194
|
+
}
|
|
195
|
+
else {
|
|
196
|
+
return {
|
|
197
|
+
success: false,
|
|
198
|
+
error: response.data.error || 'Analysis failed',
|
|
199
|
+
};
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
catch (error) {
|
|
203
|
+
if (axios_1.default.isAxiosError(error)) {
|
|
204
|
+
const message = error.response?.data?.error || error.message;
|
|
205
|
+
return {
|
|
206
|
+
success: false,
|
|
207
|
+
error: `Analysis failed: ${message}`,
|
|
208
|
+
};
|
|
209
|
+
}
|
|
210
|
+
return {
|
|
211
|
+
success: false,
|
|
212
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
213
|
+
};
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
exports.UploadService = UploadService;
|
|
218
|
+
const createUploadService = () => {
|
|
219
|
+
return new UploadService();
|
|
220
|
+
};
|
|
221
|
+
exports.createUploadService = createUploadService;
|
|
222
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"uploadService.js","sourceRoot":"","sources":["../../src/services/uploadService.ts"],"names":[],"mappings":"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","sourcesContent":["import axios, { AxiosProgressEvent } from 'axios';\nimport FormData from 'form-data';\nimport fs from 'fs';\nimport { ProjectType, BuildMode } from '../types';\nimport { getConfig } from '../utils/config';\nimport { logger } from '../utils/logger';\n\nexport interface UploadOptions {\n  zipPath: string;\n  projectName: string;\n  projectType: ProjectType;\n  version?: string;\n  buildMode: BuildMode;\n  confirmServerBuild?: boolean;\n  // ENV mutability tracking\n  deploymentSource?: 'ui' | 'cli';\n  envInjected?: boolean;\n}\n\nexport interface UploadResult {\n  success: boolean;\n  projectId?: string;\n  deploymentId?: string;\n  error?: string;\n}\n\nexport interface DeploymentStatus {\n  status: 'queued' | 'building' | 'deploying' | 'success' | 'failed';\n  progress?: number;\n  logs?: string;\n  error?: string;\n}\n\nexport interface DeployWarning {\n  level: 'info' | 'warning' | 'critical';\n  message: string;\n  code: string;\n}\n\n/**\n * Analysis result from backend\n * Backend trusts user-declared type - no auto-detection\n */\nexport interface DeployAnalysis {\n  // User's declared type (trusted, not validated)\n  declaredType: ProjectType;\n  // Package state\n  hasPackageJson: boolean;\n  hasBuildScript: boolean;\n  hasStartScript: boolean;\n  // Build state (generic detection)\n  hasBuildOutput: boolean;\n  buildOutputDir: string | null;\n  requiresBuild: boolean;\n  // Prebuilt detection (generic)\n  isPrebuiltProject: boolean;\n  // Static site detection (generic)\n  isStaticSite: boolean;\n  // Deployment strategy\n  strategy: 'static' | 'build-and-serve' | 'serve-prebuilt';\n  serveMethod: 'caddy-static' | 'pm2-proxy';\n  // Warnings\n  warnings: DeployWarning[];\n  requiresConfirmation: boolean;\n  confirmationReason?: string;\n}\n\nexport interface AnalyzeResult {\n  success: boolean;\n  analysis?: DeployAnalysis;\n  error?: string;\n}\n\nexport class UploadService {\n  private serverUrl: string;\n  private token: string;\n\n  constructor() {\n    const config = getConfig();\n    if (!config.serverUrl || !config.token) {\n      throw new Error('CLI not configured. Run \"runway init\" first.');\n    }\n    this.serverUrl = config.serverUrl;\n    this.token = config.token;\n  }\n\n  async upload(options: UploadOptions): Promise<UploadResult> {\n    const { zipPath, projectName, projectType, version, buildMode, confirmServerBuild, deploymentSource, envInjected } = options;\n\n    if (!fs.existsSync(zipPath)) {\n      return {\n        success: false,\n        error: `Zip file not found: ${zipPath}`,\n      };\n    }\n\n    const formData = new FormData();\n    formData.append('file', fs.createReadStream(zipPath));\n    formData.append('name', projectName);\n    formData.append('type', projectType);\n    formData.append('buildMode', buildMode);\n    if (version) {\n      formData.append('version', version);\n    }\n    if (confirmServerBuild) {\n      formData.append('confirmServerBuild', 'true');\n    }\n    // ENV mutability tracking\n    if (deploymentSource) {\n      formData.append('deploymentSource', deploymentSource);\n    }\n    if (envInjected !== undefined) {\n      formData.append('envInjected', envInjected ? 'true' : 'false');\n    }\n\n    // Choose endpoint based on build mode\n    const endpoint = buildMode === 'local'\n      ? '/api/project/deploy-prebuilt'\n      : '/api/project/deploy';\n\n    logger.info(`Uploading to ${this.serverUrl}${endpoint}...`);\n\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}${endpoint}`,\n        formData,\n        {\n          headers: {\n            ...formData.getHeaders(),\n            Authorization: `Bearer ${this.token}`,\n          },\n          maxBodyLength: Infinity,\n          maxContentLength: Infinity,\n          timeout: 300000, // 5 minutes\n          onUploadProgress: (progressEvent: AxiosProgressEvent) => {\n            if (progressEvent.total) {\n              const percent = Math.round((progressEvent.loaded * 100) / progressEvent.total);\n              process.stdout.write(`\\r  Uploading: ${percent}%`);\n            }\n          },\n        }\n      );\n\n      process.stdout.write('\\n');\n\n      if (response.data.success) {\n        return {\n          success: true,\n          projectId: response.data.data?.projectId,\n          deploymentId: response.data.data?.deploymentId,\n        };\n      } else {\n        return {\n          success: false,\n          error: response.data.error || 'Unknown error',\n        };\n      }\n    } catch (error) {\n      process.stdout.write('\\n');\n\n      if (axios.isAxiosError(error)) {\n        const message = error.response?.data?.error || error.message;\n        return {\n          success: false,\n          error: `Upload failed: ${message}`,\n        };\n      }\n\n      return {\n        success: false,\n        error: error instanceof Error ? error.message : 'Unknown error',\n      };\n    }\n  }\n\n  async getDeploymentStatus(deploymentId: string): Promise<DeploymentStatus> {\n    try {\n      const response = await axios.get(\n        `${this.serverUrl}/api/project/status/${deploymentId}`,\n        {\n          headers: {\n            Authorization: `Bearer ${this.token}`,\n          },\n        }\n      );\n\n      return response.data.data;\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  async pollDeploymentStatus(\n    deploymentId: string,\n    onUpdate: (status: DeploymentStatus) => void,\n    timeoutMs: number = 300000 // 5 minutes\n  ): Promise<DeploymentStatus> {\n    const startTime = Date.now();\n    const pollInterval = 2000; // 2 seconds\n\n    while (Date.now() - startTime < timeoutMs) {\n      const status = await this.getDeploymentStatus(deploymentId);\n      onUpdate(status);\n\n      if (status.status === 'success' || status.status === 'failed') {\n        return status;\n      }\n\n      await new Promise(resolve => setTimeout(resolve, pollInterval));\n    }\n\n    throw new Error('Deployment timed out');\n  }\n\n  async login(username: string, password: string): Promise<string> {\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}/api/auth/login`,\n        { username, password },\n        {\n          timeout: 10000,\n        }\n      );\n\n      if (response.data.success && response.data.data?.token) {\n        return response.data.data.token;\n      }\n\n      throw new Error(response.data.error || 'Login failed');\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  async listProjects(): Promise<Array<{ id: string; name: string; type: ProjectType; status: string }>> {\n    try {\n      const response = await axios.get(\n        `${this.serverUrl}/api/project`,\n        {\n          headers: {\n            Authorization: `Bearer ${this.token}`,\n          },\n        }\n      );\n\n      return response.data.data || [];\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        throw new Error(error.response?.data?.error || error.message);\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Analyze a package before deployment to get server warnings and recommendations\n   */\n  /**\n   * Analyze a package before deployment\n   * @param zipPath - Path to the zip file\n   * @param declaredType - REQUIRED - User-selected project type (backend trusts this)\n   */\n  async analyzePackage(zipPath: string, declaredType: ProjectType): Promise<AnalyzeResult> {\n    if (!fs.existsSync(zipPath)) {\n      return {\n        success: false,\n        error: `Zip file not found: ${zipPath}`,\n      };\n    }\n\n    const formData = new FormData();\n    formData.append('file', fs.createReadStream(zipPath));\n    formData.append('type', declaredType);\n\n    try {\n      const response = await axios.post(\n        `${this.serverUrl}/api/project/analyze`,\n        formData,\n        {\n          headers: {\n            ...formData.getHeaders(),\n            Authorization: `Bearer ${this.token}`,\n          },\n          maxBodyLength: Infinity,\n          maxContentLength: Infinity,\n          timeout: 60000, // 1 minute for analysis\n        }\n      );\n\n      if (response.data.success) {\n        return {\n          success: true,\n          analysis: response.data.data,\n        };\n      } else {\n        return {\n          success: false,\n          error: response.data.error || 'Analysis failed',\n        };\n      }\n    } catch (error) {\n      if (axios.isAxiosError(error)) {\n        const message = error.response?.data?.error || error.message;\n        return {\n          success: false,\n          error: `Analysis failed: ${message}`,\n        };\n      }\n\n      return {\n        success: false,\n        error: error instanceof Error ? error.message : 'Unknown error',\n      };\n    }\n  }\n}\n\nexport const createUploadService = (): UploadService => {\n  return new UploadService();\n};\n"]}
|
package/dist/types.d.ts
ADDED
package/dist/types.js
ADDED
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Types bundled from @runway/shared for npm distribution
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLHlEQUF5RCIsInNvdXJjZXNDb250ZW50IjpbIi8vIFR5cGVzIGJ1bmRsZWQgZnJvbSBAcnVud2F5L3NoYXJlZCBmb3IgbnBtIGRpc3RyaWJ1dGlvblxuXG5leHBvcnQgdHlwZSBQcm9qZWN0VHlwZSA9ICdyZWFjdCcgfCAnbmV4dCcgfCAnbm9kZScgfCAnc3RhdGljJztcbmV4cG9ydCB0eXBlIFBhY2thZ2VNYW5hZ2VyID0gJ25wbScgfCAneWFybicgfCAncG5wbScgfCAnbm9uZSc7XG5leHBvcnQgdHlwZSBCdWlsZE1vZGUgPSAnbG9jYWwnIHwgJ3NlcnZlcic7XG4iXX0=
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
export type SecurityMode = 'ip-http' | 'domain-https';
|
|
2
|
+
export interface CLIConfig {
|
|
3
|
+
serverUrl?: string;
|
|
4
|
+
token?: string;
|
|
5
|
+
tokenExpiresAt?: string;
|
|
6
|
+
securityMode?: SecurityMode;
|
|
7
|
+
defaultBuildMode?: 'local' | 'server';
|
|
8
|
+
}
|
|
9
|
+
export declare const getConfig: () => CLIConfig;
|
|
10
|
+
export declare const setServerUrl: (url: string) => void;
|
|
11
|
+
export declare const setToken: (token: string) => void;
|
|
12
|
+
export declare const setDefaultBuildMode: (mode: "local" | "server") => void;
|
|
13
|
+
export declare const clearConfig: () => void;
|
|
14
|
+
export declare const isConfigured: () => boolean;
|
|
15
|
+
export declare const setSecurityMode: (mode: SecurityMode) => void;
|
|
16
|
+
export declare const setTokenExpiresAt: (expiresAt: string) => void;
|
|
17
|
+
export declare const setAuthData: (token: string, expiresAt: string, securityMode: SecurityMode) => void;
|
|
18
|
+
export declare const isTokenExpired: () => boolean;
|
|
19
|
+
export declare const getTokenTimeRemaining: () => number;
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.getTokenTimeRemaining = exports.isTokenExpired = exports.setAuthData = exports.setTokenExpiresAt = exports.setSecurityMode = exports.isConfigured = exports.clearConfig = exports.setDefaultBuildMode = exports.setToken = exports.setServerUrl = exports.getConfig = void 0;
|
|
7
|
+
const fs_1 = __importDefault(require("fs"));
|
|
8
|
+
const path_1 = __importDefault(require("path"));
|
|
9
|
+
const os_1 = __importDefault(require("os"));
|
|
10
|
+
const CONFIG_DIR = path_1.default.join(os_1.default.homedir(), '.runway');
|
|
11
|
+
const CONFIG_FILE = path_1.default.join(CONFIG_DIR, 'config.json');
|
|
12
|
+
const ensureConfigDir = () => {
|
|
13
|
+
if (!fs_1.default.existsSync(CONFIG_DIR)) {
|
|
14
|
+
fs_1.default.mkdirSync(CONFIG_DIR, { recursive: true });
|
|
15
|
+
}
|
|
16
|
+
};
|
|
17
|
+
const readConfig = () => {
|
|
18
|
+
ensureConfigDir();
|
|
19
|
+
if (!fs_1.default.existsSync(CONFIG_FILE)) {
|
|
20
|
+
return { defaultBuildMode: 'local' };
|
|
21
|
+
}
|
|
22
|
+
try {
|
|
23
|
+
const content = fs_1.default.readFileSync(CONFIG_FILE, 'utf-8');
|
|
24
|
+
return JSON.parse(content);
|
|
25
|
+
}
|
|
26
|
+
catch {
|
|
27
|
+
return { defaultBuildMode: 'local' };
|
|
28
|
+
}
|
|
29
|
+
};
|
|
30
|
+
const writeConfig = (config) => {
|
|
31
|
+
ensureConfigDir();
|
|
32
|
+
fs_1.default.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));
|
|
33
|
+
};
|
|
34
|
+
const getConfig = () => {
|
|
35
|
+
return readConfig();
|
|
36
|
+
};
|
|
37
|
+
exports.getConfig = getConfig;
|
|
38
|
+
const setServerUrl = (url) => {
|
|
39
|
+
const config = readConfig();
|
|
40
|
+
config.serverUrl = url;
|
|
41
|
+
writeConfig(config);
|
|
42
|
+
};
|
|
43
|
+
exports.setServerUrl = setServerUrl;
|
|
44
|
+
const setToken = (token) => {
|
|
45
|
+
const config = readConfig();
|
|
46
|
+
config.token = token;
|
|
47
|
+
writeConfig(config);
|
|
48
|
+
};
|
|
49
|
+
exports.setToken = setToken;
|
|
50
|
+
const setDefaultBuildMode = (mode) => {
|
|
51
|
+
const config = readConfig();
|
|
52
|
+
config.defaultBuildMode = mode;
|
|
53
|
+
writeConfig(config);
|
|
54
|
+
};
|
|
55
|
+
exports.setDefaultBuildMode = setDefaultBuildMode;
|
|
56
|
+
const clearConfig = () => {
|
|
57
|
+
if (fs_1.default.existsSync(CONFIG_FILE)) {
|
|
58
|
+
fs_1.default.unlinkSync(CONFIG_FILE);
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
exports.clearConfig = clearConfig;
|
|
62
|
+
const isConfigured = () => {
|
|
63
|
+
const cfg = (0, exports.getConfig)();
|
|
64
|
+
return !!(cfg.serverUrl && cfg.token);
|
|
65
|
+
};
|
|
66
|
+
exports.isConfigured = isConfigured;
|
|
67
|
+
const setSecurityMode = (mode) => {
|
|
68
|
+
const config = readConfig();
|
|
69
|
+
config.securityMode = mode;
|
|
70
|
+
writeConfig(config);
|
|
71
|
+
};
|
|
72
|
+
exports.setSecurityMode = setSecurityMode;
|
|
73
|
+
const setTokenExpiresAt = (expiresAt) => {
|
|
74
|
+
const config = readConfig();
|
|
75
|
+
config.tokenExpiresAt = expiresAt;
|
|
76
|
+
writeConfig(config);
|
|
77
|
+
};
|
|
78
|
+
exports.setTokenExpiresAt = setTokenExpiresAt;
|
|
79
|
+
const setAuthData = (token, expiresAt, securityMode) => {
|
|
80
|
+
const config = readConfig();
|
|
81
|
+
config.token = token;
|
|
82
|
+
config.tokenExpiresAt = expiresAt;
|
|
83
|
+
config.securityMode = securityMode;
|
|
84
|
+
writeConfig(config);
|
|
85
|
+
};
|
|
86
|
+
exports.setAuthData = setAuthData;
|
|
87
|
+
const isTokenExpired = () => {
|
|
88
|
+
const config = (0, exports.getConfig)();
|
|
89
|
+
if (!config.tokenExpiresAt)
|
|
90
|
+
return true;
|
|
91
|
+
return new Date(config.tokenExpiresAt) < new Date();
|
|
92
|
+
};
|
|
93
|
+
exports.isTokenExpired = isTokenExpired;
|
|
94
|
+
const getTokenTimeRemaining = () => {
|
|
95
|
+
const config = (0, exports.getConfig)();
|
|
96
|
+
if (!config.tokenExpiresAt)
|
|
97
|
+
return 0;
|
|
98
|
+
const remaining = new Date(config.tokenExpiresAt).getTime() - Date.now();
|
|
99
|
+
return Math.max(0, remaining);
|
|
100
|
+
};
|
|
101
|
+
exports.getTokenTimeRemaining = getTokenTimeRemaining;
|
|
102
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"config.js","sourceRoot":"","sources":["../../src/utils/config.ts"],"names":[],"mappings":";;;;;;AAAA,4CAAoB;AACpB,gDAAwB;AACxB,4CAAoB;AAYpB,MAAM,UAAU,GAAG,cAAI,CAAC,IAAI,CAAC,YAAE,CAAC,OAAO,EAAE,EAAE,SAAS,CAAC,CAAC;AACtD,MAAM,WAAW,GAAG,cAAI,CAAC,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;AAEzD,MAAM,eAAe,GAAG,GAAS,EAAE;IACjC,IAAI,CAAC,YAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;QAC/B,YAAE,CAAC,SAAS,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAChD,CAAC;AACH,CAAC,CAAC;AAEF,MAAM,UAAU,GAAG,GAAc,EAAE;IACjC,eAAe,EAAE,CAAC;IAClB,IAAI,CAAC,YAAE,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAChC,OAAO,EAAE,gBAAgB,EAAE,OAAO,EAAE,CAAC;IACvC,CAAC;IACD,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,YAAE,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;QACtD,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IAC7B,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,gBAAgB,EAAE,OAAO,EAAE,CAAC;IACvC,CAAC;AACH,CAAC,CAAC;AAEF,MAAM,WAAW,GAAG,CAAC,MAAiB,EAAQ,EAAE;IAC9C,eAAe,EAAE,CAAC;IAClB,YAAE,CAAC,aAAa,CAAC,WAAW,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;AACjE,CAAC,CAAC;AAEK,MAAM,SAAS,GAAG,GAAc,EAAE;IACvC,OAAO,UAAU,EAAE,CAAC;AACtB,CAAC,CAAC;AAFW,QAAA,SAAS,aAEpB;AAEK,MAAM,YAAY,GAAG,CAAC,GAAW,EAAQ,EAAE;IAChD,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,CAAC,SAAS,GAAG,GAAG,CAAC;IACvB,WAAW,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC,CAAC;AAJW,QAAA,YAAY,gBAIvB;AAEK,MAAM,QAAQ,GAAG,CAAC,KAAa,EAAQ,EAAE;IAC9C,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC;IACrB,WAAW,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC,CAAC;AAJW,QAAA,QAAQ,YAInB;AAEK,MAAM,mBAAmB,GAAG,CAAC,IAAwB,EAAQ,EAAE;IACpE,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,CAAC,gBAAgB,GAAG,IAAI,CAAC;IAC/B,WAAW,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC,CAAC;AAJW,QAAA,mBAAmB,uBAI9B;AAEK,MAAM,WAAW,GAAG,GAAS,EAAE;IACpC,IAAI,YAAE,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC/B,YAAE,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;IAC7B,CAAC;AACH,CAAC,CAAC;AAJW,QAAA,WAAW,eAItB;AAEK,MAAM,YAAY,GAAG,GAAY,EAAE;IACxC,MAAM,GAAG,GAAG,IAAA,iBAAS,GAAE,CAAC;IACxB,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,KAAK,CAAC,CAAC;AACxC,CAAC,CAAC;AAHW,QAAA,YAAY,gBAGvB;AAEK,MAAM,eAAe,GAAG,CAAC,IAAkB,EAAQ,EAAE;IAC1D,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,CAAC,YAAY,GAAG,IAAI,CAAC;IAC3B,WAAW,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC,CAAC;AAJW,QAAA,eAAe,mBAI1B;AAEK,MAAM,iBAAiB,GAAG,CAAC,SAAiB,EAAQ,EAAE;IAC3D,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,CAAC,cAAc,GAAG,SAAS,CAAC;IAClC,WAAW,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC,CAAC;AAJW,QAAA,iBAAiB,qBAI5B;AAEK,MAAM,WAAW,GAAG,CACzB,KAAa,EACb,SAAiB,EACjB,YAA0B,EACpB,EAAE;IACR,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC;IACrB,MAAM,CAAC,cAAc,GAAG,SAAS,CAAC;IAClC,MAAM,CAAC,YAAY,GAAG,YAAY,CAAC;IACnC,WAAW,CAAC,MAAM,CAAC,CAAC;AACtB,CAAC,CAAC;AAVW,QAAA,WAAW,eAUtB;AAEK,MAAM,cAAc,GAAG,GAAY,EAAE;IAC1C,MAAM,MAAM,GAAG,IAAA,iBAAS,GAAE,CAAC;IAC3B,IAAI,CAAC,MAAM,CAAC,cAAc;QAAE,OAAO,IAAI,CAAC;IACxC,OAAO,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,GAAG,IAAI,IAAI,EAAE,CAAC;AACtD,CAAC,CAAC;AAJW,QAAA,cAAc,kBAIzB;AAEK,MAAM,qBAAqB,GAAG,GAAW,EAAE;IAChD,MAAM,MAAM,GAAG,IAAA,iBAAS,GAAE,CAAC;IAC3B,IAAI,CAAC,MAAM,CAAC,cAAc;QAAE,OAAO,CAAC,CAAC;IACrC,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IACzE,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;AAChC,CAAC,CAAC;AALW,QAAA,qBAAqB,yBAKhC","sourcesContent":["import fs from 'fs';\nimport path from 'path';\nimport os from 'os';\n\nexport type SecurityMode = 'ip-http' | 'domain-https';\n\nexport interface CLIConfig {\n  serverUrl?: string;\n  token?: string;\n  tokenExpiresAt?: string;\n  securityMode?: SecurityMode;\n  defaultBuildMode?: 'local' | 'server';\n}\n\nconst CONFIG_DIR = path.join(os.homedir(), '.runway');\nconst CONFIG_FILE = path.join(CONFIG_DIR, 'config.json');\n\nconst ensureConfigDir = (): void => {\n  if (!fs.existsSync(CONFIG_DIR)) {\n    fs.mkdirSync(CONFIG_DIR, { recursive: true });\n  }\n};\n\nconst readConfig = (): CLIConfig => {\n  ensureConfigDir();\n  if (!fs.existsSync(CONFIG_FILE)) {\n    return { defaultBuildMode: 'local' };\n  }\n  try {\n    const content = fs.readFileSync(CONFIG_FILE, 'utf-8');\n    return JSON.parse(content);\n  } catch {\n    return { defaultBuildMode: 'local' };\n  }\n};\n\nconst writeConfig = (config: CLIConfig): void => {\n  ensureConfigDir();\n  fs.writeFileSync(CONFIG_FILE, JSON.stringify(config, null, 2));\n};\n\nexport const getConfig = (): CLIConfig => {\n  return readConfig();\n};\n\nexport const setServerUrl = (url: string): void => {\n  const config = readConfig();\n  config.serverUrl = url;\n  writeConfig(config);\n};\n\nexport const setToken = (token: string): void => {\n  const config = readConfig();\n  config.token = token;\n  writeConfig(config);\n};\n\nexport const setDefaultBuildMode = (mode: 'local' | 'server'): void => {\n  const config = readConfig();\n  config.defaultBuildMode = mode;\n  writeConfig(config);\n};\n\nexport const clearConfig = (): void => {\n  if (fs.existsSync(CONFIG_FILE)) {\n    fs.unlinkSync(CONFIG_FILE);\n  }\n};\n\nexport const isConfigured = (): boolean => {\n  const cfg = getConfig();\n  return !!(cfg.serverUrl && cfg.token);\n};\n\nexport const setSecurityMode = (mode: SecurityMode): void => {\n  const config = readConfig();\n  config.securityMode = mode;\n  writeConfig(config);\n};\n\nexport const setTokenExpiresAt = (expiresAt: string): void => {\n  const config = readConfig();\n  config.tokenExpiresAt = expiresAt;\n  writeConfig(config);\n};\n\nexport const setAuthData = (\n  token: string,\n  expiresAt: string,\n  securityMode: SecurityMode\n): void => {\n  const config = readConfig();\n  config.token = token;\n  config.tokenExpiresAt = expiresAt;\n  config.securityMode = securityMode;\n  writeConfig(config);\n};\n\nexport const isTokenExpired = (): boolean => {\n  const config = getConfig();\n  if (!config.tokenExpiresAt) return true;\n  return new Date(config.tokenExpiresAt) < new Date();\n};\n\nexport const getTokenTimeRemaining = (): number => {\n  const config = getConfig();\n  if (!config.tokenExpiresAt) return 0;\n  const remaining = new Date(config.tokenExpiresAt).getTime() - Date.now();\n  return Math.max(0, remaining);\n};\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export declare const logger: {
|
|
2
|
+
info: (message: string) => void;
|
|
3
|
+
success: (message: string) => void;
|
|
4
|
+
warn: (message: string) => void;
|
|
5
|
+
error: (message: string) => void;
|
|
6
|
+
step: (step: number, total: number, message: string) => void;
|
|
7
|
+
dim: (message: string) => void;
|
|
8
|
+
blank: () => void;
|
|
9
|
+
header: (message: string) => void;
|
|
10
|
+
};
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.logger = void 0;
|
|
7
|
+
const chalk_1 = __importDefault(require("chalk"));
|
|
8
|
+
exports.logger = {
|
|
9
|
+
info: (message) => {
|
|
10
|
+
console.log(chalk_1.default.blue('ℹ'), message);
|
|
11
|
+
},
|
|
12
|
+
success: (message) => {
|
|
13
|
+
console.log(chalk_1.default.green('✓'), message);
|
|
14
|
+
},
|
|
15
|
+
warn: (message) => {
|
|
16
|
+
console.log(chalk_1.default.yellow('⚠'), message);
|
|
17
|
+
},
|
|
18
|
+
error: (message) => {
|
|
19
|
+
console.log(chalk_1.default.red('✗'), message);
|
|
20
|
+
},
|
|
21
|
+
step: (step, total, message) => {
|
|
22
|
+
console.log(chalk_1.default.cyan(`[${step}/${total}]`), message);
|
|
23
|
+
},
|
|
24
|
+
dim: (message) => {
|
|
25
|
+
console.log(chalk_1.default.dim(message));
|
|
26
|
+
},
|
|
27
|
+
blank: () => {
|
|
28
|
+
console.log('');
|
|
29
|
+
},
|
|
30
|
+
header: (message) => {
|
|
31
|
+
console.log('');
|
|
32
|
+
console.log(chalk_1.default.bold.cyan('━'.repeat(50)));
|
|
33
|
+
console.log(chalk_1.default.bold.cyan(` ${message}`));
|
|
34
|
+
console.log(chalk_1.default.bold.cyan('━'.repeat(50)));
|
|
35
|
+
console.log('');
|
|
36
|
+
},
|
|
37
|
+
};
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3V0aWxzL2xvZ2dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSxrREFBMEI7QUFFYixRQUFBLE1BQU0sR0FBRztJQUNwQixJQUFJLEVBQUUsQ0FBQyxPQUFlLEVBQUUsRUFBRTtRQUN4QixPQUFPLENBQUMsR0FBRyxDQUFDLGVBQUssQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDeEMsQ0FBQztJQUVELE9BQU8sRUFBRSxDQUFDLE9BQWUsRUFBRSxFQUFFO1FBQzNCLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBSyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsRUFBRSxPQUFPLENBQUMsQ0FBQztJQUN6QyxDQUFDO0lBRUQsSUFBSSxFQUFFLENBQUMsT0FBZSxFQUFFLEVBQUU7UUFDeEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0lBQzFDLENBQUM7SUFFRCxLQUFLLEVBQUUsQ0FBQyxPQUFlLEVBQUUsRUFBRTtRQUN6QixPQUFPLENBQUMsR0FBRyxDQUFDLGVBQUssQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDdkMsQ0FBQztJQUVELElBQUksRUFBRSxDQUFDLElBQVksRUFBRSxLQUFhLEVBQUUsT0FBZSxFQUFFLEVBQUU7UUFDckQsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsSUFBSSxDQUFDLElBQUksSUFBSSxJQUFJLEtBQUssR0FBRyxDQUFDLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDekQsQ0FBQztJQUVELEdBQUcsRUFBRSxDQUFDLE9BQWUsRUFBRSxFQUFFO1FBQ3ZCLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBSyxDQUFDLEdBQUcsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO0lBQ2xDLENBQUM7SUFFRCxLQUFLLEVBQUUsR0FBRyxFQUFFO1FBQ1YsT0FBTyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUNsQixDQUFDO0lBRUQsTUFBTSxFQUFFLENBQUMsT0FBZSxFQUFFLEVBQUU7UUFDMUIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQztRQUNoQixPQUFPLENBQUMsR0FBRyxDQUFDLGVBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzdDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBSyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxPQUFPLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDN0MsT0FBTyxDQUFDLEdBQUcsQ0FBQyxlQUFLLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUM3QyxPQUFPLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ2xCLENBQUM7Q0FDRixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGNoYWxrIGZyb20gJ2NoYWxrJztcblxuZXhwb3J0IGNvbnN0IGxvZ2dlciA9IHtcbiAgaW5mbzogKG1lc3NhZ2U6IHN0cmluZykgPT4ge1xuICAgIGNvbnNvbGUubG9nKGNoYWxrLmJsdWUoJ+KEuScpLCBtZXNzYWdlKTtcbiAgfSxcblxuICBzdWNjZXNzOiAobWVzc2FnZTogc3RyaW5nKSA9PiB7XG4gICAgY29uc29sZS5sb2coY2hhbGsuZ3JlZW4oJ+KckycpLCBtZXNzYWdlKTtcbiAgfSxcblxuICB3YXJuOiAobWVzc2FnZTogc3RyaW5nKSA9PiB7XG4gICAgY29uc29sZS5sb2coY2hhbGsueWVsbG93KCfimqAnKSwgbWVzc2FnZSk7XG4gIH0sXG5cbiAgZXJyb3I6IChtZXNzYWdlOiBzdHJpbmcpID0+IHtcbiAgICBjb25zb2xlLmxvZyhjaGFsay5yZWQoJ+KclycpLCBtZXNzYWdlKTtcbiAgfSxcblxuICBzdGVwOiAoc3RlcDogbnVtYmVyLCB0b3RhbDogbnVtYmVyLCBtZXNzYWdlOiBzdHJpbmcpID0+IHtcbiAgICBjb25zb2xlLmxvZyhjaGFsay5jeWFuKGBbJHtzdGVwfS8ke3RvdGFsfV1gKSwgbWVzc2FnZSk7XG4gIH0sXG5cbiAgZGltOiAobWVzc2FnZTogc3RyaW5nKSA9PiB7XG4gICAgY29uc29sZS5sb2coY2hhbGsuZGltKG1lc3NhZ2UpKTtcbiAgfSxcblxuICBibGFuazogKCkgPT4ge1xuICAgIGNvbnNvbGUubG9nKCcnKTtcbiAgfSxcblxuICBoZWFkZXI6IChtZXNzYWdlOiBzdHJpbmcpID0+IHtcbiAgICBjb25zb2xlLmxvZygnJyk7XG4gICAgY29uc29sZS5sb2coY2hhbGsuYm9sZC5jeWFuKCfilIEnLnJlcGVhdCg1MCkpKTtcbiAgICBjb25zb2xlLmxvZyhjaGFsay5ib2xkLmN5YW4oYCAgJHttZXNzYWdlfWApKTtcbiAgICBjb25zb2xlLmxvZyhjaGFsay5ib2xkLmN5YW4oJ+KUgScucmVwZWF0KDUwKSkpO1xuICAgIGNvbnNvbGUubG9nKCcnKTtcbiAgfSxcbn07XG4iXX0=
|
package/package.json
ADDED
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "runway-cli",
|
|
3
|
+
"version": "0.8.0",
|
|
4
|
+
"description": "CLI tool for deploying projects to Runway",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"bin": {
|
|
7
|
+
"runway": "./dist/index.js"
|
|
8
|
+
},
|
|
9
|
+
"files": [
|
|
10
|
+
"dist",
|
|
11
|
+
"README.md"
|
|
12
|
+
],
|
|
13
|
+
"scripts": {
|
|
14
|
+
"build": "tsc",
|
|
15
|
+
"dev": "ts-node src/index.ts",
|
|
16
|
+
"start": "node dist/index.js",
|
|
17
|
+
"prepublishOnly": "npm run build"
|
|
18
|
+
},
|
|
19
|
+
"keywords": [
|
|
20
|
+
"cli",
|
|
21
|
+
"deployment",
|
|
22
|
+
"runway",
|
|
23
|
+
"deploy",
|
|
24
|
+
"hosting"
|
|
25
|
+
],
|
|
26
|
+
"author": "",
|
|
27
|
+
"license": "MIT",
|
|
28
|
+
"repository": {
|
|
29
|
+
"type": "git",
|
|
30
|
+
"url": ""
|
|
31
|
+
},
|
|
32
|
+
"dependencies": {
|
|
33
|
+
"archiver": "^6.0.0",
|
|
34
|
+
"axios": "^1.7.9",
|
|
35
|
+
"chalk": "^4.1.2",
|
|
36
|
+
"commander": "^11.1.0",
|
|
37
|
+
"form-data": "^4.0.0",
|
|
38
|
+
"inquirer": "^8.2.6",
|
|
39
|
+
"ora": "^5.4.1"
|
|
40
|
+
},
|
|
41
|
+
"devDependencies": {
|
|
42
|
+
"@types/archiver": "^6.0.3",
|
|
43
|
+
"@types/inquirer": "^8.2.10",
|
|
44
|
+
"@types/node": "^20.0.0",
|
|
45
|
+
"ts-node": "^10.9.1",
|
|
46
|
+
"typescript": "^5.0.0"
|
|
47
|
+
},
|
|
48
|
+
"engines": {
|
|
49
|
+
"node": ">=18.0.0"
|
|
50
|
+
}
|
|
51
|
+
}
|