ChannelsNotice.tsx
components/LogoV2/ChannelsNotice.tsx
266
Lines
29503
Bytes
1
Exports
11
Imports
10
Keywords
What this is
This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.
Beginner explanation
This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.
How it is used
Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.
Expert explanation
Architecturally, this file intersects with ui-flow. It contains 266 lines, 11 detected imports, and 1 detected exports.
Important relationships
Detected exports
ChannelsNotice
Keywords
textentrylistflagelseunmatchedchannelsnamereactallowlist
Detected imports
react/compiler-runtimereactreact../../bootstrap/state.js../../ink.js../../services/mcp/channelAllowlist.js../../services/mcp/channelNotification.js../../services/mcp/config.js../../utils/auth.js../../utils/plugins/installedPluginsManager.js../../utils/settings/settings.js
Source notes
This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.
Full source
import { c as _c } from "react/compiler-runtime";
// Conditionally require()'d in LogoV2.tsx behind feature('KAIROS') ||
// feature('KAIROS_CHANNELS'). No feature() guard here — the whole file
// tree-shakes via the require pattern when both flags are false (see
// docs/feature-gating.md). Do NOT import this module statically from
// unguarded code.
import * as React from 'react';
import { useState } from 'react';
import { type ChannelEntry, getAllowedChannels, getHasDevChannels } from '../../bootstrap/state.js';
import { Box, Text } from '../../ink.js';
import { isChannelsEnabled } from '../../services/mcp/channelAllowlist.js';
import { getEffectiveChannelAllowlist } from '../../services/mcp/channelNotification.js';
import { getMcpConfigsByScope } from '../../services/mcp/config.js';
import { getClaudeAIOAuthTokens, getSubscriptionType } from '../../utils/auth.js';
import { loadInstalledPluginsV2 } from '../../utils/plugins/installedPluginsManager.js';
import { getSettingsForSource } from '../../utils/settings/settings.js';
export function ChannelsNotice() {
const $ = _c(32);
const [t0] = useState(_temp);
const {
channels,
disabled,
noAuth,
policyBlocked,
list,
unmatched
} = t0;
if (channels.length === 0) {
return null;
}
const hasNonDev = channels.some(_temp2);
const flag = getHasDevChannels() && hasNonDev ? "Channels" : getHasDevChannels() ? "--dangerously-load-development-channels" : "--channels";
if (disabled) {
let t1;
if ($[0] !== flag || $[1] !== list) {
t1 = <Text color="error">{flag} ignored ({list})</Text>;
$[0] = flag;
$[1] = list;
$[2] = t1;
} else {
t1 = $[2];
}
let t2;
if ($[3] === Symbol.for("react.memo_cache_sentinel")) {
t2 = <Text dimColor={true}>Channels are not currently available</Text>;
$[3] = t2;
} else {
t2 = $[3];
}
let t3;
if ($[4] !== t1) {
t3 = <Box paddingLeft={2} flexDirection="column">{t1}{t2}</Box>;
$[4] = t1;
$[5] = t3;
} else {
t3 = $[5];
}
return t3;
}
if (noAuth) {
let t1;
if ($[6] !== flag || $[7] !== list) {
t1 = <Text color="error">{flag} ignored ({list})</Text>;
$[6] = flag;
$[7] = list;
$[8] = t1;
} else {
t1 = $[8];
}
let t2;
if ($[9] === Symbol.for("react.memo_cache_sentinel")) {
t2 = <Text dimColor={true}>Channels require claude.ai authentication · run /login, then restart</Text>;
$[9] = t2;
} else {
t2 = $[9];
}
let t3;
if ($[10] !== t1) {
t3 = <Box paddingLeft={2} flexDirection="column">{t1}{t2}</Box>;
$[10] = t1;
$[11] = t3;
} else {
t3 = $[11];
}
return t3;
}
if (policyBlocked) {
let t1;
if ($[12] !== flag || $[13] !== list) {
t1 = <Text color="error">{flag} blocked by org policy ({list})</Text>;
$[12] = flag;
$[13] = list;
$[14] = t1;
} else {
t1 = $[14];
}
let t2;
let t3;
if ($[15] === Symbol.for("react.memo_cache_sentinel")) {
t2 = <Text dimColor={true}>Inbound messages will be silently dropped</Text>;
t3 = <Text dimColor={true}>Have an administrator set channelsEnabled: true in managed settings to enable</Text>;
$[15] = t2;
$[16] = t3;
} else {
t2 = $[15];
t3 = $[16];
}
let t4;
if ($[17] !== unmatched) {
t4 = unmatched.map(_temp3);
$[17] = unmatched;
$[18] = t4;
} else {
t4 = $[18];
}
let t5;
if ($[19] !== t1 || $[20] !== t4) {
t5 = <Box paddingLeft={2} flexDirection="column">{t1}{t2}{t3}{t4}</Box>;
$[19] = t1;
$[20] = t4;
$[21] = t5;
} else {
t5 = $[21];
}
return t5;
}
let t1;
if ($[22] !== list) {
t1 = <Text color="error">Listening for channel messages from: {list}</Text>;
$[22] = list;
$[23] = t1;
} else {
t1 = $[23];
}
let t2;
if ($[24] !== flag) {
t2 = <Text dimColor={true}>Experimental · inbound messages will be pushed into this session, this carries prompt injection risks. Restart Claude Code without {flag} to disable.</Text>;
$[24] = flag;
$[25] = t2;
} else {
t2 = $[25];
}
let t3;
if ($[26] !== unmatched) {
t3 = unmatched.map(_temp4);
$[26] = unmatched;
$[27] = t3;
} else {
t3 = $[27];
}
let t4;
if ($[28] !== t1 || $[29] !== t2 || $[30] !== t3) {
t4 = <Box paddingLeft={2} flexDirection="column">{t1}{t2}{t3}</Box>;
$[28] = t1;
$[29] = t2;
$[30] = t3;
$[31] = t4;
} else {
t4 = $[31];
}
return t4;
}
function _temp4(u_0) {
return <Text key={`${formatEntry(u_0.entry)}:${u_0.why}`} color="warning">{formatEntry(u_0.entry)} · {u_0.why}</Text>;
}
function _temp3(u) {
return <Text key={`${formatEntry(u.entry)}:${u.why}`} color="warning">{formatEntry(u.entry)} · {u.why}</Text>;
}
function _temp2(c) {
return !c.dev;
}
function _temp() {
const ch = getAllowedChannels();
if (ch.length === 0) {
return {
channels: ch,
disabled: false,
noAuth: false,
policyBlocked: false,
list: "",
unmatched: [] as Unmatched[]
};
}
const l = ch.map(formatEntry).join(", ");
const sub = getSubscriptionType();
const managed = sub === "team" || sub === "enterprise";
const policy = getSettingsForSource("policySettings");
const allowlist = getEffectiveChannelAllowlist(sub, policy?.allowedChannelPlugins);
return {
channels: ch,
disabled: !isChannelsEnabled(),
noAuth: !getClaudeAIOAuthTokens()?.accessToken,
policyBlocked: managed && policy?.channelsEnabled !== true,
list: l,
unmatched: findUnmatched(ch, allowlist)
};
}
function formatEntry(c: ChannelEntry): string {
return c.kind === 'plugin' ? `plugin:${c.name}@${c.marketplace}` : `server:${c.name}`;
}
type Unmatched = {
entry: ChannelEntry;
why: string;
};
function findUnmatched(entries: readonly ChannelEntry[], allowlist: ReturnType<typeof getEffectiveChannelAllowlist>): Unmatched[] {
// Server-kind: build one Set from all scopes up front. getMcpConfigsByScope
// is not cached (project scope walks the dir tree); getMcpConfigByName would
// redo that walk per entry.
const scopes = ['enterprise', 'user', 'project', 'local'] as const;
const configured = new Set<string>();
for (const scope of scopes) {
for (const name of Object.keys(getMcpConfigsByScope(scope).servers)) {
configured.add(name);
}
}
// Plugin-kind installed check: installed_plugins.json keys are
// `name@marketplace`. loadInstalledPluginsV2 is cached.
const installedPluginIds = new Set(Object.keys(loadInstalledPluginsV2().plugins));
// Plugin-kind allowlist check: same {marketplace, plugin} test as the
// gate at channelNotification.ts. entry.dev bypasses (dev flag opts out
// of the allowlist). Org list replaces ledger when set (team/enterprise).
// GrowthBook _CACHED_MAY_BE_STALE — cold cache yields [] so every plugin
// entry warns; same tradeoff the gate already accepts.
const {
entries: allowed,
source
} = allowlist;
// Independent ifs — a plugin entry that's both uninstalled AND
// unlisted shows two lines. Server kind checks config + dev flag.
const out: Unmatched[] = [];
for (const entry of entries) {
if (entry.kind === 'server') {
if (!configured.has(entry.name)) {
out.push({
entry,
why: 'no MCP server configured with that name'
});
}
if (!entry.dev) {
out.push({
entry,
why: 'server: entries need --dangerously-load-development-channels'
});
}
continue;
}
if (!installedPluginIds.has(`${entry.name}@${entry.marketplace}`)) {
out.push({
entry,
why: 'plugin not installed'
});
}
if (!entry.dev && !allowed.some(e => e.plugin === entry.name && e.marketplace === entry.marketplace)) {
out.push({
entry,
why: source === 'org' ? "not on your org's approved channels list" : 'not on the approved channels allowlist'
});
}
}
return out;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","useState","ChannelEntry","getAllowedChannels","getHasDevChannels","Box","Text","isChannelsEnabled","getEffectiveChannelAllowlist","getMcpConfigsByScope","getClaudeAIOAuthTokens","getSubscriptionType","loadInstalledPluginsV2","getSettingsForSource","ChannelsNotice","$","_c","t0","_temp","channels","disabled","noAuth","policyBlocked","list","unmatched","length","hasNonDev","some","_temp2","flag","t1","t2","Symbol","for","t3","t4","map","_temp3","t5","_temp4","u_0","formatEntry","u","entry","why","c","dev","ch","Unmatched","l","join","sub","managed","policy","allowlist","allowedChannelPlugins","accessToken","channelsEnabled","findUnmatched","kind","name","marketplace","entries","ReturnType","scopes","const","configured","Set","scope","Object","keys","servers","add","installedPluginIds","plugins","allowed","source","out","has","push","e","plugin"],"sources":["ChannelsNotice.tsx"],"sourcesContent":["// Conditionally require()'d in LogoV2.tsx behind feature('KAIROS') ||\n// feature('KAIROS_CHANNELS'). No feature() guard here — the whole file\n// tree-shakes via the require pattern when both flags are false (see\n// docs/feature-gating.md). Do NOT import this module statically from\n// unguarded code.\n\nimport * as React from 'react'\nimport { useState } from 'react'\nimport {\n  type ChannelEntry,\n  getAllowedChannels,\n  getHasDevChannels,\n} from '../../bootstrap/state.js'\nimport { Box, Text } from '../../ink.js'\nimport { isChannelsEnabled } from '../../services/mcp/channelAllowlist.js'\nimport { getEffectiveChannelAllowlist } from '../../services/mcp/channelNotification.js'\nimport { getMcpConfigsByScope } from '../../services/mcp/config.js'\nimport {\n  getClaudeAIOAuthTokens,\n  getSubscriptionType,\n} from '../../utils/auth.js'\nimport { loadInstalledPluginsV2 } from '../../utils/plugins/installedPluginsManager.js'\nimport { getSettingsForSource } from '../../utils/settings/settings.js'\n\nexport function ChannelsNotice(): React.ReactNode {\n  // Snapshot all reads at mount. This notice enters scrollback immediately\n  // after the logo; any re-render past that point forces a full terminal\n  // reset. getAllowedChannels (bootstrap state), getSettingsForSource\n  // (session cache updated by background polling / /login), and\n  // isChannelsEnabled (GrowthBook 5-min refresh) must be captured once\n  // so a later re-render cannot flip branches.\n  const [{ channels, disabled, noAuth, policyBlocked, list, unmatched }] =\n    useState(() => {\n      const ch = getAllowedChannels()\n      if (ch.length === 0)\n        return {\n          channels: ch,\n          disabled: false,\n          noAuth: false,\n          policyBlocked: false,\n          list: '',\n          unmatched: [] as Unmatched[],\n        }\n      const l = ch.map(formatEntry).join(', ')\n      const sub = getSubscriptionType()\n      const managed = sub === 'team' || sub === 'enterprise'\n      const policy = getSettingsForSource('policySettings')\n      const allowlist = getEffectiveChannelAllowlist(\n        sub,\n        policy?.allowedChannelPlugins,\n      )\n      return {\n        channels: ch,\n        disabled: !isChannelsEnabled(),\n        noAuth: !getClaudeAIOAuthTokens()?.accessToken,\n        policyBlocked: managed && policy?.channelsEnabled !== true,\n        list: l,\n        unmatched: findUnmatched(ch, allowlist),\n      }\n    })\n  if (channels.length === 0) return null\n\n  // When both flags are passed, the list mixes entries and a single flag\n  // name would be wrong for half of it. entry.dev distinguishes origin.\n  const hasNonDev = channels.some(c => !c.dev)\n  const flag =\n    getHasDevChannels() && hasNonDev\n      ? 'Channels'\n      : getHasDevChannels()\n        ? '--dangerously-load-development-channels'\n        : '--channels'\n\n  if (disabled) {\n    return (\n      <Box paddingLeft={2} flexDirection=\"column\">\n        <Text color=\"error\">\n          {flag} ignored ({list})\n        </Text>\n        <Text dimColor>Channels are not currently available</Text>\n      </Box>\n    )\n  }\n\n  if (noAuth) {\n    return (\n      <Box paddingLeft={2} flexDirection=\"column\">\n        <Text color=\"error\">\n          {flag} ignored ({list})\n        </Text>\n        <Text dimColor>\n          Channels require claude.ai authentication · run /login, then restart\n        </Text>\n      </Box>\n    )\n  }\n\n  if (policyBlocked) {\n    return (\n      <Box paddingLeft={2} flexDirection=\"column\">\n        <Text color=\"error\">\n          {flag} blocked by org policy ({list})\n        </Text>\n        <Text dimColor>Inbound messages will be silently dropped</Text>\n        <Text dimColor>\n          Have an administrator set channelsEnabled: true in managed settings to\n          enable\n        </Text>\n        {unmatched.map(u => (\n          <Text key={`${formatEntry(u.entry)}:${u.why}`} color=\"warning\">\n            {formatEntry(u.entry)} · {u.why}\n          </Text>\n        ))}\n      </Box>\n    )\n  }\n\n  // \"Listening for\" not \"active\" — at this point we only know the allowlist\n  // was set. Server connection, capability declaration, and whether the name\n  // even matches a configured MCP server are all still unknown.\n  return (\n    <Box paddingLeft={2} flexDirection=\"column\">\n      <Text color=\"error\">Listening for channel messages from: {list}</Text>\n      <Text dimColor>\n        Experimental · inbound messages will be pushed into this session, this\n        carries prompt injection risks. Restart Claude Code without {flag} to\n        disable.\n      </Text>\n      {unmatched.map(u => (\n        <Text key={`${formatEntry(u.entry)}:${u.why}`} color=\"warning\">\n          {formatEntry(u.entry)} · {u.why}\n        </Text>\n      ))}\n    </Box>\n  )\n}\n\nfunction formatEntry(c: ChannelEntry): string {\n  return c.kind === 'plugin'\n    ? `plugin:${c.name}@${c.marketplace}`\n    : `server:${c.name}`\n}\n\ntype Unmatched = { entry: ChannelEntry; why: string }\n\nfunction findUnmatched(\n  entries: readonly ChannelEntry[],\n  allowlist: ReturnType<typeof getEffectiveChannelAllowlist>,\n): Unmatched[] {\n  // Server-kind: build one Set from all scopes up front. getMcpConfigsByScope\n  // is not cached (project scope walks the dir tree); getMcpConfigByName would\n  // redo that walk per entry.\n  const scopes = ['enterprise', 'user', 'project', 'local'] as const\n  const configured = new Set<string>()\n  for (const scope of scopes) {\n    for (const name of Object.keys(getMcpConfigsByScope(scope).servers)) {\n      configured.add(name)\n    }\n  }\n\n  // Plugin-kind installed check: installed_plugins.json keys are\n  // `name@marketplace`. loadInstalledPluginsV2 is cached.\n  const installedPluginIds = new Set(\n    Object.keys(loadInstalledPluginsV2().plugins),\n  )\n\n  // Plugin-kind allowlist check: same {marketplace, plugin} test as the\n  // gate at channelNotification.ts. entry.dev bypasses (dev flag opts out\n  // of the allowlist). Org list replaces ledger when set (team/enterprise).\n  // GrowthBook _CACHED_MAY_BE_STALE — cold cache yields [] so every plugin\n  // entry warns; same tradeoff the gate already accepts.\n  const { entries: allowed, source } = allowlist\n\n  // Independent ifs — a plugin entry that's both uninstalled AND\n  // unlisted shows two lines. Server kind checks config + dev flag.\n  const out: Unmatched[] = []\n  for (const entry of entries) {\n    if (entry.kind === 'server') {\n      if (!configured.has(entry.name)) {\n        out.push({ entry, why: 'no MCP server configured with that name' })\n      }\n      if (!entry.dev) {\n        out.push({\n          entry,\n          why: 'server: entries need --dangerously-load-development-channels',\n        })\n      }\n      continue\n    }\n    if (!installedPluginIds.has(`${entry.name}@${entry.marketplace}`)) {\n      out.push({ entry, why: 'plugin not installed' })\n    }\n    if (\n      !entry.dev &&\n      !allowed.some(\n        e => e.plugin === entry.name && e.marketplace === entry.marketplace,\n      )\n    ) {\n      out.push({\n        entry,\n        why:\n          source === 'org'\n            ? \"not on your org's approved channels list\"\n            : 'not on the approved channels allowlist',\n      })\n    }\n  }\n  return out\n}\n"],"mappings":"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","ignoreList":[]}