@camstack/addon-cloudflare-turn 0.1.13 → 0.1.15

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.
@@ -1,125 +1,86 @@
1
1
  "use strict";
2
- var __defProp = Object.defineProperty;
3
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
- var __getOwnPropNames = Object.getOwnPropertyNames;
5
- var __hasOwnProp = Object.prototype.hasOwnProperty;
6
- var __export = (target, all) => {
7
- for (var name in all)
8
- __defProp(target, name, { get: all[name], enumerable: true });
9
- };
10
- var __copyProps = (to, from, except, desc) => {
11
- if (from && typeof from === "object" || typeof from === "function") {
12
- for (let key of __getOwnPropNames(from))
13
- if (!__hasOwnProp.call(to, key) && key !== except)
14
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
- }
16
- return to;
17
- };
18
- var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
-
20
- // src/cloudflare-turn.addon.ts
21
- var cloudflare_turn_addon_exports = {};
22
- __export(cloudflare_turn_addon_exports, {
23
- CloudflareTurnAddon: () => CloudflareTurnAddon
24
- });
25
- module.exports = __toCommonJS(cloudflare_turn_addon_exports);
26
-
27
- // src/cloudflare-turn.ts
28
- var CloudflareTurnService = class {
29
- constructor(config, logger) {
30
- this.config = config;
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const types = require("@camstack/types");
4
+ class CloudflareTurnService {
5
+ constructor(_config, logger) {
31
6
  this.logger = logger;
32
7
  }
33
- /** Get temporary TURN credentials from Cloudflare API */
34
- async getCredentials() {
35
- this.logger.debug("Fetching TURN credentials from Cloudflare");
36
- return {
37
- urls: [
38
- "turn:turn.cloudflare.com:3478?transport=udp",
39
- "turn:turn.cloudflare.com:3478?transport=tcp"
40
- ],
41
- username: "temp-user",
42
- credential: "temp-credential",
43
- ttl: 86400
44
- };
8
+ id = "cloudflare-turn";
9
+ name = "Cloudflare TURN";
10
+ /**
11
+ * Return the current TURN/STUN server list with credentials.
12
+ * Implements `turn-provider` capability.
13
+ */
14
+ getTurnServers() {
15
+ this.logger.debug("Fetching TURN servers from Cloudflare");
16
+ return [
17
+ {
18
+ urls: [
19
+ "turn:turn.cloudflare.com:3478?transport=udp",
20
+ "turn:turn.cloudflare.com:3478?transport=tcp"
21
+ ],
22
+ username: "temp-user",
23
+ credential: "temp-credential"
24
+ }
25
+ ];
45
26
  }
46
- };
47
-
48
- // src/cloudflare-turn.addon.ts
49
- var CloudflareTurnAddon = class {
50
- manifest = {
51
- id: "cloudflare-turn",
52
- name: "Cloudflare TURN Relay",
53
- version: "1.0.0",
54
- capabilities: ["turn-provider"]
55
- };
27
+ }
28
+ class CloudflareTurnAddon extends types.BaseAddon {
56
29
  service = null;
57
- currentConfig = {
58
- apiToken: "",
59
- accountId: ""
60
- };
61
- async initialize(context) {
62
- this.currentConfig = {
63
- apiToken: context.addonConfig.apiToken ?? this.currentConfig.apiToken,
64
- accountId: context.addonConfig.accountId ?? this.currentConfig.accountId
65
- };
66
- this.service = new CloudflareTurnService(this.currentConfig, context.logger);
67
- context.logger.info("Cloudflare TURN initialized");
30
+ constructor() {
31
+ super({ apiToken: "", accountId: "" });
32
+ }
33
+ async onInitialize() {
34
+ this.service = new CloudflareTurnService(this.config, this.ctx.logger);
35
+ this.ctx.logger.info("Cloudflare TURN initialized");
36
+ return [{ capability: types.turnProviderCapability, provider: this.service }];
68
37
  }
69
- async shutdown() {
38
+ async onShutdown() {
70
39
  this.service = null;
71
40
  }
72
41
  getService() {
73
42
  if (!this.service) throw new Error("Cloudflare TURN not initialized");
74
43
  return this.service;
75
44
  }
76
- getCapabilityProvider(name) {
77
- if (name === "turn-provider" && this.service) {
78
- return this.service;
79
- }
80
- return null;
81
- }
82
- getConfigSchema() {
83
- return {
45
+ globalSettingsSchema() {
46
+ return this.schema({
84
47
  sections: [
85
48
  {
86
49
  id: "credentials",
87
50
  title: "Cloudflare Credentials",
88
- description: "API credentials for fetching TURN relay tokens from Cloudflare.",
51
+ description: "API credentials for fetching TURN relay tokens from Cloudflare Realtime.",
52
+ immediate: true,
89
53
  fields: [
90
54
  {
55
+ type: "info",
56
+ key: "turnHelp",
57
+ label: "How to get the Account ID + API Token",
58
+ format: "html",
59
+ content: '<ul><li><strong>Account ID</strong> — open <a href="https://dash.cloudflare.com/">dash.cloudflare.com</a>, pick your account; the URL shows <code>dash.cloudflare.com/&lt;accountId&gt;/...</code>. Copy that hex id.</li><li><strong>TURN Token</strong> — go to <a href="https://dash.cloudflare.com/?to=/:account/calls">Realtime → TURN</a>, click <strong>Create TURN App</strong>, copy the generated <em>Token ID</em> and <em>API Token</em>.</li><li>Paste the <em>API Token</em> below (the Token ID is the Account ID variant for TURN apps — Cloudflare uses both naming conventions across their dashboard).</li><li>Permissions auto-set when you create a Realtime app — no manual scope picker needed.</li></ul>',
60
+ variant: "info"
61
+ },
62
+ this.field({
91
63
  type: "text",
92
64
  key: "accountId",
93
65
  label: "Account ID",
94
- description: "Your Cloudflare account ID (found in the Cloudflare dashboard URL)",
66
+ description: "Your Cloudflare account ID (the hex slug in the dashboard URL).",
95
67
  placeholder: "a1b2c3d4e5f6...",
96
68
  required: true
97
- },
98
- {
69
+ }),
70
+ this.field({
99
71
  type: "password",
100
72
  key: "apiToken",
101
73
  label: "API Token",
102
- description: "Cloudflare API token with Calls: Read permission",
74
+ description: "Cloudflare Realtime TURN app token (Calls: Read).",
103
75
  showToggle: true,
104
76
  required: true
105
- }
77
+ })
106
78
  ]
107
79
  }
108
80
  ]
109
- };
110
- }
111
- getConfig() {
112
- return { accountId: this.currentConfig.accountId, apiToken: this.currentConfig.apiToken };
113
- }
114
- async onConfigChange(config) {
115
- this.currentConfig = {
116
- apiToken: config.apiToken ?? this.currentConfig.apiToken,
117
- accountId: config.accountId ?? this.currentConfig.accountId
118
- };
81
+ });
119
82
  }
120
- };
121
- // Annotate the CommonJS export names for ESM import in node:
122
- 0 && (module.exports = {
123
- CloudflareTurnAddon
124
- });
125
- //# sourceMappingURL=cloudflare-turn.addon.js.map
83
+ }
84
+ exports.CloudflareTurnAddon = CloudflareTurnAddon;
85
+ exports.CloudflareTurnService = CloudflareTurnService;
86
+ //# sourceMappingURL=cloudflare-turn.addon.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/cloudflare-turn.addon.ts","../src/cloudflare-turn.ts"],"sourcesContent":["import type {\n ICamstackAddon, AddonManifest, AddonContext,\n IConfigurable, ConfigUISchema, CapabilityProviderMap,\n} from '@camstack/types'\nimport { CloudflareTurnService } from './cloudflare-turn'\nimport type { CloudflareTurnConfig } from './cloudflare-turn'\n\nexport class CloudflareTurnAddon implements ICamstackAddon, IConfigurable {\n readonly manifest: AddonManifest = {\n id: 'cloudflare-turn',\n name: 'Cloudflare TURN Relay',\n version: '1.0.0',\n capabilities: ['turn-provider'],\n }\n\n private service: CloudflareTurnService | null = null\n private currentConfig: CloudflareTurnConfig = {\n apiToken: '',\n accountId: '',\n }\n\n async initialize(context: AddonContext): Promise<void> {\n this.currentConfig = {\n apiToken: (context.addonConfig.apiToken as string) ?? this.currentConfig.apiToken,\n accountId: (context.addonConfig.accountId as string) ?? this.currentConfig.accountId,\n }\n this.service = new CloudflareTurnService(this.currentConfig, context.logger)\n context.logger.info('Cloudflare TURN initialized')\n }\n\n async shutdown(): Promise<void> {\n this.service = null\n }\n\n getService(): CloudflareTurnService {\n if (!this.service) throw new Error('Cloudflare TURN not initialized')\n return this.service\n }\n\n getCapabilityProvider<K extends keyof CapabilityProviderMap>(\n name: K,\n ): CapabilityProviderMap[K] | null {\n if (name === 'turn-provider' as string && this.service) {\n return this.service as unknown as CapabilityProviderMap[K]\n }\n return null\n }\n\n getConfigSchema(): ConfigUISchema {\n return {\n sections: [\n {\n id: 'credentials',\n title: 'Cloudflare Credentials',\n description: 'API credentials for fetching TURN relay tokens from Cloudflare.',\n fields: [\n {\n type: 'text',\n key: 'accountId',\n label: 'Account ID',\n description: 'Your Cloudflare account ID (found in the Cloudflare dashboard URL)',\n placeholder: 'a1b2c3d4e5f6...',\n required: true,\n },\n {\n type: 'password',\n key: 'apiToken',\n label: 'API Token',\n description: 'Cloudflare API token with Calls: Read permission',\n showToggle: true,\n required: true,\n },\n ],\n },\n ],\n }\n }\n\n getConfig(): Record<string, unknown> {\n return { accountId: this.currentConfig.accountId, apiToken: this.currentConfig.apiToken }\n }\n\n async onConfigChange(config: Record<string, unknown>): Promise<void> {\n this.currentConfig = {\n apiToken: (config.apiToken as string) ?? this.currentConfig.apiToken,\n accountId: (config.accountId as string) ?? this.currentConfig.accountId,\n }\n }\n}\n","import type { IScopedLogger } from '@camstack/types'\n\nexport interface CloudflareTurnConfig {\n readonly apiToken: string\n readonly accountId: string\n}\n\nexport interface TurnCredentials {\n readonly urls: readonly string[]\n readonly username: string\n readonly credential: string\n readonly ttl: number\n}\n\nexport class CloudflareTurnService {\n constructor(\n private readonly config: CloudflareTurnConfig,\n private readonly logger: IScopedLogger,\n ) {}\n\n /** Get temporary TURN credentials from Cloudflare API */\n async getCredentials(): Promise<TurnCredentials> {\n this.logger.debug('Fetching TURN credentials from Cloudflare')\n\n return {\n urls: [\n 'turn:turn.cloudflare.com:3478?transport=udp',\n 'turn:turn.cloudflare.com:3478?transport=tcp',\n ],\n username: 'temp-user',\n credential: 'temp-credential',\n ttl: 86400,\n }\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;;;ACcO,IAAM,wBAAN,MAA4B;AAAA,EACjC,YACmB,QACA,QACjB;AAFiB;AACA;AAAA,EAChB;AAAA;AAAA,EAGH,MAAM,iBAA2C;AAC/C,SAAK,OAAO,MAAM,2CAA2C;AAE7D,WAAO;AAAA,MACL,MAAM;AAAA,QACJ;AAAA,QACA;AAAA,MACF;AAAA,MACA,UAAU;AAAA,MACV,YAAY;AAAA,MACZ,KAAK;AAAA,IACP;AAAA,EACF;AACF;;;AD3BO,IAAM,sBAAN,MAAmE;AAAA,EAC/D,WAA0B;AAAA,IACjC,IAAI;AAAA,IACJ,MAAM;AAAA,IACN,SAAS;AAAA,IACT,cAAc,CAAC,eAAe;AAAA,EAChC;AAAA,EAEQ,UAAwC;AAAA,EACxC,gBAAsC;AAAA,IAC5C,UAAU;AAAA,IACV,WAAW;AAAA,EACb;AAAA,EAEA,MAAM,WAAW,SAAsC;AACrD,SAAK,gBAAgB;AAAA,MACnB,UAAW,QAAQ,YAAY,YAAuB,KAAK,cAAc;AAAA,MACzE,WAAY,QAAQ,YAAY,aAAwB,KAAK,cAAc;AAAA,IAC7E;AACA,SAAK,UAAU,IAAI,sBAAsB,KAAK,eAAe,QAAQ,MAAM;AAC3E,YAAQ,OAAO,KAAK,6BAA6B;AAAA,EACnD;AAAA,EAEA,MAAM,WAA0B;AAC9B,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,aAAoC;AAClC,QAAI,CAAC,KAAK,QAAS,OAAM,IAAI,MAAM,iCAAiC;AACpE,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,sBACE,MACiC;AACjC,QAAI,SAAS,mBAA6B,KAAK,SAAS;AACtD,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EAEA,kBAAkC;AAChC,WAAO;AAAA,MACL,UAAU;AAAA,QACR;AAAA,UACE,IAAI;AAAA,UACJ,OAAO;AAAA,UACP,aAAa;AAAA,UACb,QAAQ;AAAA,YACN;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,aAAa;AAAA,cACb,UAAU;AAAA,YACZ;AAAA,YACA;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,YAAY;AAAA,cACZ,UAAU;AAAA,YACZ;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,YAAqC;AACnC,WAAO,EAAE,WAAW,KAAK,cAAc,WAAW,UAAU,KAAK,cAAc,SAAS;AAAA,EAC1F;AAAA,EAEA,MAAM,eAAe,QAAgD;AACnE,SAAK,gBAAgB;AAAA,MACnB,UAAW,OAAO,YAAuB,KAAK,cAAc;AAAA,MAC5D,WAAY,OAAO,aAAwB,KAAK,cAAc;AAAA,IAChE;AAAA,EACF;AACF;","names":[]}
1
+ {"version":3,"file":"cloudflare-turn.addon.js","sources":["../src/cloudflare-turn.ts","../src/cloudflare-turn.addon.ts"],"sourcesContent":["import type { IScopedLogger } from '@camstack/types'\n\nexport interface CloudflareTurnConfig {\n readonly apiToken: string\n readonly accountId: string\n}\n\ntype TurnServer = { urls: string | string[]; username?: string; credential?: string }\n\n/**\n * Cloudflare TURN/STUN provider.\n *\n * Implements the `turn-provider` capability (system collection). Called\n * by the `webrtc` capability implementations when building ICE server\n * lists for a new peer connection. Each call SHOULD fetch fresh\n * short-lived credentials — currently stubbed with static values until\n * the Cloudflare Calls API integration is wired.\n */\nexport class CloudflareTurnService {\n readonly id = 'cloudflare-turn'\n readonly name = 'Cloudflare TURN'\n\n constructor(\n _config: CloudflareTurnConfig,\n private readonly logger: IScopedLogger,\n ) {}\n\n /**\n * Return the current TURN/STUN server list with credentials.\n * Implements `turn-provider` capability.\n */\n getTurnServers(): readonly TurnServer[] {\n this.logger.debug('Fetching TURN servers from Cloudflare')\n // TODO: implement real Cloudflare TURN integration.\n // Real integration requires a Cloudflare Calls API call using this.apiToken/accountId.\n return [\n {\n urls: [\n 'turn:turn.cloudflare.com:3478?transport=udp',\n 'turn:turn.cloudflare.com:3478?transport=tcp',\n ],\n username: 'temp-user',\n credential: 'temp-credential',\n },\n ]\n }\n}\n","import type { ProviderRegistration } from '@camstack/types'\nimport { BaseAddon, turnProviderCapability } from '@camstack/types'\nimport { CloudflareTurnService } from './cloudflare-turn'\nimport type { CloudflareTurnConfig } from './cloudflare-turn'\n\n/**\n * Settings redesign Phase 3: cloudflare-turn is node-level credentials\n * storage. Implements `getGlobalSettings`.\n */\nexport class CloudflareTurnAddon extends BaseAddon<CloudflareTurnConfig> {\n private service: CloudflareTurnService | null = null\n\n constructor() {\n super({ apiToken: '', accountId: '' })\n }\n\n protected async onInitialize(): Promise<ProviderRegistration[]> {\n this.service = new CloudflareTurnService(this.config, this.ctx.logger)\n this.ctx.logger.info('Cloudflare TURN initialized')\n return [{ capability: turnProviderCapability, provider: this.service }]\n }\n\n protected async onShutdown(): Promise<void> {\n this.service = null\n }\n\n getService(): CloudflareTurnService {\n if (!this.service) throw new Error('Cloudflare TURN not initialized')\n return this.service\n }\n\n protected globalSettingsSchema() {\n return this.schema({\n sections: [\n {\n id: 'credentials',\n title: 'Cloudflare Credentials',\n description: 'API credentials for fetching TURN relay tokens from Cloudflare Realtime.',\n immediate: true,\n fields: [\n {\n type: 'info' as const,\n key: 'turnHelp',\n label: 'How to get the Account ID + API Token',\n format: 'html' as const,\n content:\n '<ul>' +\n '<li><strong>Account ID</strong> — open ' +\n '<a href=\"https://dash.cloudflare.com/\">dash.cloudflare.com</a>, ' +\n 'pick your account; the URL shows ' +\n '<code>dash.cloudflare.com/&lt;accountId&gt;/...</code>. ' +\n 'Copy that hex id.</li>' +\n '<li><strong>TURN Token</strong> go to ' +\n '<a href=\"https://dash.cloudflare.com/?to=/:account/calls\">' +\n 'Realtime → TURN</a>, click <strong>Create TURN App</strong>, ' +\n 'copy the generated <em>Token ID</em> and <em>API Token</em>.</li>' +\n '<li>Paste the <em>API Token</em> below (the Token ID is the ' +\n 'Account ID variant for TURN apps Cloudflare uses both naming ' +\n 'conventions across their dashboard).</li>' +\n '<li>Permissions auto-set when you create a Realtime app ' +\n 'no manual scope picker needed.</li>' +\n '</ul>',\n variant: 'info' as const,\n },\n this.field({\n type: 'text',\n key: 'accountId',\n label: 'Account ID',\n description: 'Your Cloudflare account ID (the hex slug in the dashboard URL).',\n placeholder: 'a1b2c3d4e5f6...',\n required: true,\n }),\n this.field({\n type: 'password',\n key: 'apiToken',\n label: 'API Token',\n description: 'Cloudflare Realtime TURN app token (Calls: Read).',\n showToggle: true,\n required: true,\n }),\n ],\n },\n ],\n })\n }\n}\n"],"names":["BaseAddon","turnProviderCapability"],"mappings":";;;AAkBO,MAAM,sBAAsB;AAAA,EAIjC,YACE,SACiB,QACjB;AADiB,SAAA,SAAA;AAAA,EAChB;AAAA,EANM,KAAK;AAAA,EACL,OAAO;AAAA;AAAA;AAAA;AAAA;AAAA,EAWhB,iBAAwC;AACtC,SAAK,OAAO,MAAM,uCAAuC;AAGzD,WAAO;AAAA,MACL;AAAA,QACE,MAAM;AAAA,UACJ;AAAA,UACA;AAAA,QAAA;AAAA,QAEF,UAAU;AAAA,QACV,YAAY;AAAA,MAAA;AAAA,IACd;AAAA,EAEJ;AACF;ACrCO,MAAM,4BAA4BA,MAAAA,UAAgC;AAAA,EAC/D,UAAwC;AAAA,EAEhD,cAAc;AACZ,UAAM,EAAE,UAAU,IAAI,WAAW,IAAI;AAAA,EACvC;AAAA,EAEA,MAAgB,eAAgD;AAC9D,SAAK,UAAU,IAAI,sBAAsB,KAAK,QAAQ,KAAK,IAAI,MAAM;AACrE,SAAK,IAAI,OAAO,KAAK,6BAA6B;AAClD,WAAO,CAAC,EAAE,YAAYC,MAAAA,wBAAwB,UAAU,KAAK,SAAS;AAAA,EACxE;AAAA,EAEA,MAAgB,aAA4B;AAC1C,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,aAAoC;AAClC,QAAI,CAAC,KAAK,QAAS,OAAM,IAAI,MAAM,iCAAiC;AACpE,WAAO,KAAK;AAAA,EACd;AAAA,EAEU,uBAAuB;AAC/B,WAAO,KAAK,OAAO;AAAA,MACjB,UAAU;AAAA,QACR;AAAA,UACE,IAAI;AAAA,UACJ,OAAO;AAAA,UACP,aAAa;AAAA,UACb,WAAW;AAAA,UACX,QAAQ;AAAA,YACN;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,QAAQ;AAAA,cACR,SACE;AAAA,cAgBF,SAAS;AAAA,YAAA;AAAA,YAEX,KAAK,MAAM;AAAA,cACT,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,aAAa;AAAA,cACb,UAAU;AAAA,YAAA,CACX;AAAA,YACD,KAAK,MAAM;AAAA,cACT,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,YAAY;AAAA,cACZ,UAAU;AAAA,YAAA,CACX;AAAA,UAAA;AAAA,QACH;AAAA,MACF;AAAA,IACF,CACD;AAAA,EACH;AACF;;;"}
@@ -1,7 +1,86 @@
1
- import {
2
- CloudflareTurnAddon
3
- } from "./chunk-V6KRQ44N.mjs";
1
+ import { BaseAddon, turnProviderCapability } from "@camstack/types";
2
+ class CloudflareTurnService {
3
+ constructor(_config, logger) {
4
+ this.logger = logger;
5
+ }
6
+ id = "cloudflare-turn";
7
+ name = "Cloudflare TURN";
8
+ /**
9
+ * Return the current TURN/STUN server list with credentials.
10
+ * Implements `turn-provider` capability.
11
+ */
12
+ getTurnServers() {
13
+ this.logger.debug("Fetching TURN servers from Cloudflare");
14
+ return [
15
+ {
16
+ urls: [
17
+ "turn:turn.cloudflare.com:3478?transport=udp",
18
+ "turn:turn.cloudflare.com:3478?transport=tcp"
19
+ ],
20
+ username: "temp-user",
21
+ credential: "temp-credential"
22
+ }
23
+ ];
24
+ }
25
+ }
26
+ class CloudflareTurnAddon extends BaseAddon {
27
+ service = null;
28
+ constructor() {
29
+ super({ apiToken: "", accountId: "" });
30
+ }
31
+ async onInitialize() {
32
+ this.service = new CloudflareTurnService(this.config, this.ctx.logger);
33
+ this.ctx.logger.info("Cloudflare TURN initialized");
34
+ return [{ capability: turnProviderCapability, provider: this.service }];
35
+ }
36
+ async onShutdown() {
37
+ this.service = null;
38
+ }
39
+ getService() {
40
+ if (!this.service) throw new Error("Cloudflare TURN not initialized");
41
+ return this.service;
42
+ }
43
+ globalSettingsSchema() {
44
+ return this.schema({
45
+ sections: [
46
+ {
47
+ id: "credentials",
48
+ title: "Cloudflare Credentials",
49
+ description: "API credentials for fetching TURN relay tokens from Cloudflare Realtime.",
50
+ immediate: true,
51
+ fields: [
52
+ {
53
+ type: "info",
54
+ key: "turnHelp",
55
+ label: "How to get the Account ID + API Token",
56
+ format: "html",
57
+ content: '<ul><li><strong>Account ID</strong> — open <a href="https://dash.cloudflare.com/">dash.cloudflare.com</a>, pick your account; the URL shows <code>dash.cloudflare.com/&lt;accountId&gt;/...</code>. Copy that hex id.</li><li><strong>TURN Token</strong> — go to <a href="https://dash.cloudflare.com/?to=/:account/calls">Realtime → TURN</a>, click <strong>Create TURN App</strong>, copy the generated <em>Token ID</em> and <em>API Token</em>.</li><li>Paste the <em>API Token</em> below (the Token ID is the Account ID variant for TURN apps — Cloudflare uses both naming conventions across their dashboard).</li><li>Permissions auto-set when you create a Realtime app — no manual scope picker needed.</li></ul>',
58
+ variant: "info"
59
+ },
60
+ this.field({
61
+ type: "text",
62
+ key: "accountId",
63
+ label: "Account ID",
64
+ description: "Your Cloudflare account ID (the hex slug in the dashboard URL).",
65
+ placeholder: "a1b2c3d4e5f6...",
66
+ required: true
67
+ }),
68
+ this.field({
69
+ type: "password",
70
+ key: "apiToken",
71
+ label: "API Token",
72
+ description: "Cloudflare Realtime TURN app token (Calls: Read).",
73
+ showToggle: true,
74
+ required: true
75
+ })
76
+ ]
77
+ }
78
+ ]
79
+ });
80
+ }
81
+ }
4
82
  export {
83
+ CloudflareTurnService as C,
5
84
  CloudflareTurnAddon
6
85
  };
7
- //# sourceMappingURL=cloudflare-turn.addon.mjs.map
86
+ //# sourceMappingURL=cloudflare-turn.addon.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
1
+ {"version":3,"file":"cloudflare-turn.addon.mjs","sources":["../src/cloudflare-turn.ts","../src/cloudflare-turn.addon.ts"],"sourcesContent":["import type { IScopedLogger } from '@camstack/types'\n\nexport interface CloudflareTurnConfig {\n readonly apiToken: string\n readonly accountId: string\n}\n\ntype TurnServer = { urls: string | string[]; username?: string; credential?: string }\n\n/**\n * Cloudflare TURN/STUN provider.\n *\n * Implements the `turn-provider` capability (system collection). Called\n * by the `webrtc` capability implementations when building ICE server\n * lists for a new peer connection. Each call SHOULD fetch fresh\n * short-lived credentials — currently stubbed with static values until\n * the Cloudflare Calls API integration is wired.\n */\nexport class CloudflareTurnService {\n readonly id = 'cloudflare-turn'\n readonly name = 'Cloudflare TURN'\n\n constructor(\n _config: CloudflareTurnConfig,\n private readonly logger: IScopedLogger,\n ) {}\n\n /**\n * Return the current TURN/STUN server list with credentials.\n * Implements `turn-provider` capability.\n */\n getTurnServers(): readonly TurnServer[] {\n this.logger.debug('Fetching TURN servers from Cloudflare')\n // TODO: implement real Cloudflare TURN integration.\n // Real integration requires a Cloudflare Calls API call using this.apiToken/accountId.\n return [\n {\n urls: [\n 'turn:turn.cloudflare.com:3478?transport=udp',\n 'turn:turn.cloudflare.com:3478?transport=tcp',\n ],\n username: 'temp-user',\n credential: 'temp-credential',\n },\n ]\n }\n}\n","import type { ProviderRegistration } from '@camstack/types'\nimport { BaseAddon, turnProviderCapability } from '@camstack/types'\nimport { CloudflareTurnService } from './cloudflare-turn'\nimport type { CloudflareTurnConfig } from './cloudflare-turn'\n\n/**\n * Settings redesign Phase 3: cloudflare-turn is node-level credentials\n * storage. Implements `getGlobalSettings`.\n */\nexport class CloudflareTurnAddon extends BaseAddon<CloudflareTurnConfig> {\n private service: CloudflareTurnService | null = null\n\n constructor() {\n super({ apiToken: '', accountId: '' })\n }\n\n protected async onInitialize(): Promise<ProviderRegistration[]> {\n this.service = new CloudflareTurnService(this.config, this.ctx.logger)\n this.ctx.logger.info('Cloudflare TURN initialized')\n return [{ capability: turnProviderCapability, provider: this.service }]\n }\n\n protected async onShutdown(): Promise<void> {\n this.service = null\n }\n\n getService(): CloudflareTurnService {\n if (!this.service) throw new Error('Cloudflare TURN not initialized')\n return this.service\n }\n\n protected globalSettingsSchema() {\n return this.schema({\n sections: [\n {\n id: 'credentials',\n title: 'Cloudflare Credentials',\n description: 'API credentials for fetching TURN relay tokens from Cloudflare Realtime.',\n immediate: true,\n fields: [\n {\n type: 'info' as const,\n key: 'turnHelp',\n label: 'How to get the Account ID + API Token',\n format: 'html' as const,\n content:\n '<ul>' +\n '<li><strong>Account ID</strong> — open ' +\n '<a href=\"https://dash.cloudflare.com/\">dash.cloudflare.com</a>, ' +\n 'pick your account; the URL shows ' +\n '<code>dash.cloudflare.com/&lt;accountId&gt;/...</code>. ' +\n 'Copy that hex id.</li>' +\n '<li><strong>TURN Token</strong> — go to ' +\n '<a href=\"https://dash.cloudflare.com/?to=/:account/calls\">' +\n 'Realtime → TURN</a>, click <strong>Create TURN App</strong>, ' +\n 'copy the generated <em>Token ID</em> and <em>API Token</em>.</li>' +\n '<li>Paste the <em>API Token</em> below (the Token ID is the ' +\n 'Account ID variant for TURN apps — Cloudflare uses both naming ' +\n 'conventions across their dashboard).</li>' +\n '<li>Permissions auto-set when you create a Realtime app — ' +\n 'no manual scope picker needed.</li>' +\n '</ul>',\n variant: 'info' as const,\n },\n this.field({\n type: 'text',\n key: 'accountId',\n label: 'Account ID',\n description: 'Your Cloudflare account ID (the hex slug in the dashboard URL).',\n placeholder: 'a1b2c3d4e5f6...',\n required: true,\n }),\n this.field({\n type: 'password',\n key: 'apiToken',\n label: 'API Token',\n description: 'Cloudflare Realtime TURN app token (Calls: Read).',\n showToggle: true,\n required: true,\n }),\n ],\n },\n ],\n })\n }\n}\n"],"names":[],"mappings":";AAkBO,MAAM,sBAAsB;AAAA,EAIjC,YACE,SACiB,QACjB;AADiB,SAAA,SAAA;AAAA,EAChB;AAAA,EANM,KAAK;AAAA,EACL,OAAO;AAAA;AAAA;AAAA;AAAA;AAAA,EAWhB,iBAAwC;AACtC,SAAK,OAAO,MAAM,uCAAuC;AAGzD,WAAO;AAAA,MACL;AAAA,QACE,MAAM;AAAA,UACJ;AAAA,UACA;AAAA,QAAA;AAAA,QAEF,UAAU;AAAA,QACV,YAAY;AAAA,MAAA;AAAA,IACd;AAAA,EAEJ;AACF;ACrCO,MAAM,4BAA4B,UAAgC;AAAA,EAC/D,UAAwC;AAAA,EAEhD,cAAc;AACZ,UAAM,EAAE,UAAU,IAAI,WAAW,IAAI;AAAA,EACvC;AAAA,EAEA,MAAgB,eAAgD;AAC9D,SAAK,UAAU,IAAI,sBAAsB,KAAK,QAAQ,KAAK,IAAI,MAAM;AACrE,SAAK,IAAI,OAAO,KAAK,6BAA6B;AAClD,WAAO,CAAC,EAAE,YAAY,wBAAwB,UAAU,KAAK,SAAS;AAAA,EACxE;AAAA,EAEA,MAAgB,aAA4B;AAC1C,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,aAAoC;AAClC,QAAI,CAAC,KAAK,QAAS,OAAM,IAAI,MAAM,iCAAiC;AACpE,WAAO,KAAK;AAAA,EACd;AAAA,EAEU,uBAAuB;AAC/B,WAAO,KAAK,OAAO;AAAA,MACjB,UAAU;AAAA,QACR;AAAA,UACE,IAAI;AAAA,UACJ,OAAO;AAAA,UACP,aAAa;AAAA,UACb,WAAW;AAAA,UACX,QAAQ;AAAA,YACN;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,QAAQ;AAAA,cACR,SACE;AAAA,cAgBF,SAAS;AAAA,YAAA;AAAA,YAEX,KAAK,MAAM;AAAA,cACT,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,aAAa;AAAA,cACb,UAAU;AAAA,YAAA,CACX;AAAA,YACD,KAAK,MAAM;AAAA,cACT,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,YAAY;AAAA,cACZ,UAAU;AAAA,YAAA,CACX;AAAA,UAAA;AAAA,QACH;AAAA,MACF;AAAA,IACF,CACD;AAAA,EACH;AACF;"}
package/dist/index.js CHANGED
@@ -1,127 +1,6 @@
1
1
  "use strict";
2
- var __defProp = Object.defineProperty;
3
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
- var __getOwnPropNames = Object.getOwnPropertyNames;
5
- var __hasOwnProp = Object.prototype.hasOwnProperty;
6
- var __export = (target, all) => {
7
- for (var name in all)
8
- __defProp(target, name, { get: all[name], enumerable: true });
9
- };
10
- var __copyProps = (to, from, except, desc) => {
11
- if (from && typeof from === "object" || typeof from === "function") {
12
- for (let key of __getOwnPropNames(from))
13
- if (!__hasOwnProp.call(to, key) && key !== except)
14
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
15
- }
16
- return to;
17
- };
18
- var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
19
-
20
- // src/index.ts
21
- var index_exports = {};
22
- __export(index_exports, {
23
- CloudflareTurnAddon: () => CloudflareTurnAddon,
24
- CloudflareTurnService: () => CloudflareTurnService
25
- });
26
- module.exports = __toCommonJS(index_exports);
27
-
28
- // src/cloudflare-turn.ts
29
- var CloudflareTurnService = class {
30
- constructor(config, logger) {
31
- this.config = config;
32
- this.logger = logger;
33
- }
34
- /** Get temporary TURN credentials from Cloudflare API */
35
- async getCredentials() {
36
- this.logger.debug("Fetching TURN credentials from Cloudflare");
37
- return {
38
- urls: [
39
- "turn:turn.cloudflare.com:3478?transport=udp",
40
- "turn:turn.cloudflare.com:3478?transport=tcp"
41
- ],
42
- username: "temp-user",
43
- credential: "temp-credential",
44
- ttl: 86400
45
- };
46
- }
47
- };
48
-
49
- // src/cloudflare-turn.addon.ts
50
- var CloudflareTurnAddon = class {
51
- manifest = {
52
- id: "cloudflare-turn",
53
- name: "Cloudflare TURN Relay",
54
- version: "1.0.0",
55
- capabilities: ["turn-provider"]
56
- };
57
- service = null;
58
- currentConfig = {
59
- apiToken: "",
60
- accountId: ""
61
- };
62
- async initialize(context) {
63
- this.currentConfig = {
64
- apiToken: context.addonConfig.apiToken ?? this.currentConfig.apiToken,
65
- accountId: context.addonConfig.accountId ?? this.currentConfig.accountId
66
- };
67
- this.service = new CloudflareTurnService(this.currentConfig, context.logger);
68
- context.logger.info("Cloudflare TURN initialized");
69
- }
70
- async shutdown() {
71
- this.service = null;
72
- }
73
- getService() {
74
- if (!this.service) throw new Error("Cloudflare TURN not initialized");
75
- return this.service;
76
- }
77
- getCapabilityProvider(name) {
78
- if (name === "turn-provider" && this.service) {
79
- return this.service;
80
- }
81
- return null;
82
- }
83
- getConfigSchema() {
84
- return {
85
- sections: [
86
- {
87
- id: "credentials",
88
- title: "Cloudflare Credentials",
89
- description: "API credentials for fetching TURN relay tokens from Cloudflare.",
90
- fields: [
91
- {
92
- type: "text",
93
- key: "accountId",
94
- label: "Account ID",
95
- description: "Your Cloudflare account ID (found in the Cloudflare dashboard URL)",
96
- placeholder: "a1b2c3d4e5f6...",
97
- required: true
98
- },
99
- {
100
- type: "password",
101
- key: "apiToken",
102
- label: "API Token",
103
- description: "Cloudflare API token with Calls: Read permission",
104
- showToggle: true,
105
- required: true
106
- }
107
- ]
108
- }
109
- ]
110
- };
111
- }
112
- getConfig() {
113
- return { accountId: this.currentConfig.accountId, apiToken: this.currentConfig.apiToken };
114
- }
115
- async onConfigChange(config) {
116
- this.currentConfig = {
117
- apiToken: config.apiToken ?? this.currentConfig.apiToken,
118
- accountId: config.accountId ?? this.currentConfig.accountId
119
- };
120
- }
121
- };
122
- // Annotate the CommonJS export names for ESM import in node:
123
- 0 && (module.exports = {
124
- CloudflareTurnAddon,
125
- CloudflareTurnService
126
- });
127
- //# sourceMappingURL=index.js.map
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const cloudflareTurn_addon = require("./cloudflare-turn.addon.js");
4
+ exports.CloudflareTurnAddon = cloudflareTurn_addon.CloudflareTurnAddon;
5
+ exports.CloudflareTurnService = cloudflareTurn_addon.CloudflareTurnService;
6
+ //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/cloudflare-turn.ts","../src/cloudflare-turn.addon.ts"],"sourcesContent":["export { CloudflareTurnService } from './cloudflare-turn'\nexport type { CloudflareTurnConfig, TurnCredentials } from './cloudflare-turn'\nexport { CloudflareTurnAddon } from './cloudflare-turn.addon'\n","import type { IScopedLogger } from '@camstack/types'\n\nexport interface CloudflareTurnConfig {\n readonly apiToken: string\n readonly accountId: string\n}\n\nexport interface TurnCredentials {\n readonly urls: readonly string[]\n readonly username: string\n readonly credential: string\n readonly ttl: number\n}\n\nexport class CloudflareTurnService {\n constructor(\n private readonly config: CloudflareTurnConfig,\n private readonly logger: IScopedLogger,\n ) {}\n\n /** Get temporary TURN credentials from Cloudflare API */\n async getCredentials(): Promise<TurnCredentials> {\n this.logger.debug('Fetching TURN credentials from Cloudflare')\n\n return {\n urls: [\n 'turn:turn.cloudflare.com:3478?transport=udp',\n 'turn:turn.cloudflare.com:3478?transport=tcp',\n ],\n username: 'temp-user',\n credential: 'temp-credential',\n ttl: 86400,\n }\n }\n}\n","import type {\n ICamstackAddon, AddonManifest, AddonContext,\n IConfigurable, ConfigUISchema, CapabilityProviderMap,\n} from '@camstack/types'\nimport { CloudflareTurnService } from './cloudflare-turn'\nimport type { CloudflareTurnConfig } from './cloudflare-turn'\n\nexport class CloudflareTurnAddon implements ICamstackAddon, IConfigurable {\n readonly manifest: AddonManifest = {\n id: 'cloudflare-turn',\n name: 'Cloudflare TURN Relay',\n version: '1.0.0',\n capabilities: ['turn-provider'],\n }\n\n private service: CloudflareTurnService | null = null\n private currentConfig: CloudflareTurnConfig = {\n apiToken: '',\n accountId: '',\n }\n\n async initialize(context: AddonContext): Promise<void> {\n this.currentConfig = {\n apiToken: (context.addonConfig.apiToken as string) ?? this.currentConfig.apiToken,\n accountId: (context.addonConfig.accountId as string) ?? this.currentConfig.accountId,\n }\n this.service = new CloudflareTurnService(this.currentConfig, context.logger)\n context.logger.info('Cloudflare TURN initialized')\n }\n\n async shutdown(): Promise<void> {\n this.service = null\n }\n\n getService(): CloudflareTurnService {\n if (!this.service) throw new Error('Cloudflare TURN not initialized')\n return this.service\n }\n\n getCapabilityProvider<K extends keyof CapabilityProviderMap>(\n name: K,\n ): CapabilityProviderMap[K] | null {\n if (name === 'turn-provider' as string && this.service) {\n return this.service as unknown as CapabilityProviderMap[K]\n }\n return null\n }\n\n getConfigSchema(): ConfigUISchema {\n return {\n sections: [\n {\n id: 'credentials',\n title: 'Cloudflare Credentials',\n description: 'API credentials for fetching TURN relay tokens from Cloudflare.',\n fields: [\n {\n type: 'text',\n key: 'accountId',\n label: 'Account ID',\n description: 'Your Cloudflare account ID (found in the Cloudflare dashboard URL)',\n placeholder: 'a1b2c3d4e5f6...',\n required: true,\n },\n {\n type: 'password',\n key: 'apiToken',\n label: 'API Token',\n description: 'Cloudflare API token with Calls: Read permission',\n showToggle: true,\n required: true,\n },\n ],\n },\n ],\n }\n }\n\n getConfig(): Record<string, unknown> {\n return { accountId: this.currentConfig.accountId, apiToken: this.currentConfig.apiToken }\n }\n\n async onConfigChange(config: Record<string, unknown>): Promise<void> {\n this.currentConfig = {\n apiToken: (config.apiToken as string) ?? this.currentConfig.apiToken,\n accountId: (config.accountId as string) ?? this.currentConfig.accountId,\n }\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;;ACcO,IAAM,wBAAN,MAA4B;AAAA,EACjC,YACmB,QACA,QACjB;AAFiB;AACA;AAAA,EAChB;AAAA;AAAA,EAGH,MAAM,iBAA2C;AAC/C,SAAK,OAAO,MAAM,2CAA2C;AAE7D,WAAO;AAAA,MACL,MAAM;AAAA,QACJ;AAAA,QACA;AAAA,MACF;AAAA,MACA,UAAU;AAAA,MACV,YAAY;AAAA,MACZ,KAAK;AAAA,IACP;AAAA,EACF;AACF;;;AC3BO,IAAM,sBAAN,MAAmE;AAAA,EAC/D,WAA0B;AAAA,IACjC,IAAI;AAAA,IACJ,MAAM;AAAA,IACN,SAAS;AAAA,IACT,cAAc,CAAC,eAAe;AAAA,EAChC;AAAA,EAEQ,UAAwC;AAAA,EACxC,gBAAsC;AAAA,IAC5C,UAAU;AAAA,IACV,WAAW;AAAA,EACb;AAAA,EAEA,MAAM,WAAW,SAAsC;AACrD,SAAK,gBAAgB;AAAA,MACnB,UAAW,QAAQ,YAAY,YAAuB,KAAK,cAAc;AAAA,MACzE,WAAY,QAAQ,YAAY,aAAwB,KAAK,cAAc;AAAA,IAC7E;AACA,SAAK,UAAU,IAAI,sBAAsB,KAAK,eAAe,QAAQ,MAAM;AAC3E,YAAQ,OAAO,KAAK,6BAA6B;AAAA,EACnD;AAAA,EAEA,MAAM,WAA0B;AAC9B,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,aAAoC;AAClC,QAAI,CAAC,KAAK,QAAS,OAAM,IAAI,MAAM,iCAAiC;AACpE,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,sBACE,MACiC;AACjC,QAAI,SAAS,mBAA6B,KAAK,SAAS;AACtD,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EAEA,kBAAkC;AAChC,WAAO;AAAA,MACL,UAAU;AAAA,QACR;AAAA,UACE,IAAI;AAAA,UACJ,OAAO;AAAA,UACP,aAAa;AAAA,UACb,QAAQ;AAAA,YACN;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,aAAa;AAAA,cACb,UAAU;AAAA,YACZ;AAAA,YACA;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,YAAY;AAAA,cACZ,UAAU;AAAA,YACZ;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,YAAqC;AACnC,WAAO,EAAE,WAAW,KAAK,cAAc,WAAW,UAAU,KAAK,cAAc,SAAS;AAAA,EAC1F;AAAA,EAEA,MAAM,eAAe,QAAgD;AACnE,SAAK,gBAAgB;AAAA,MACnB,UAAW,OAAO,YAAuB,KAAK,cAAc;AAAA,MAC5D,WAAY,OAAO,aAAwB,KAAK,cAAc;AAAA,IAChE;AAAA,EACF;AACF;","names":[]}
1
+ {"version":3,"file":"index.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;;"}
package/dist/index.mjs CHANGED
@@ -1,9 +1,6 @@
1
- import {
2
- CloudflareTurnAddon,
3
- CloudflareTurnService
4
- } from "./chunk-V6KRQ44N.mjs";
1
+ import { CloudflareTurnAddon, C } from "./cloudflare-turn.addon.mjs";
5
2
  export {
6
3
  CloudflareTurnAddon,
7
- CloudflareTurnService
4
+ C as CloudflareTurnService
8
5
  };
9
- //# sourceMappingURL=index.mjs.map
6
+ //# sourceMappingURL=index.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
1
+ {"version":3,"file":"index.mjs","sources":[],"sourcesContent":[],"names":[],"mappings":";"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@camstack/addon-cloudflare-turn",
3
- "version": "0.1.13",
3
+ "version": "0.1.15",
4
4
  "description": "Cloudflare TURN relay addon for CamStack",
5
5
  "keywords": [
6
6
  "camstack",
@@ -32,12 +32,16 @@
32
32
  "addons": [
33
33
  {
34
34
  "id": "cloudflare-turn",
35
+ "name": "Cloudflare TURN Relay",
36
+ "version": "1.0.0",
35
37
  "entry": "./dist/cloudflare-turn.addon.js",
36
- "slot": null,
38
+ "execution": {
39
+ "placement": "any-node",
40
+ "group": "cloudflare-turn"
41
+ },
37
42
  "capabilities": [
38
43
  {
39
- "name": "turn-provider",
40
- "mode": "collection"
44
+ "name": "turn-provider"
41
45
  }
42
46
  ]
43
47
  }
@@ -47,7 +51,7 @@
47
51
  "dist"
48
52
  ],
49
53
  "scripts": {
50
- "build": "tsup",
54
+ "build": "vite build",
51
55
  "dev": "tsup --watch",
52
56
  "typecheck": "tsc --noEmit",
53
57
  "publish": "npm publish --access public"
@@ -1,100 +0,0 @@
1
- // src/cloudflare-turn.ts
2
- var CloudflareTurnService = class {
3
- constructor(config, logger) {
4
- this.config = config;
5
- this.logger = logger;
6
- }
7
- /** Get temporary TURN credentials from Cloudflare API */
8
- async getCredentials() {
9
- this.logger.debug("Fetching TURN credentials from Cloudflare");
10
- return {
11
- urls: [
12
- "turn:turn.cloudflare.com:3478?transport=udp",
13
- "turn:turn.cloudflare.com:3478?transport=tcp"
14
- ],
15
- username: "temp-user",
16
- credential: "temp-credential",
17
- ttl: 86400
18
- };
19
- }
20
- };
21
-
22
- // src/cloudflare-turn.addon.ts
23
- var CloudflareTurnAddon = class {
24
- manifest = {
25
- id: "cloudflare-turn",
26
- name: "Cloudflare TURN Relay",
27
- version: "1.0.0",
28
- capabilities: ["turn-provider"]
29
- };
30
- service = null;
31
- currentConfig = {
32
- apiToken: "",
33
- accountId: ""
34
- };
35
- async initialize(context) {
36
- this.currentConfig = {
37
- apiToken: context.addonConfig.apiToken ?? this.currentConfig.apiToken,
38
- accountId: context.addonConfig.accountId ?? this.currentConfig.accountId
39
- };
40
- this.service = new CloudflareTurnService(this.currentConfig, context.logger);
41
- context.logger.info("Cloudflare TURN initialized");
42
- }
43
- async shutdown() {
44
- this.service = null;
45
- }
46
- getService() {
47
- if (!this.service) throw new Error("Cloudflare TURN not initialized");
48
- return this.service;
49
- }
50
- getCapabilityProvider(name) {
51
- if (name === "turn-provider" && this.service) {
52
- return this.service;
53
- }
54
- return null;
55
- }
56
- getConfigSchema() {
57
- return {
58
- sections: [
59
- {
60
- id: "credentials",
61
- title: "Cloudflare Credentials",
62
- description: "API credentials for fetching TURN relay tokens from Cloudflare.",
63
- fields: [
64
- {
65
- type: "text",
66
- key: "accountId",
67
- label: "Account ID",
68
- description: "Your Cloudflare account ID (found in the Cloudflare dashboard URL)",
69
- placeholder: "a1b2c3d4e5f6...",
70
- required: true
71
- },
72
- {
73
- type: "password",
74
- key: "apiToken",
75
- label: "API Token",
76
- description: "Cloudflare API token with Calls: Read permission",
77
- showToggle: true,
78
- required: true
79
- }
80
- ]
81
- }
82
- ]
83
- };
84
- }
85
- getConfig() {
86
- return { accountId: this.currentConfig.accountId, apiToken: this.currentConfig.apiToken };
87
- }
88
- async onConfigChange(config) {
89
- this.currentConfig = {
90
- apiToken: config.apiToken ?? this.currentConfig.apiToken,
91
- accountId: config.accountId ?? this.currentConfig.accountId
92
- };
93
- }
94
- };
95
-
96
- export {
97
- CloudflareTurnService,
98
- CloudflareTurnAddon
99
- };
100
- //# sourceMappingURL=chunk-V6KRQ44N.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/cloudflare-turn.ts","../src/cloudflare-turn.addon.ts"],"sourcesContent":["import type { IScopedLogger } from '@camstack/types'\n\nexport interface CloudflareTurnConfig {\n readonly apiToken: string\n readonly accountId: string\n}\n\nexport interface TurnCredentials {\n readonly urls: readonly string[]\n readonly username: string\n readonly credential: string\n readonly ttl: number\n}\n\nexport class CloudflareTurnService {\n constructor(\n private readonly config: CloudflareTurnConfig,\n private readonly logger: IScopedLogger,\n ) {}\n\n /** Get temporary TURN credentials from Cloudflare API */\n async getCredentials(): Promise<TurnCredentials> {\n this.logger.debug('Fetching TURN credentials from Cloudflare')\n\n return {\n urls: [\n 'turn:turn.cloudflare.com:3478?transport=udp',\n 'turn:turn.cloudflare.com:3478?transport=tcp',\n ],\n username: 'temp-user',\n credential: 'temp-credential',\n ttl: 86400,\n }\n }\n}\n","import type {\n ICamstackAddon, AddonManifest, AddonContext,\n IConfigurable, ConfigUISchema, CapabilityProviderMap,\n} from '@camstack/types'\nimport { CloudflareTurnService } from './cloudflare-turn'\nimport type { CloudflareTurnConfig } from './cloudflare-turn'\n\nexport class CloudflareTurnAddon implements ICamstackAddon, IConfigurable {\n readonly manifest: AddonManifest = {\n id: 'cloudflare-turn',\n name: 'Cloudflare TURN Relay',\n version: '1.0.0',\n capabilities: ['turn-provider'],\n }\n\n private service: CloudflareTurnService | null = null\n private currentConfig: CloudflareTurnConfig = {\n apiToken: '',\n accountId: '',\n }\n\n async initialize(context: AddonContext): Promise<void> {\n this.currentConfig = {\n apiToken: (context.addonConfig.apiToken as string) ?? this.currentConfig.apiToken,\n accountId: (context.addonConfig.accountId as string) ?? this.currentConfig.accountId,\n }\n this.service = new CloudflareTurnService(this.currentConfig, context.logger)\n context.logger.info('Cloudflare TURN initialized')\n }\n\n async shutdown(): Promise<void> {\n this.service = null\n }\n\n getService(): CloudflareTurnService {\n if (!this.service) throw new Error('Cloudflare TURN not initialized')\n return this.service\n }\n\n getCapabilityProvider<K extends keyof CapabilityProviderMap>(\n name: K,\n ): CapabilityProviderMap[K] | null {\n if (name === 'turn-provider' as string && this.service) {\n return this.service as unknown as CapabilityProviderMap[K]\n }\n return null\n }\n\n getConfigSchema(): ConfigUISchema {\n return {\n sections: [\n {\n id: 'credentials',\n title: 'Cloudflare Credentials',\n description: 'API credentials for fetching TURN relay tokens from Cloudflare.',\n fields: [\n {\n type: 'text',\n key: 'accountId',\n label: 'Account ID',\n description: 'Your Cloudflare account ID (found in the Cloudflare dashboard URL)',\n placeholder: 'a1b2c3d4e5f6...',\n required: true,\n },\n {\n type: 'password',\n key: 'apiToken',\n label: 'API Token',\n description: 'Cloudflare API token with Calls: Read permission',\n showToggle: true,\n required: true,\n },\n ],\n },\n ],\n }\n }\n\n getConfig(): Record<string, unknown> {\n return { accountId: this.currentConfig.accountId, apiToken: this.currentConfig.apiToken }\n }\n\n async onConfigChange(config: Record<string, unknown>): Promise<void> {\n this.currentConfig = {\n apiToken: (config.apiToken as string) ?? this.currentConfig.apiToken,\n accountId: (config.accountId as string) ?? this.currentConfig.accountId,\n }\n }\n}\n"],"mappings":";AAcO,IAAM,wBAAN,MAA4B;AAAA,EACjC,YACmB,QACA,QACjB;AAFiB;AACA;AAAA,EAChB;AAAA;AAAA,EAGH,MAAM,iBAA2C;AAC/C,SAAK,OAAO,MAAM,2CAA2C;AAE7D,WAAO;AAAA,MACL,MAAM;AAAA,QACJ;AAAA,QACA;AAAA,MACF;AAAA,MACA,UAAU;AAAA,MACV,YAAY;AAAA,MACZ,KAAK;AAAA,IACP;AAAA,EACF;AACF;;;AC3BO,IAAM,sBAAN,MAAmE;AAAA,EAC/D,WAA0B;AAAA,IACjC,IAAI;AAAA,IACJ,MAAM;AAAA,IACN,SAAS;AAAA,IACT,cAAc,CAAC,eAAe;AAAA,EAChC;AAAA,EAEQ,UAAwC;AAAA,EACxC,gBAAsC;AAAA,IAC5C,UAAU;AAAA,IACV,WAAW;AAAA,EACb;AAAA,EAEA,MAAM,WAAW,SAAsC;AACrD,SAAK,gBAAgB;AAAA,MACnB,UAAW,QAAQ,YAAY,YAAuB,KAAK,cAAc;AAAA,MACzE,WAAY,QAAQ,YAAY,aAAwB,KAAK,cAAc;AAAA,IAC7E;AACA,SAAK,UAAU,IAAI,sBAAsB,KAAK,eAAe,QAAQ,MAAM;AAC3E,YAAQ,OAAO,KAAK,6BAA6B;AAAA,EACnD;AAAA,EAEA,MAAM,WAA0B;AAC9B,SAAK,UAAU;AAAA,EACjB;AAAA,EAEA,aAAoC;AAClC,QAAI,CAAC,KAAK,QAAS,OAAM,IAAI,MAAM,iCAAiC;AACpE,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,sBACE,MACiC;AACjC,QAAI,SAAS,mBAA6B,KAAK,SAAS;AACtD,aAAO,KAAK;AAAA,IACd;AACA,WAAO;AAAA,EACT;AAAA,EAEA,kBAAkC;AAChC,WAAO;AAAA,MACL,UAAU;AAAA,QACR;AAAA,UACE,IAAI;AAAA,UACJ,OAAO;AAAA,UACP,aAAa;AAAA,UACb,QAAQ;AAAA,YACN;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,aAAa;AAAA,cACb,UAAU;AAAA,YACZ;AAAA,YACA;AAAA,cACE,MAAM;AAAA,cACN,KAAK;AAAA,cACL,OAAO;AAAA,cACP,aAAa;AAAA,cACb,YAAY;AAAA,cACZ,UAAU;AAAA,YACZ;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,YAAqC;AACnC,WAAO,EAAE,WAAW,KAAK,cAAc,WAAW,UAAU,KAAK,cAAc,SAAS;AAAA,EAC1F;AAAA,EAEA,MAAM,eAAe,QAAgD;AACnE,SAAK,gBAAgB;AAAA,MACnB,UAAW,OAAO,YAAuB,KAAK,cAAc;AAAA,MAC5D,WAAY,OAAO,aAAwB,KAAK,cAAc;AAAA,IAChE;AAAA,EACF;AACF;","names":[]}
@@ -1,34 +0,0 @@
1
- import { IScopedLogger, ICamstackAddon, IConfigurable, AddonManifest, AddonContext, CapabilityProviderMap, ConfigUISchema } from '@camstack/types';
2
-
3
- interface CloudflareTurnConfig {
4
- readonly apiToken: string;
5
- readonly accountId: string;
6
- }
7
- interface TurnCredentials {
8
- readonly urls: readonly string[];
9
- readonly username: string;
10
- readonly credential: string;
11
- readonly ttl: number;
12
- }
13
- declare class CloudflareTurnService {
14
- private readonly config;
15
- private readonly logger;
16
- constructor(config: CloudflareTurnConfig, logger: IScopedLogger);
17
- /** Get temporary TURN credentials from Cloudflare API */
18
- getCredentials(): Promise<TurnCredentials>;
19
- }
20
-
21
- declare class CloudflareTurnAddon implements ICamstackAddon, IConfigurable {
22
- readonly manifest: AddonManifest;
23
- private service;
24
- private currentConfig;
25
- initialize(context: AddonContext): Promise<void>;
26
- shutdown(): Promise<void>;
27
- getService(): CloudflareTurnService;
28
- getCapabilityProvider<K extends keyof CapabilityProviderMap>(name: K): CapabilityProviderMap[K] | null;
29
- getConfigSchema(): ConfigUISchema;
30
- getConfig(): Record<string, unknown>;
31
- onConfigChange(config: Record<string, unknown>): Promise<void>;
32
- }
33
-
34
- export { type CloudflareTurnConfig as C, CloudflareTurnAddon, type TurnCredentials as T, CloudflareTurnService as a };
@@ -1,34 +0,0 @@
1
- import { IScopedLogger, ICamstackAddon, IConfigurable, AddonManifest, AddonContext, CapabilityProviderMap, ConfigUISchema } from '@camstack/types';
2
-
3
- interface CloudflareTurnConfig {
4
- readonly apiToken: string;
5
- readonly accountId: string;
6
- }
7
- interface TurnCredentials {
8
- readonly urls: readonly string[];
9
- readonly username: string;
10
- readonly credential: string;
11
- readonly ttl: number;
12
- }
13
- declare class CloudflareTurnService {
14
- private readonly config;
15
- private readonly logger;
16
- constructor(config: CloudflareTurnConfig, logger: IScopedLogger);
17
- /** Get temporary TURN credentials from Cloudflare API */
18
- getCredentials(): Promise<TurnCredentials>;
19
- }
20
-
21
- declare class CloudflareTurnAddon implements ICamstackAddon, IConfigurable {
22
- readonly manifest: AddonManifest;
23
- private service;
24
- private currentConfig;
25
- initialize(context: AddonContext): Promise<void>;
26
- shutdown(): Promise<void>;
27
- getService(): CloudflareTurnService;
28
- getCapabilityProvider<K extends keyof CapabilityProviderMap>(name: K): CapabilityProviderMap[K] | null;
29
- getConfigSchema(): ConfigUISchema;
30
- getConfig(): Record<string, unknown>;
31
- onConfigChange(config: Record<string, unknown>): Promise<void>;
32
- }
33
-
34
- export { type CloudflareTurnConfig as C, CloudflareTurnAddon, type TurnCredentials as T, CloudflareTurnService as a };
package/dist/index.d.mts DELETED
@@ -1,2 +0,0 @@
1
- export { CloudflareTurnAddon, C as CloudflareTurnConfig, a as CloudflareTurnService, T as TurnCredentials } from './cloudflare-turn.addon.mjs';
2
- import '@camstack/types';
package/dist/index.d.ts DELETED
@@ -1,2 +0,0 @@
1
- export { CloudflareTurnAddon, C as CloudflareTurnConfig, a as CloudflareTurnService, T as TurnCredentials } from './cloudflare-turn.addon.js';
2
- import '@camstack/types';