Canal API Point de terminaison
Les canaux constituent un mécanisme essentiel pour gérer les mises à jour des applications dans Capgo. En mode auto-hébergé, vous devez implémenter des points de terminaison de canal pour gérer les attributions de périphériques, les requêtes de canal et les opérations de gestion des canaux.
Comprendre les chaînes
Section titled “Comprendre les chaînes”Les chaînes vous permettent de :
- Contrôler la distribution des mises à jour : attribuez différentes versions d’application à différents groupes d’utilisateurs
- Tests A/B : testez de nouvelles fonctionnalités avec des segments d’utilisateurs spécifiques
- Déploiements par étapes : déployez progressivement les mises à jour pour minimiser les risques
- Séparation de l’environnement : mises à jour séparées pour le développement, la préparation et la production
##Configuration
Configurez l’URL du point de terminaison du canal dans votre capacitor.config.json :
{ "plugins": { "CapacitorUpdater": { "channelUrl": "https://myserver.com/api/channel_self" } }}Opérations de canal
Section titled “Opérations de canal”Le plugin effectue différentes opérations de canal que votre point de terminaison doit gérer :
1. Liste des chaînes compatibles (demande GET)
Section titled “1. Liste des chaînes compatibles (demande GET)”Lorsque le plugin appelle listChannels(), il envoie une requête GET pour récupérer tous les canaux compatibles avec l’appareil. Cela renvoie des canaux qui correspondent à l’environnement de l’appareil (dev/prod, émulateur/appareil réel) et permettent soit un accès public, soit une auto-attribution.
Format de la demande
Section titled “Format de la demande”// 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}Format de réponse
Section titled “Format de réponse”[ { "id": 1, "name": "production", "public": true, "allow_self_set": false }, { "id": 2, "name": "beta", "public": false, "allow_self_set": true }]Comprendre les types de canaux
Section titled “Comprendre les types de canaux”La réponse comprend deux indicateurs importants pour chaque canal :
-
public: true: Il s’agit d’un canal par défaut. Les appareils ne peuvent pas s’y attribuer automatiquement à l’aide desetChannel(). Au lieu de cela, si un appareil supprime son attribution de canal (à l’aide deunsetChannel()), il recevra automatiquement les mises à jour de ce canal public si cela correspond aux conditions de l’appareil. -
allow_self_set: true: Il s’agit d’un canal auto-attribuable. Les appareils peuvent s’attribuer explicitement à ce canal à l’aide desetChannel(). Ceci est utile pour les tests bêta, les tests A/B ou pour permettre aux utilisateurs de s’inscrire à des pistes de mise à jour spécifiques.
2. Obtenir le canal (demande PUT)
Section titled “2. Obtenir le canal (demande PUT)”Lorsque le plugin appelle getChannel(), il envoie une requête PUT pour récupérer l’attribution de canal actuelle de l’appareil.
Format de la demande
Section titled “Format de la demande”// 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}Format de réponse
Section titled “Format de réponse”{ "status": "ok", "channel": "production", "allowSet": true, "message": "", "error": ""}3. Définir le canal (requête POST)
Section titled “3. Définir le canal (requête POST)”Lorsque le plugin appelle setChannel(), il envoie une requête POST pour attribuer l’appareil à un canal spécifique.
Format de la demande
Section titled “Format de la demande”// POST /api/channel_selfinterface 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 de réponse
Section titled “Format de réponse”{ "status": "ok", "message": "Device assigned to channel successfully", "error": ""}Cas d’erreur
Section titled “Cas d’erreur”Lorsqu’un appareil tente de s’attribuer un canal public (un avec public: true), votre point de terminaison doit renvoyer une erreur :
{ "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."}Lorsqu’un appareil tente de s’attribuer à un canal qui ne permet pas l’auto-affectation :
{ "status": "error", "error": "channel_self_set_not_allowed", "message": "This channel does not allow devices to self associate"}4. Désactiver le canal (demande DELETE)
Section titled “4. Désactiver le canal (demande DELETE)”Lorsque le plugin appelle unsetChannel(), il envoie une requête DELETE pour supprimer l’attribution de canal de l’appareil.
Format de la demande
Section titled “Format de la demande”// DELETE /api/channel_selfinterface UnsetChannelRequest { device_id: string app_id: string platform: "ios" | "android" | "electron" plugin_version: string version_build: string version_code: string version_name: string}Exemple d’implémentation
Section titled “Exemple d’implémentation”Voici un exemple JavaScript de la façon d’implémenter le point de terminaison du canal :
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" }}Configuration du canal
Section titled “Configuration du canal”Votre système de canaux doit prendre en charge ces options de configuration :
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}Logique de filtrage des appareilsLorsque vous répertoriez les chaînes compatibles (requête GET), vous devez filtrer les chaînes en fonction de ces conditions :
Section titled “Logique de filtrage des appareilsLorsque vous répertoriez les chaînes compatibles (requête GET), vous devez filtrer les chaînes en fonction de ces conditions :”- Vérification de la plate-forme : le canal doit autoriser la plate-forme de l’appareil (
ios,androidouelectron) - Vérification du type d’appareil :
- Si
is_emulator=true: le canal doit avoirallow_emulator=true - Si
is_emulator=false: le canal doit avoirallow_device=true
- Si
- Vérification du type de build :
- Si
is_prod=true: le canal doit avoirallow_prod=true - Si
is_prod=false: le canal doit avoirallow_dev=true
- Si
- Contrôle de visibilité : le canal doit être soit
public=trueOUallow_device_self_set=true.
// Example filtering logicfunction 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 })}Exemple de schéma de base de données
Section titled “Exemple de schéma de base de données”Vous devrez stocker les configurations de canaux et les attributions de périphériques :
-- Channels tableCREATE 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 tableCREATE 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));Gestion des erreurs
Section titled “Gestion des erreurs”Gérez les scénarios d’erreur courants :
// 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"}## meilleures pratiques
- Sécurité : validez toutes les attributions de canaux par rapport à vos règles métier
- Logging : enregistrez toutes les opérations du canal pour l’audit et le débogage.
- Performances : configurations du canal de cache pour réduire les requêtes de base de données
- Validation : Vérifiez l’authenticité de device_id et app_id
- Limitation du débit : mettez en œuvre une limitation du débit pour éviter les abus
Intégration avec les mises à jour
Section titled “Intégration avec les mises à jour”Les attributions de canaux fonctionnent avec votre Mettre à jour API Endpoint. Lorsqu’un appareil demande une mise à jour, vérifiez son attribution de canal pour déterminer la version à diffuser :
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 }}Cela crée un système complet de gestion de canaux auto-hébergé qui vous donne un contrôle total sur la manière dont les mises à jour sont distribuées à vos utilisateurs.