Lompat ke Konten

Endpoint Channel API

GitHub

Saluran adalah mekanisme inti untuk mengelola pembaruan aplikasi di Capgo. Dalam mode self-hosted, Anda perlu mengimplementasikan endpoint saluran untuk mengelola pengalokasian perangkat, kueri saluran, dan operasi manajemen saluran.

Saluran memungkinkan Anda untuk:

  • Distribusi pembaruan kendali: Tugaskan versi aplikasi yang berbeda kepada kelompok pengguna yang berbeda
  • Pengujian A/B: Uji fitur baru dengan segment pengguna tertentu
  • Peluncuran rolut yang dipersiapkan: Rilis pembaruan secara bertahap untuk mengurangi risiko
  • Pemisahan lingkungan: Pisahkan pembaruan pengembangan, pengujian, dan produksi

Konfigurasi URL endpoint saluran di capacitor.config.json:

{
"plugins": {
"CapacitorUpdater": {
"channelUrl": "https://myserver.com/api/channel_self"
}
}
}

Plugin melakukan berbagai operasi saluran yang perlu dijalankan oleh endpoint Anda:

1. Daftar Saluran yang Kompatibel (Pengajuan GET)

Judul Bagian “1. Daftar Saluran yang Kompatibel (Pengajuan GET)”

Ketika plugin memanggil listChannels(), plugin mengirimkan pengajuan GET untuk mengambil semua saluran yang kompatibel dengan perangkat. Hal ini mengembalikan saluran yang sesuai dengan lingkungan perangkat (dev/prod, emulator/real device) dan memungkinkan akses publik atau pengasasan sendiri.

Format Pengajuan

Salin ke clipboard
// GET /api/channel_self
// Headers:
{
"Content-Type": "application/json"
}
// Query parameters:
interface ListChannelsRequest {
app_id: string
platform: "ios" | "android" | "electron"
is_emulator: boolean
is_prod: boolean
key_id?: string
}

Judul Bagian “Format Respon”

__CAPGO_KEEP_0__
[
{
"id": 1,
"name": "production",
"public": true,
"allow_self_set": false
},
{
"id": 2,
"name": "beta",
"public": false,
"allow_self_set": true
}
]

Balasan termasuk dua flag penting untuk setiap saluran:

  • public: true: Ini adalah saluran standar. Perangkat tidak dapat menetapkan diri sendiri ke saluran ini menggunakan setChannel(). Sebaliknya, jika perangkat menghapus penetapan saluran (menggunakan unsetChannel()), maka perangkat akan secara otomatis menerima update dari saluran publik ini jika saluran tersebut sesuai dengan kondisi perangkat.

  • allow_self_set: true: Ini adalah saluran yang dapat ditempatkan secara eksplisit. Perangkat dapat menetapkan diri sendiri ke saluran ini menggunakan setChannel(). Ini berguna untuk tes beta, tes A/B, atau memungkinkan pengguna untuk memilih masuk ke jalur pembaruan tertentu.

Ketika plugin memanggil getChannel(), maka plugin mengirimkan request PUT untuk memperoleh penugasan saluran perangkat saat ini.

// PUT /api/channel_self
// Headers:
{
"Content-Type": "application/json"
}
// Body:
interface GetChannelRequest {
device_id: string
app_id: string
platform: "ios" | "android" | "electron"
plugin_version: string
version_build: string
version_code: string
version_name: string
is_emulator: boolean
is_prod: boolean
defaultChannel?: string
channel?: string // For newer plugin versions, contains local channel override
}
{
"status": "ok",
"channel": "production",
"allowSet": true,
"message": "",
"error": ""
}

Ketika plugin memanggil setChannel(), maka plugin mengirimkan permintaan POST untuk menetapkan perangkat ke saluran tertentu.

// POST /api/channel_self
interface SetChannelRequest {
device_id: string
app_id: string
channel: string
platform: "ios" | "android" | "electron"
plugin_version: string
version_build: string
version_code: string
version_name: string
is_emulator: boolean
is_prod: boolean
}

Format Respon

Format Tanggapan
{
"status": "ok",
"message": "Device assigned to channel successfully",
"error": ""
}

Kasus Kesalahan

Kasus Kesalahan

Ketika perangkat mencoba menetapkan dirinya ke sebuah saluran publik (saluran dengan public: true), endpoint Anda harus mengembalikan kesalahan:

{
"status": "error",
"error": "public_channel_self_set_not_allowed",
"message": "This channel is public and does not allow device self-assignment. Unset the channel and the device will automatically use the public channel."
}

Ketika perangkat mencoba menetapkan dirinya ke saluran yang tidak memungkinkan penetapan diri:

{
"status": "error",
"error": "channel_self_set_not_allowed",
"message": "This channel does not allow devices to self associate"
}

4. Menghapus Saluran (Menggunakan Metode DELETE)

Bagian berjudul “4. Menghapus Saluran (DELETE Request)”

Ketika plugin memanggil unsetChannel(), maka plugin mengirimkan permintaan DELETE untuk menghapus pengaturan saluran perangkat.

// DELETE /api/channel_self
interface UnsetChannelRequest {
device_id: string
app_id: string
platform: "ios" | "android" | "electron"
plugin_version: string
version_build: string
version_code: string
version_name: string
}

Berikut adalah contoh JavaScript bagaimana mengimplementasikan endpoint saluran:

interface ChannelRequest {
device_id: string
app_id: string
channel?: string
platform: "ios" | "android" | "electron"
plugin_version: string
version_build: string
version_code: string
version_name: string
}
interface ChannelResponse {
status: "ok" | "error"
channel?: string
allowSet?: boolean
message?: string
error?: string
}
export const handler = async (event) => {
const method = event.httpMethod || event.method
const body = JSON.parse(event.body || '{}') as ChannelRequest
const { device_id, app_id, channel, platform } = body
try {
switch (method) {
case 'GET':
return await getDeviceChannel(device_id, app_id)
case 'POST':
return await setDeviceChannel(device_id, app_id, channel!, platform)
case 'DELETE':
return await unsetDeviceChannel(device_id, app_id)
default:
return {
status: "error",
error: "Method not allowed"
}
}
} catch (error) {
return {
status: "error",
error: error.message
}
}
}
async function getDeviceChannel(deviceId: string, appId: string): Promise<ChannelResponse> {
// Query your database for device channel assignment
const assignment = await database.getDeviceChannel(deviceId, appId)
if (assignment) {
return {
status: "ok",
channel: assignment.channel,
allowSet: assignment.allowSelfAssign
}
}
// Return default channel if no assignment found
return {
status: "ok",
channel: "production", // Your default channel
allowSet: true
}
}
async function setDeviceChannel(
deviceId: string,
appId: string,
channel: string,
platform: string
): Promise<ChannelResponse> {
// Validate channel exists and allows self-assignment
const channelConfig = await database.getChannelConfig(channel, appId)
if (!channelConfig) {
return {
status: "error",
error: "Channel not found"
}
}
if (!channelConfig.allowDeviceSelfSet) {
return {
status: "error",
error: "Channel does not allow self-assignment"
}
}
// Check platform restrictions
if (platform === "ios" && !channelConfig.ios) {
return {
status: "error",
error: "Channel not available for iOS"
}
}
if (platform === "android" && !channelConfig.android) {
return {
status: "error",
error: "Channel not available for Android"
}
}
if (platform === "electron" && !channelConfig.electron) {
return {
status: "error",
error: "Channel not available for Electron"
}
}
// Save the assignment
await database.setDeviceChannel(deviceId, appId, channel)
return {
status: "ok",
message: "Device assigned to channel successfully"
}
}
async function unsetDeviceChannel(deviceId: string, appId: string): Promise<ChannelResponse> {
// Remove device channel assignment
await database.removeDeviceChannel(deviceId, appId)
return {
status: "ok",
message: "Device channel assignment removed"
}
}

Sistem saluran Anda harus mendukung opsi pengaturan ini:

interface ChannelConfig {
name: string
appId: string
// Platform targeting
ios: boolean // Allow updates to iOS devices
android: boolean // Allow updates to Android devices
electron: boolean // Allow updates to Electron apps
// Device type restrictions
allow_emulator: boolean // Allow updates on emulator/simulator devices
allow_device: boolean // Allow updates on real/physical devices
// Build type restrictions
allow_dev: boolean // Allow updates on development builds (is_prod=false)
allow_prod: boolean // Allow updates on production builds (is_prod=true)
// Channel assignment
public: boolean // Default channel - devices fall back to this when no override
allowDeviceSelfSet: boolean // Allow devices to self-assign via setChannel()
// Update policies
disableAutoUpdate: "major" | "minor" | "version_number" | "none"
disableAutoUpdateUnderNative: boolean
}

Ketika menampilkan saluran yang kompatibel (permintaan GET), Anda harus memfilter saluran berdasarkan kondisi-kondisi ini:

  1. Pengecekan platform: Saluran harus memungkinkan platform perangkat (ios, android, atau electron)
  2. Pengecekan jenis perangkat:
    • Jika is_emulator=true: Saluran harus memiliki allow_emulator=true
    • Jika is_emulator=false: Channel harus memiliki allow_device=true
  3. Build type check:
    • Jika is_prod=true: Channel harus memiliki allow_prod=true
    • Jika is_prod=false: Channel harus memiliki allow_dev=true
  4. Pemeriksaan visibilitas: Channel harus berupa public=true ATAU allow_device_self_set=true
// Example filtering logic
function getCompatibleChannels(
platform: 'ios' | 'android' | 'electron',
isEmulator: boolean,
isProd: boolean,
channels: ChannelConfig[]
): ChannelConfig[] {
return channels.filter(channel => {
// Platform check
if (!channel[platform]) return false
// Device type check
if (isEmulator && !channel.allow_emulator) return false
if (!isEmulator && !channel.allow_device) return false
// Build type check
if (isProd && !channel.allow_prod) return false
if (!isProd && !channel.allow_dev) return false
// Must be accessible (public or self-assignable)
if (!channel.public && !channel.allowDeviceSelfSet) return false
return true
})
}

Anda perlu menyimpan konfigurasi saluran dan pengaturan perangkat:

-- Channels table
CREATE TABLE channels (
id SERIAL PRIMARY KEY,
name VARCHAR(255) NOT NULL,
app_id VARCHAR(255) NOT NULL,
-- Platform targeting
ios BOOLEAN DEFAULT true,
android BOOLEAN DEFAULT true,
electron BOOLEAN DEFAULT true,
-- Device type restrictions
allow_emulator BOOLEAN DEFAULT true, -- Allow emulator/simulator devices
allow_device BOOLEAN DEFAULT true, -- Allow real/physical devices
-- Build type restrictions
allow_dev BOOLEAN DEFAULT true, -- Allow development builds
allow_prod BOOLEAN DEFAULT true, -- Allow production builds
-- Channel assignment
public BOOLEAN DEFAULT false, -- Default channel (fallback)
allow_device_self_set BOOLEAN DEFAULT false, -- Allow self-assignment
-- Update policies
disable_auto_update VARCHAR(50) DEFAULT 'none',
disable_auto_update_under_native BOOLEAN DEFAULT false,
created_at TIMESTAMP DEFAULT NOW(),
UNIQUE(name, app_id)
);
-- Device channel assignments table
CREATE TABLE device_channels (
id SERIAL PRIMARY KEY,
device_id VARCHAR(255) NOT NULL,
app_id VARCHAR(255) NOT NULL,
channel_name VARCHAR(255) NOT NULL,
assigned_at TIMESTAMP DEFAULT NOW(),
UNIQUE(device_id, app_id)
);

Tangani skenario kesalahan umum:

// Channel not found
{
"status": "error",
"error": "Channel 'beta' not found"
}
// Self-assignment not allowed
{
"status": "error",
"error": "Channel does not allow device self-assignment"
}
// Platform not supported
{
"status": "error",
"error": "Channel not available for this platform"
}
// Invalid request
{
"status": "error",
"error": "Missing required field: device_id"
}
  1. Keamanan: Validasi semua pengaturan saluran terhadap aturan bisnis Anda
  2. Pengelolaan Log : Catat semua operasi saluran untuk auditing dan debugging
  3. Kinerja: Simpan konfigurasi kanal cache untuk mengurangi kueri database
  4. Validasi: Verifikasi keaslian device_id dan app_id
  5. Pengaturan Batas: Implementasikan pengaturan batas untuk mencegah penyalahgunaan

Penugasan saluran bekerja sama dengan Perbarui API Endpoint. Ketika perangkat meminta perbarui, periksa penugasan salurannya untuk menentukan versi mana yang harus disajikan:

async function getUpdateForDevice(deviceId: string, appId: string) {
// Get device's channel assignment
const channelAssignment = await getDeviceChannel(deviceId, appId)
const channel = channelAssignment.channel || 'production'
// Get the version assigned to this channel
const channelVersion = await getChannelVersion(channel, appId)
return {
version: channelVersion.version,
url: channelVersion.url,
checksum: channelVersion.checksum
}
}

Membuat sistem manajemen saluran mandiri yang lengkap yang memberi Anda kendali penuh atas cara bagaimana pembaruan didistribusikan kepada pengguna Anda.

Jika Anda menggunakan Saluran API Endpoint untuk merencanakan routing saluran dan peluncuran tahap demi tahap, hubungkannya dengan Menggunakan @capgo/capacitor-updater untuk kemampuan asli dalam Menggunakan @capgo/capacitor-updater, Saluran untuk detail implementasi dalam Saluran, Saluran untuk detail implementasi dalam Saluran, Saluran untuk detail implementasi di Saluran, dan Solusi Uji Beta untuk alur kerja produk di Solusi Uji Beta.