__CAPGO_KEEP_0__
复制一个包含安装步骤和完整 Markdown 指南的配置提示。
捆绑包是核心更新包在 Capgo 中。每个捆绑包包含构成应用内容的 Web 资产(HTML、CSS、JS)。捆绑包 API 允许您管理这些更新包,包括列出和删除它们。
了解捆绑包
标题:了解捆绑包捆绑包代表应用的 Web 内容的特定捆绑包(版本),包括:
- 捆绑包(版本): 语义版本号 捆绑包的版本
- 校验和: 验证捆绑包完整性的唯一哈希
- 存储信息: __CAPGO_KEEP_0__
- Native Requirements: __CAPGO_KEEP_1__
- Metadata: __CAPGO_KEEP_2__
手动打包创建(不使用CLI)
手动打包创建(不使用CLI)以下是手动创建和上传打包文件的步骤,未使用Capgo CLI:
步骤 1:构建应用
步骤 1:构建应用首先,构建应用的 Web 资产:
npm run build步骤 2:使用相同的包创建捆绑 ZIP 文件,Capgo CLI
标题:步骤 2:使用相同的包创建捆绑 ZIP 文件,Capgo CLI重要: 使用 Capgo CLI 内部使用的相同 JavaScript 包来确保兼容性。
安装必需的包
标题:安装必需的包npm install adm-zip @tomasklaen/checksum使用 JavaScript 创建 ZIP 捆绑包(与 Capgo CLI 相同)
标题:使用 JavaScript 创建 ZIP 捆绑包(与 Capgo CLI 相同)注意:以下示例中 version 指的是由 API 使用的捆绑包(版本)名称。
const fs = require('node:fs');const path = require('node:path');const os = require('node:os');const AdmZip = require('adm-zip');const { checksum: getChecksum } = require('@tomasklaen/checksum');
// Exact same implementation as Capgo CLIfunction zipFileUnix(filePath) { const zip = new AdmZip(); zip.addLocalFolder(filePath); return zip.toBuffer();}
async function zipFileWindows(filePath) { console.log('Zipping file windows mode'); const zip = new AdmZip();
const addToZip = (folderPath, zipPath) => { const items = fs.readdirSync(folderPath);
for (const item of items) { const itemPath = path.join(folderPath, item); const stats = fs.statSync(itemPath);
if (stats.isFile()) { const fileContent = fs.readFileSync(itemPath); zip.addFile(path.join(zipPath, item).split(path.sep).join('/'), fileContent); } else if (stats.isDirectory()) { addToZip(itemPath, path.join(zipPath, item)); } } };
addToZip(filePath, ''); return zip.toBuffer();}
// Main zipFile function (exact same logic as CLI)async function zipFile(filePath) { if (os.platform() === 'win32') { return zipFileWindows(filePath); } else { return zipFileUnix(filePath); }}
async function createBundle(inputPath, outputPath, version) { // Create zip using exact same method as Capgo CLI const zipped = await zipFile(inputPath);
// Write to file fs.writeFileSync(outputPath, zipped);
// Calculate checksum using exact same package as CLI const checksum = await getChecksum(zipped, 'sha256');
return { filename: path.basename(outputPath), version: version, size: zipped.length, checksum: checksum };}
// Usageasync function main() { try { const result = await createBundle('./dist', './my-app-1.2.3.zip', '1.2.3'); console.log('Bundle info:', JSON.stringify(result, null, 2)); } catch (error) { console.error('Error creating bundle:', error); }}
main();步骤 3:使用相同的包计算 SHA256 检验和 CLI
标题:步骤 3:使用相同的包计算 SHA256 检验和 CLIconst fs = require('node:fs');const { checksum: getChecksum } = require('@tomasklaen/checksum');
async function calculateChecksum(filePath) { const fileBuffer = fs.readFileSync(filePath); // Use exact same package and method as Capgo CLI const checksum = await getChecksum(fileBuffer, 'sha256'); return checksum;}
// Usageasync function main() { const checksum = await calculateChecksum('./my-app-1.2.3.zip'); console.log('Checksum:', checksum);}
main();步骤 4:上传捆绑包到您的存储
标题:步骤 4:上传捆绑包到您的存储将您的 zip 文件上传到任何可访问的存储:
# Example: Upload to your server via scpscp my-app-1.2.3.zip user@your-server.com:/var/www/bundles/
# Example: Upload to S3 using AWS CLIaws s3 cp my-app-1.2.3.zip s3://your-bucket/bundles/
# Example: Upload via curl to a custom endpointcurl -X POST https://your-storage-api.com/upload \ -H "Authorization: Bearer YOUR_TOKEN" \ -F "file=@my-app-1.2.3.zip"重要:您的捆绑包必须 通过 HTTPS URL(无需身份验证)访问。 Capgo需要从此 URL下载捆绑包。
有效的公共 URL示例:
https://your-storage.com/bundles/my-app-1.2.3.ziphttps://github.com/username/repo/releases/download/v1.2.3/bundle.ziphttps://cdn.jsdelivr.net/gh/username/repo@v1.2.3/dist.zip
步骤 5:将捆绑包注册到Capgo API
步骤 5:将捆绑包注册到Capgo API使用直接API调用将外部捆绑包注册到Capgo:
async function registerWithCapgo(appId, version, bundleUrl, checksum, apiKey) { const fetch = require('node-fetch');
// Create bundle (version) const response = await fetch('https://api.capgo.app/bundle/', { method: 'POST', headers: { 'Content-Type': 'application/json', 'authorization': apiKey }, body: JSON.stringify({ app_id: appId, version: version, external_url: bundleUrl, checksum: checksum }) });
if (!response.ok) { throw new Error(`Failed to create bundle: ${response.statusText}`); }
const data = await response.json(); console.log('Bundle created:', data);
return data;}API参数
API参数| 参数 | 描述 | 必填 |
|---|---|---|
app_id | __CAPGO_KEEP_0__ | 是 |
version | __CAPGO_KEEP_1__ (版本) 语义版本 (例如,“1.2.3”) | 是 |
external_url | 公开可访问 HTTPS URL 可下载的包位置(无需认证) | 是 |
checksum | zip 文件的 SHA256 校验和 | 是 |
__CAPGO_KEEP_12__
Section titled “Bundle Structure Requirements”您的打包压缩包必须遵循这些要求:
- Root Index File: 必须有
index.html在根目录 - Capacitor Integration: 必须调用
notifyAppReady()在您的应用程序中 code - Asset Paths: 为所有资产使用相对路径
Valid Bundle Structure
Section titled “Valid Bundle Structure”bundle.zip├── index.html├── assets/│ ├── app.js│ └── styles.css└── images/完成手动工作流程示例
标题:完成手动工作流程示例一个简单的 Node.js 脚本来压缩、校验和上传到 Capgo:
const fs = require('node:fs');const os = require('node:os');const AdmZip = require('adm-zip');const { checksum: getChecksum } = require('@tomasklaen/checksum');const fetch = require('node-fetch');
async function deployToCapgo() { const APP_ID = 'com.example.app'; const VERSION = '1.2.3'; const BUNDLE_URL = 'https://your-storage.com/bundles/app-1.2.3.zip'; const API_KEY = process.env.CAPGO_API_KEY;
// 1. Create zip (same as Capgo CLI) const zip = new AdmZip(); zip.addLocalFolder('./dist'); const zipped = zip.toBuffer();
// 2. Calculate checksum (same as Capgo CLI) const checksum = await getChecksum(zipped, 'sha256'); console.log('Checksum:', checksum);
// 3. Upload to your storage (replace with your upload logic) // fs.writeFileSync('./bundle.zip', zipped); // ... upload bundle.zip to your storage ...
// 4. Register with Capgo API const response = await fetch('https://api.capgo.app/bundle/', { method: 'POST', headers: { 'Content-Type': 'application/json', 'authorization': API_KEY }, body: JSON.stringify({ app_id: APP_ID, version: VERSION, external_url: BUNDLE_URL, checksum: checksum }) });
if (!response.ok) { throw new Error(`Failed: ${response.statusText}`); }
console.log('Bundle registered with Capgo!');}
deployToCapgo().catch(console.error);安装依赖项:
npm install adm-zip @tomasklaen/checksum node-fetch校验和验证
标题:校验和验证JavaScript 校验和计算(与 Capgo CLI 相同)
标题:JavaScript 校验和计算(与 Capgo CLI 相同)使用相同的包和方法,Capgo CLI 内部使用:
const fs = require('node:fs');const { checksum: getChecksum } = require('@tomasklaen/checksum');
async function calculateChecksum(filePath) { const fileBuffer = fs.readFileSync(filePath); // Use exact same package and method as Capgo CLI const checksum = await getChecksum(fileBuffer, 'sha256'); return checksum;}
// Verify checksum matchesasync function verifyChecksum(filePath, expectedChecksum) { const actualChecksum = await calculateChecksum(filePath); const isValid = actualChecksum === expectedChecksum;
console.log(`File: ${filePath}`); console.log(`Expected: ${expectedChecksum}`); console.log(`Actual: ${actualChecksum}`); console.log(`Valid: ${isValid}`);
return isValid;}
// Usageasync function main() { const bundleChecksum = await calculateChecksum('./my-app-1.2.3.zip'); console.log('SHA256 Checksum:', bundleChecksum);}
main();校验和重要性
校验和重要性部分- 捆绑完整性: 确保捆绑在传输过程中没有被损坏
- API 验证:Capgo 在接受捆绑之前验证校验和
- 插件验证:移动插件在应用更新之前验证校验和
最佳实践
最佳实践部分- 版本管理: 使用 一致的语义版本 存储优化
- : 定期清除未使用的捆绑包版本兼容性
- : 设置合适的最低本机版本要求备份策略
- : 保持关键捆绑包 (版本) 的备份端点
关于端点的部分
EndpointsGET
标题:GEThttps://api.capgo.app/bundle/
获取包信息。每页返回50个包。
查询参数
标题:查询参数app_id: 必填。您的应用IDpage: 可选。分页的页码
响应类型
标题:响应类型interface Bundle { app_id: string bucket_id: string | null checksum: string | null created_at: string | null deleted: boolean external_url: string | null id: number minUpdateVersion: string | null name: string native_packages: Json[] | null owner_org: string r2_path: string | null session_key: string | null storage_provider: string updated_at: string | null user_id: string | null}示例请求
标题:示例请求# Get all bundlescurl -H "authorization: your-api-key" \ "https://api.capgo.app/bundle/?app_id=app_123"
# Get next pagecurl -H "authorization: your-api-key" \ "https://api.capgo.app/bundle/?app_id=app_123&page=1"示例响应
标题为“示例响应”的部分{ "data": [ { "id": 1, "app_id": "app_123", "name": "1.0.0", "checksum": "abc123...", "minUpdateVersion": "1.0.0", "storage_provider": "r2", "created_at": "2024-01-01T00:00:00Z", "updated_at": "2024-01-01T00:00:00Z", "deleted": false, "owner_org": "org_123", "user_id": "user_123" } ]}DELETE
标题为“DELETE”的部分https://api.capgo.app/bundle/
删除一个或所有应用程序的包。请谨慎使用,因为此操作不可逆。
查询参数
标题为“查询参数”的部分删除特定包:
interface BundleDelete { app_id: string version: string}For deleting all bundles:
interface BundleDeleteAll { app_id: string}示例请求
标题:示例请求# Delete specific bundlecurl -X DELETE \ -H "authorization: your-api-key" \ -H "Content-Type: application/json" \ -d '{ "app_id": "app_123", "version": "1.0.0" }' \ https://api.capgo.app/bundle/
# Delete all bundlescurl -X DELETE \ -H "authorization: your-api-key" \ -H "Content-Type: application/json" \ -d '{ "app_id": "app_123" }' \ https://api.capgo.app/bundle/成功响应
标题:成功响应{ "status": "ok"}POST
标题:POSThttps://api.capgo.app/bundle/
创建一个新的带外部 URL 的捆绑包。
请求体
请求体部分interface CreateBundleBody { app_id: string version: string external_url: string // Must be publicly accessible HTTPS URL checksum: string}示例请求
示例请求部分curl -X POST \ -H "authorization: your-api-key" \ -H "Content-Type: application/json" \ -d '{ "app_id": "com.example.app", "version": "1.2.3", "external_url": "https://your-storage.com/bundles/app-1.2.3.zip", "checksum": "a1b2c3d4e5f6789abcdef123456789abcdef123456789abcdef123456789abcd" }' \ https://api.capgo.app/bundle/成功响应
成功响应部分{ "status": "ok"}POST (元数据)
POST (元数据)部分https://api.capgo.app/bundle/metadata
更新捆绑包元数据,如链接和注释信息。
请求体
请求体interface UpdateMetadataBody { app_id: string version_id: number // bundle (version) id link?: string comment?: string}示例请求
示例请求curl -X POST \ -H "authorization: your-api-key" \ -H "Content-Type: application/json" \ -d '{ "app_id": "app_123", "version_id": 456, "link": "https://github.com/myorg/myapp/releases/tag/v1.0.0", "comment": "Fixed critical bug in authentication" }' \ https://api.capgo.app/bundle/metadata成功响应
成功响应{ "status": "success"}https://api.capgo.app/bundle/
将一个捆绑包设置到一个特定的频道。这个操作将一个捆绑包(版本)与一个频道关联起来,用于分发。
interface SetChannelBody { app_id: string version_id: number // bundle (version) id channel_id: number}curl -X PUT \ -H "authorization: your-api-key" \ -H "Content-Type: application/json" \ -d '{ "app_id": "app_123", "version_id": 456, "channel_id": 789 }' \ https://api.capgo.app/bundle/{ "status": "success", "message": "Bundle 1.0.0 set to channel production"}错误处理
标题:错误处理常见错误场景和响应:
// Bundle not found{ "error": "Bundle not found", "status": "KO"}
// Invalid bundle (version) format{ "error": "Invalid version format", "status": "KO"}
// Storage error{ "error": "Failed to delete bundle from storage", "status": "KO"}
// Permission denied{ "error": "Insufficient permissions to manage bundles", "status": "KO"}常见用例
标题:常见用例- 清理旧版本(包)
// Delete outdated beta bundles (versions){ "app_id": "app_123", "version": "1.0.0-beta.1"}- 应用重置
// Remove all bundles to start fresh{ "app_id": "app_123"}存储考虑
标题:存储考虑- 保留策略: 定义保留旧包的时间
- 大小管理: 监控包大小和存储使用情况
- 备份策略: 考虑备份关键包(版本)
- 成本优化: 删除不必要的包以优化存储成本
如果您正在使用 包 为了规划存储和文件处理,连接它与 @capgo/capacitor-data-storage-sqlite 在 @capgo/capacitor-data-storage-sqlite 中的实现细节, 使用 @capgo/capacitor-data-storage-sqlite 在 @capgo/capacitor-data-storage-sqlite 中的原生能力, @capgo/capacitor-file 在 @capgo/capacitor-file 中的实现细节, 使用 @capgo/capacitor-file 在 @capgo/capacitor-file 中的原生能力, @capgo/capacitor-uploader 在 @capgo/capacitor-uploader 中的实现细节,