mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-09 23:20:44 +00:00
2351 lines
85 KiB
TypeScript
2351 lines
85 KiB
TypeScript
import { mkdir, writeFile } from "node:fs/promises";
|
|
import path from "node:path";
|
|
import { createInterface } from "node:readline/promises";
|
|
import { format } from "node:util";
|
|
import type { Command } from "commander";
|
|
import { formatErrorMessage } from "openclaw/plugin-sdk/error-runtime";
|
|
import { callGatewayFromCli } from "openclaw/plugin-sdk/gateway-runtime";
|
|
import {
|
|
buildGoogleMeetCalendarDayWindow,
|
|
findGoogleMeetCalendarEvent,
|
|
listGoogleMeetCalendarEvents,
|
|
type GoogleMeetCalendarLookupResult,
|
|
} from "./calendar.js";
|
|
import type { GoogleMeetConfig, GoogleMeetModeInput, GoogleMeetTransport } from "./config.js";
|
|
import { hasCreateSpaceConfigInput, resolveCreateSpaceConfig } from "./create.js";
|
|
import {
|
|
buildGoogleMeetPreflightReport,
|
|
createGoogleMeetSpace,
|
|
endGoogleMeetActiveConference,
|
|
fetchGoogleMeetArtifacts,
|
|
fetchGoogleMeetAttendance,
|
|
fetchLatestGoogleMeetConferenceRecord,
|
|
fetchGoogleMeetSpace,
|
|
type GoogleMeetArtifactsResult,
|
|
type GoogleMeetAttendanceResult,
|
|
type GoogleMeetLatestConferenceRecordResult,
|
|
} from "./meet.js";
|
|
import {
|
|
buildGoogleMeetAuthUrl,
|
|
createGoogleMeetOAuthState,
|
|
createGoogleMeetPkce,
|
|
exchangeGoogleMeetAuthCode,
|
|
resolveGoogleMeetAccessToken,
|
|
waitForGoogleMeetAuthCode,
|
|
} from "./oauth.js";
|
|
import type { GoogleMeetRuntime } from "./runtime.js";
|
|
|
|
type JoinOptions = {
|
|
transport?: GoogleMeetTransport;
|
|
mode?: GoogleMeetModeInput;
|
|
message?: string;
|
|
timeoutMs?: string;
|
|
dialInNumber?: string;
|
|
pin?: string;
|
|
dtmfSequence?: string;
|
|
};
|
|
|
|
type OAuthLoginOptions = {
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
manual?: boolean;
|
|
json?: boolean;
|
|
timeoutSec?: string;
|
|
};
|
|
|
|
type ResolveSpaceOptions = {
|
|
meeting?: string;
|
|
today?: boolean;
|
|
event?: string;
|
|
calendar?: string;
|
|
accessToken?: string;
|
|
refreshToken?: string;
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
expiresAt?: string;
|
|
json?: boolean;
|
|
};
|
|
|
|
type MeetArtifactOptions = ResolveSpaceOptions & {
|
|
conferenceRecord?: string;
|
|
pageSize?: string;
|
|
transcriptEntries?: boolean;
|
|
allConferenceRecords?: boolean;
|
|
includeDocBodies?: boolean;
|
|
mergeDuplicates?: boolean;
|
|
lateAfterMinutes?: string;
|
|
earlyBeforeMinutes?: string;
|
|
zip?: boolean;
|
|
dryRun?: boolean;
|
|
format?: "summary" | "markdown" | "csv";
|
|
output?: string;
|
|
};
|
|
|
|
export type GoogleMeetExportRequest = {
|
|
meeting?: string;
|
|
conferenceRecord?: string;
|
|
calendarEventId?: string;
|
|
calendarEventSummary?: string;
|
|
calendarId?: string;
|
|
pageSize?: number;
|
|
includeTranscriptEntries?: boolean;
|
|
includeDocumentBodies?: boolean;
|
|
allConferenceRecords?: boolean;
|
|
mergeDuplicateParticipants?: boolean;
|
|
lateAfterMinutes?: number;
|
|
earlyBeforeMinutes?: number;
|
|
};
|
|
|
|
export type GoogleMeetExportWarning = {
|
|
type:
|
|
| "smart_notes"
|
|
| "transcript_entries"
|
|
| "transcript_document_body"
|
|
| "smart_note_document_body";
|
|
conferenceRecord: string;
|
|
resource?: string;
|
|
message: string;
|
|
};
|
|
|
|
export type GoogleMeetExportManifest = {
|
|
generatedAt: string;
|
|
request?: GoogleMeetExportRequest;
|
|
tokenSource?: "cached-access-token" | "refresh-token";
|
|
calendarEvent?: GoogleMeetCalendarLookupResult;
|
|
inputs: {
|
|
artifacts?: string;
|
|
attendance?: string;
|
|
};
|
|
counts: {
|
|
conferenceRecords: number;
|
|
artifacts: number;
|
|
attendanceRows: number;
|
|
recordings: number;
|
|
transcripts: number;
|
|
transcriptEntries: number;
|
|
smartNotes: number;
|
|
warnings: number;
|
|
};
|
|
conferenceRecords: string[];
|
|
files: string[];
|
|
zipFile?: string;
|
|
warnings: GoogleMeetExportWarning[];
|
|
};
|
|
|
|
type SetupOptions = {
|
|
json?: boolean;
|
|
mode?: GoogleMeetModeInput;
|
|
transport?: GoogleMeetTransport;
|
|
};
|
|
|
|
type GoogleMeetGatewayMethod =
|
|
| "googlemeet.create"
|
|
| "googlemeet.join"
|
|
| "googlemeet.leave"
|
|
| "googlemeet.speak"
|
|
| "googlemeet.status"
|
|
| "googlemeet.testListen"
|
|
| "googlemeet.testSpeech";
|
|
|
|
type GoogleMeetGatewayCallResult = { ok: true; payload: unknown } | { ok: false; error: unknown };
|
|
|
|
const GOOGLE_MEET_GATEWAY_DEFAULT_TIMEOUT_MS = 5000;
|
|
|
|
type DoctorOptions = {
|
|
json?: boolean;
|
|
oauth?: boolean;
|
|
meeting?: string;
|
|
createSpace?: boolean;
|
|
accessToken?: string;
|
|
refreshToken?: string;
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
expiresAt?: string;
|
|
};
|
|
|
|
type JsonOptions = {
|
|
json?: boolean;
|
|
};
|
|
|
|
type RecoverTabOptions = JsonOptions & {
|
|
transport?: GoogleMeetTransport;
|
|
};
|
|
|
|
type CreateOptions = {
|
|
accessToken?: string;
|
|
refreshToken?: string;
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
expiresAt?: string;
|
|
accessType?: string;
|
|
entryPointAccess?: string;
|
|
join?: boolean;
|
|
transport?: GoogleMeetTransport;
|
|
mode?: GoogleMeetModeInput;
|
|
message?: string;
|
|
dialInNumber?: string;
|
|
pin?: string;
|
|
dtmfSequence?: string;
|
|
json?: boolean;
|
|
};
|
|
|
|
function writeStdoutJson(value: unknown): void {
|
|
process.stdout.write(`${JSON.stringify(value, null, 2)}\n`);
|
|
}
|
|
|
|
function isGatewayUnavailableForLocalFallback(
|
|
err: unknown,
|
|
method: GoogleMeetGatewayMethod,
|
|
): boolean {
|
|
const message = formatErrorMessage(err);
|
|
return (
|
|
message.includes("ECONNREFUSED") ||
|
|
message.includes("ECONNRESET") ||
|
|
message.includes("EHOSTUNREACH") ||
|
|
message.includes("ENOTFOUND") ||
|
|
message.includes("gateway not connected") ||
|
|
message.includes(`unknown method: ${method}`)
|
|
);
|
|
}
|
|
|
|
function writeStdoutLine(...values: unknown[]): void {
|
|
process.stdout.write(`${format(...values)}\n`);
|
|
}
|
|
|
|
async function writeCliOutput(options: { output?: string }, text: string): Promise<void> {
|
|
if (options.output?.trim()) {
|
|
await writeFile(options.output, text.endsWith("\n") ? text : `${text}\n`, "utf8");
|
|
writeStdoutLine("wrote: %s", options.output);
|
|
return;
|
|
}
|
|
process.stdout.write(text.endsWith("\n") ? text : `${text}\n`);
|
|
}
|
|
|
|
async function promptInput(message: string): Promise<string> {
|
|
const rl = createInterface({
|
|
input: process.stdin,
|
|
output: process.stderr,
|
|
});
|
|
try {
|
|
return await rl.question(message);
|
|
} finally {
|
|
rl.close();
|
|
}
|
|
}
|
|
|
|
function parseOptionalNumber(value: string | undefined): number | undefined {
|
|
if (!value?.trim()) {
|
|
return undefined;
|
|
}
|
|
const parsed = Number(value);
|
|
if (!Number.isFinite(parsed)) {
|
|
throw new Error(`Expected a numeric value, received ${value}`);
|
|
}
|
|
return parsed;
|
|
}
|
|
|
|
function writeSetupStatus(status: Awaited<ReturnType<GoogleMeetRuntime["setupStatus"]>>): void {
|
|
writeStdoutLine("Google Meet setup: %s", status.ok ? "OK" : "needs attention");
|
|
for (const check of status.checks) {
|
|
writeStdoutLine("[%s] %s: %s", check.ok ? "ok" : "fail", check.id, check.message);
|
|
}
|
|
}
|
|
|
|
function formatBoolean(value: boolean | undefined): string {
|
|
return typeof value === "boolean" ? (value ? "yes" : "no") : "unknown";
|
|
}
|
|
|
|
function formatOptional(value: unknown): string {
|
|
return typeof value === "string" && value.trim() ? value : "n/a";
|
|
}
|
|
|
|
function parsePositiveNumber(value: string | undefined, label: string): number | undefined {
|
|
if (value === undefined) {
|
|
return undefined;
|
|
}
|
|
const parsed = Number(value);
|
|
if (!Number.isFinite(parsed) || parsed <= 0) {
|
|
throw new Error(`${label} must be a positive number`);
|
|
}
|
|
return parsed;
|
|
}
|
|
|
|
async function callGoogleMeetGateway(params: {
|
|
callGateway: typeof callGatewayFromCli;
|
|
method: GoogleMeetGatewayMethod;
|
|
payload?: Record<string, unknown>;
|
|
timeoutMs?: number;
|
|
}): Promise<GoogleMeetGatewayCallResult> {
|
|
try {
|
|
const timeoutMs =
|
|
typeof params.timeoutMs === "number" && Number.isFinite(params.timeoutMs)
|
|
? Math.max(1, Math.ceil(params.timeoutMs))
|
|
: GOOGLE_MEET_GATEWAY_DEFAULT_TIMEOUT_MS;
|
|
return {
|
|
ok: true,
|
|
payload: await params.callGateway(
|
|
params.method,
|
|
{ json: true, timeout: String(timeoutMs) },
|
|
params.payload,
|
|
{ progress: false },
|
|
),
|
|
};
|
|
} catch (err) {
|
|
if (isGatewayUnavailableForLocalFallback(err, params.method)) {
|
|
return { ok: false, error: err };
|
|
}
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
function resolveGoogleMeetGatewayOperationTimeoutMs(config: GoogleMeetConfig): number {
|
|
return Math.max(
|
|
60_000,
|
|
config.chrome.joinTimeoutMs + 30_000,
|
|
config.voiceCall.requestTimeoutMs + 10_000,
|
|
);
|
|
}
|
|
|
|
function formatDuration(value: number | undefined): string {
|
|
if (value === undefined) {
|
|
return "n/a";
|
|
}
|
|
const totalSeconds = Math.round(value / 1000);
|
|
const hours = Math.floor(totalSeconds / 3600);
|
|
const minutes = Math.floor((totalSeconds % 3600) / 60);
|
|
const seconds = totalSeconds % 60;
|
|
return hours > 0
|
|
? `${hours}h ${minutes.toString().padStart(2, "0")}m`
|
|
: `${minutes}m ${seconds.toString().padStart(2, "0")}s`;
|
|
}
|
|
|
|
function writeDoctorStatus(status: Awaited<ReturnType<GoogleMeetRuntime["status"]>>): void {
|
|
if (!status.found) {
|
|
writeStdoutLine("Google Meet session: not found");
|
|
return;
|
|
}
|
|
const sessions = status.session ? [status.session] : (status.sessions ?? []);
|
|
if (sessions.length === 0) {
|
|
writeStdoutLine("Google Meet sessions: none");
|
|
return;
|
|
}
|
|
writeStdoutLine("Google Meet sessions: %d", sessions.length);
|
|
for (const session of sessions) {
|
|
const health = session.chrome?.health;
|
|
writeStdoutLine("");
|
|
writeStdoutLine("session: %s", session.id);
|
|
writeStdoutLine("url: %s", session.url);
|
|
writeStdoutLine("state: %s", session.state);
|
|
writeStdoutLine("transport: %s", session.transport);
|
|
writeStdoutLine("mode: %s", session.mode);
|
|
if (session.twilio) {
|
|
writeStdoutLine("twilio dial-in: %s", session.twilio.dialInNumber);
|
|
writeStdoutLine("voice call id: %s", formatOptional(session.twilio.voiceCallId));
|
|
writeStdoutLine("dtmf sent: %s", formatBoolean(session.twilio.dtmfSent));
|
|
writeStdoutLine("intro sent: %s", formatBoolean(session.twilio.introSent));
|
|
}
|
|
if (!session.chrome) {
|
|
continue;
|
|
}
|
|
writeStdoutLine("node: %s", session.chrome?.nodeId ?? "local/none");
|
|
writeStdoutLine("audio bridge: %s", session.chrome?.audioBridge?.type ?? "none");
|
|
const bridgeProvider =
|
|
session.chrome?.audioBridge?.provider ??
|
|
session.realtime.transcriptionProvider ??
|
|
session.realtime.provider ??
|
|
"n/a";
|
|
writeStdoutLine(
|
|
session.mode === "agent" ? "transcription provider: %s" : "provider: %s",
|
|
bridgeProvider,
|
|
);
|
|
if (session.realtime.enabled) {
|
|
writeStdoutLine("talk-back mode: %s", session.realtime.strategy ?? session.mode);
|
|
}
|
|
writeStdoutLine("in call: %s", formatBoolean(health?.inCall));
|
|
writeStdoutLine("lobby waiting: %s", formatBoolean(health?.lobbyWaiting));
|
|
writeStdoutLine("captioning: %s", formatBoolean(health?.captioning));
|
|
writeStdoutLine("transcript lines: %s", health?.transcriptLines ?? 0);
|
|
writeStdoutLine("last caption: %s", formatOptional(health?.lastCaptionAt));
|
|
writeStdoutLine("manual action: %s", formatBoolean(health?.manualActionRequired));
|
|
if (health?.manualActionRequired) {
|
|
writeStdoutLine("manual reason: %s", formatOptional(health.manualActionReason));
|
|
writeStdoutLine("manual message: %s", formatOptional(health.manualActionMessage));
|
|
}
|
|
writeStdoutLine("speech ready: %s", formatBoolean(health?.speechReady));
|
|
if (health?.speechReady === false) {
|
|
writeStdoutLine("speech blocked reason: %s", formatOptional(health.speechBlockedReason));
|
|
writeStdoutLine("speech blocked message: %s", formatOptional(health.speechBlockedMessage));
|
|
}
|
|
writeStdoutLine("provider connected: %s", formatBoolean(health?.providerConnected));
|
|
writeStdoutLine("realtime ready: %s", formatBoolean(health?.realtimeReady));
|
|
writeStdoutLine("audio input active: %s", formatBoolean(health?.audioInputActive));
|
|
writeStdoutLine("audio output active: %s", formatBoolean(health?.audioOutputActive));
|
|
writeStdoutLine("meet output routed: %s", formatBoolean(health?.audioOutputRouted));
|
|
if (health?.audioOutputDeviceLabel || health?.audioOutputRouteError) {
|
|
writeStdoutLine("meet output device: %s", formatOptional(health.audioOutputDeviceLabel));
|
|
writeStdoutLine("meet output route error: %s", formatOptional(health.audioOutputRouteError));
|
|
}
|
|
writeStdoutLine(
|
|
"last input: %s (%s bytes)",
|
|
formatOptional(health?.lastInputAt),
|
|
health?.lastInputBytes ?? 0,
|
|
);
|
|
writeStdoutLine(
|
|
"last output: %s (%s bytes)",
|
|
formatOptional(health?.lastOutputAt),
|
|
health?.lastOutputBytes ?? 0,
|
|
);
|
|
writeStdoutLine("bridge closed: %s", formatBoolean(health?.bridgeClosed));
|
|
writeStdoutLine("browser url: %s", formatOptional(health?.browserUrl));
|
|
if (health?.lastCaptionText) {
|
|
const speaker = health.lastCaptionSpeaker ? `${health.lastCaptionSpeaker}: ` : "";
|
|
writeStdoutLine("last caption text: %s%s", speaker, health.lastCaptionText);
|
|
}
|
|
writeStdoutLine("realtime transcript lines: %s", health?.realtimeTranscriptLines ?? 0);
|
|
if (health?.lastRealtimeTranscriptText) {
|
|
const role = health.lastRealtimeTranscriptRole
|
|
? `${health.lastRealtimeTranscriptRole}: `
|
|
: "";
|
|
writeStdoutLine("last realtime transcript: %s%s", role, health.lastRealtimeTranscriptText);
|
|
}
|
|
if (health?.lastRealtimeEventType) {
|
|
const detail = health.lastRealtimeEventDetail ? ` ${health.lastRealtimeEventDetail}` : "";
|
|
writeStdoutLine("last realtime event: %s%s", health.lastRealtimeEventType, detail);
|
|
}
|
|
}
|
|
}
|
|
|
|
type OAuthDoctorCheck = {
|
|
id: string;
|
|
ok: boolean;
|
|
message: string;
|
|
};
|
|
|
|
type OAuthDoctorReport = {
|
|
ok: boolean;
|
|
configured: boolean;
|
|
tokenSource?: "cached-access-token" | "refresh-token";
|
|
expiresAt?: number;
|
|
scope?: string;
|
|
meetingUri?: string;
|
|
createdSpace?: string;
|
|
checks: OAuthDoctorCheck[];
|
|
};
|
|
|
|
function sanitizeOAuthErrorMessage(error: unknown): string {
|
|
const message = error instanceof Error ? error.message : String(error);
|
|
return message
|
|
.replace(/(access_token["'=:\s]+)[^"',\s&]+/gi, "$1[redacted]")
|
|
.replace(/(refresh_token["'=:\s]+)[^"',\s&]+/gi, "$1[redacted]")
|
|
.replace(/(client_secret["'=:\s]+)[^"',\s&]+/gi, "$1[redacted]");
|
|
}
|
|
|
|
async function buildOAuthDoctorReport(
|
|
config: GoogleMeetConfig,
|
|
options: DoctorOptions,
|
|
): Promise<OAuthDoctorReport> {
|
|
const clientId = options.clientId?.trim() || config.oauth.clientId;
|
|
const clientSecret = options.clientSecret?.trim() || config.oauth.clientSecret;
|
|
const refreshToken = options.refreshToken?.trim() || config.oauth.refreshToken;
|
|
const accessToken = options.accessToken?.trim() || config.oauth.accessToken;
|
|
const expiresAt = parseOptionalNumber(options.expiresAt) ?? config.oauth.expiresAt;
|
|
const checks: OAuthDoctorCheck[] = [];
|
|
|
|
const hasRefreshConfig = Boolean(clientId && refreshToken);
|
|
const hasAccessConfig = Boolean(accessToken);
|
|
if (!hasRefreshConfig && !hasAccessConfig) {
|
|
checks.push({
|
|
id: "oauth-config",
|
|
ok: false,
|
|
message:
|
|
"Missing Google Meet OAuth credentials. Configure oauth.clientId and oauth.refreshToken, or pass --client-id and --refresh-token.",
|
|
});
|
|
return { ok: false, configured: false, checks };
|
|
}
|
|
|
|
checks.push({
|
|
id: "oauth-config",
|
|
ok: true,
|
|
message: hasRefreshConfig
|
|
? "Google Meet OAuth refresh credentials are configured"
|
|
: "Google Meet cached access token is configured",
|
|
});
|
|
|
|
let token: Awaited<ReturnType<typeof resolveGoogleMeetAccessToken>>;
|
|
try {
|
|
token = await resolveGoogleMeetAccessToken({
|
|
clientId,
|
|
clientSecret,
|
|
refreshToken,
|
|
accessToken,
|
|
expiresAt,
|
|
});
|
|
checks.push({
|
|
id: "oauth-token",
|
|
ok: true,
|
|
message: token.refreshed
|
|
? "Refresh token minted an access token"
|
|
: "Cached access token is still valid",
|
|
});
|
|
} catch (error) {
|
|
checks.push({
|
|
id: "oauth-token",
|
|
ok: false,
|
|
message: sanitizeOAuthErrorMessage(error),
|
|
});
|
|
return { ok: false, configured: true, checks };
|
|
}
|
|
|
|
const report: OAuthDoctorReport = {
|
|
ok: true,
|
|
configured: true,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
expiresAt: token.expiresAt,
|
|
checks,
|
|
};
|
|
|
|
const meeting = options.meeting?.trim();
|
|
if (meeting) {
|
|
try {
|
|
const space = await fetchGoogleMeetSpace({ accessToken: token.accessToken, meeting });
|
|
checks.push({
|
|
id: "meet-spaces-get",
|
|
ok: true,
|
|
message: `Resolved ${space.name}`,
|
|
});
|
|
report.meetingUri = space.meetingUri;
|
|
} catch (error) {
|
|
checks.push({
|
|
id: "meet-spaces-get",
|
|
ok: false,
|
|
message: sanitizeOAuthErrorMessage(error),
|
|
});
|
|
}
|
|
}
|
|
|
|
if (options.createSpace) {
|
|
try {
|
|
const created = await createGoogleMeetSpace({ accessToken: token.accessToken });
|
|
checks.push({
|
|
id: "meet-spaces-create",
|
|
ok: true,
|
|
message: `Created ${created.space.name}`,
|
|
});
|
|
report.createdSpace = created.space.name;
|
|
report.meetingUri = created.meetingUri;
|
|
} catch (error) {
|
|
checks.push({
|
|
id: "meet-spaces-create",
|
|
ok: false,
|
|
message: sanitizeOAuthErrorMessage(error),
|
|
});
|
|
}
|
|
}
|
|
|
|
report.ok = checks.every((check) => check.ok);
|
|
return report;
|
|
}
|
|
|
|
function writeOAuthDoctorReport(report: OAuthDoctorReport): void {
|
|
writeStdoutLine("Google Meet OAuth: %s", report.ok ? "OK" : "needs attention");
|
|
writeStdoutLine("configured: %s", report.configured ? "yes" : "no");
|
|
if (report.tokenSource) {
|
|
writeStdoutLine("token source: %s", report.tokenSource);
|
|
}
|
|
if (report.meetingUri) {
|
|
writeStdoutLine("meeting uri: %s", report.meetingUri);
|
|
}
|
|
for (const check of report.checks) {
|
|
writeStdoutLine("[%s] %s: %s", check.ok ? "ok" : "fail", check.id, check.message);
|
|
}
|
|
}
|
|
|
|
function writeRecoverCurrentTabResult(
|
|
result: Awaited<ReturnType<GoogleMeetRuntime["recoverCurrentTab"]>>,
|
|
): void {
|
|
writeStdoutLine("Google Meet current tab: %s", result.found ? "found" : "not found");
|
|
writeStdoutLine("transport: %s", result.transport);
|
|
writeStdoutLine("node: %s", result.nodeId ?? "local/none");
|
|
if (result.targetId) {
|
|
writeStdoutLine("target: %s", result.targetId);
|
|
}
|
|
if (result.tab?.url) {
|
|
writeStdoutLine("tab url: %s", result.tab.url);
|
|
}
|
|
writeStdoutLine("message: %s", result.message);
|
|
if (result.browser) {
|
|
writeDoctorStatus({
|
|
found: true,
|
|
session: {
|
|
id: "current-tab",
|
|
url: result.browser.browserUrl ?? result.tab?.url ?? "unknown",
|
|
transport: result.transport,
|
|
mode: "transcribe",
|
|
state: "active",
|
|
createdAt: "",
|
|
updatedAt: "",
|
|
participantIdentity:
|
|
result.transport === "chrome-node"
|
|
? "signed-in Google Chrome profile on a paired node"
|
|
: "signed-in Google Chrome profile",
|
|
realtime: { enabled: false, toolPolicy: "safe-read-only" },
|
|
chrome: {
|
|
audioBackend: "blackhole-2ch",
|
|
launched: true,
|
|
nodeId: result.nodeId,
|
|
health: result.browser,
|
|
},
|
|
notes: [],
|
|
},
|
|
});
|
|
}
|
|
}
|
|
|
|
function resolveMeetingInput(config: GoogleMeetConfig, value?: string): string {
|
|
const meeting = value?.trim() || config.defaults.meeting;
|
|
if (!meeting) {
|
|
throw new Error(
|
|
"Meeting input is required. Pass a URL/meeting code or configure defaults.meeting.",
|
|
);
|
|
}
|
|
return meeting;
|
|
}
|
|
|
|
function resolveOAuthTokenOptions(
|
|
config: GoogleMeetConfig,
|
|
options: ResolveSpaceOptions,
|
|
): {
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
refreshToken?: string;
|
|
accessToken?: string;
|
|
expiresAt?: number;
|
|
} {
|
|
return {
|
|
clientId: options.clientId?.trim() || config.oauth.clientId,
|
|
clientSecret: options.clientSecret?.trim() || config.oauth.clientSecret,
|
|
refreshToken: options.refreshToken?.trim() || config.oauth.refreshToken,
|
|
accessToken: options.accessToken?.trim() || config.oauth.accessToken,
|
|
expiresAt: parseOptionalNumber(options.expiresAt) ?? config.oauth.expiresAt,
|
|
};
|
|
}
|
|
|
|
function resolveTokenOptions(
|
|
config: GoogleMeetConfig,
|
|
options: ResolveSpaceOptions,
|
|
): {
|
|
meeting: string;
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
refreshToken?: string;
|
|
accessToken?: string;
|
|
expiresAt?: number;
|
|
} {
|
|
return {
|
|
meeting: resolveMeetingInput(config, options.meeting),
|
|
...resolveOAuthTokenOptions(config, options),
|
|
};
|
|
}
|
|
|
|
function hasCalendarLookupOptions(options: ResolveSpaceOptions): boolean {
|
|
return Boolean(options.today || options.event?.trim());
|
|
}
|
|
|
|
async function resolveCalendarMeetingInput(params: {
|
|
accessToken: string;
|
|
options: ResolveSpaceOptions;
|
|
}): Promise<{ meeting?: string; calendarEvent?: GoogleMeetCalendarLookupResult }> {
|
|
if (!hasCalendarLookupOptions(params.options)) {
|
|
return {};
|
|
}
|
|
const window = params.options.today ? buildGoogleMeetCalendarDayWindow() : {};
|
|
const calendarEvent = await findGoogleMeetCalendarEvent({
|
|
accessToken: params.accessToken,
|
|
calendarId: params.options.calendar,
|
|
eventQuery: params.options.event,
|
|
...window,
|
|
});
|
|
return { meeting: calendarEvent.meetingUri, calendarEvent };
|
|
}
|
|
|
|
async function resolveMeetingForToken(params: {
|
|
config: GoogleMeetConfig;
|
|
options: ResolveSpaceOptions;
|
|
accessToken: string;
|
|
configuredMeeting?: string;
|
|
}): Promise<{ meeting: string; calendarEvent?: GoogleMeetCalendarLookupResult }> {
|
|
const calendarMeeting = await resolveCalendarMeetingInput({
|
|
accessToken: params.accessToken,
|
|
options: params.options,
|
|
});
|
|
const meeting =
|
|
calendarMeeting.meeting ?? params.configuredMeeting ?? params.config.defaults.meeting;
|
|
if (!meeting) {
|
|
throw new Error(
|
|
"Meeting input is required. Pass --meeting, --today, --event, or configure defaults.meeting.",
|
|
);
|
|
}
|
|
return calendarMeeting.calendarEvent
|
|
? { meeting, calendarEvent: calendarMeeting.calendarEvent }
|
|
: { meeting };
|
|
}
|
|
|
|
function resolveCreateTokenOptions(
|
|
config: GoogleMeetConfig,
|
|
options: CreateOptions,
|
|
): {
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
refreshToken?: string;
|
|
accessToken?: string;
|
|
expiresAt?: number;
|
|
} {
|
|
return {
|
|
clientId: options.clientId?.trim() || config.oauth.clientId,
|
|
clientSecret: options.clientSecret?.trim() || config.oauth.clientSecret,
|
|
refreshToken: options.refreshToken?.trim() || config.oauth.refreshToken,
|
|
accessToken: options.accessToken?.trim() || config.oauth.accessToken,
|
|
expiresAt: parseOptionalNumber(options.expiresAt) ?? config.oauth.expiresAt,
|
|
};
|
|
}
|
|
|
|
function resolveArtifactTokenOptions(
|
|
config: GoogleMeetConfig,
|
|
options: MeetArtifactOptions,
|
|
): {
|
|
meeting?: string;
|
|
conferenceRecord?: string;
|
|
clientId?: string;
|
|
clientSecret?: string;
|
|
refreshToken?: string;
|
|
accessToken?: string;
|
|
expiresAt?: number;
|
|
pageSize?: number;
|
|
includeTranscriptEntries?: boolean;
|
|
allConferenceRecords?: boolean;
|
|
includeDocumentBodies?: boolean;
|
|
mergeDuplicateParticipants?: boolean;
|
|
lateAfterMinutes?: number;
|
|
earlyBeforeMinutes?: number;
|
|
} {
|
|
const meeting = options.meeting?.trim() || config.defaults.meeting;
|
|
const conferenceRecord = options.conferenceRecord?.trim();
|
|
if (!meeting && !conferenceRecord && !hasCalendarLookupOptions(options)) {
|
|
throw new Error(
|
|
"Meeting input or conference record is required. Pass --meeting, --today, --event, --conference-record, or configure defaults.meeting.",
|
|
);
|
|
}
|
|
return {
|
|
meeting,
|
|
conferenceRecord,
|
|
clientId: options.clientId?.trim() || config.oauth.clientId,
|
|
clientSecret: options.clientSecret?.trim() || config.oauth.clientSecret,
|
|
refreshToken: options.refreshToken?.trim() || config.oauth.refreshToken,
|
|
accessToken: options.accessToken?.trim() || config.oauth.accessToken,
|
|
expiresAt: parseOptionalNumber(options.expiresAt) ?? config.oauth.expiresAt,
|
|
pageSize: parseOptionalNumber(options.pageSize),
|
|
includeTranscriptEntries: options.transcriptEntries !== false,
|
|
allConferenceRecords: Boolean(options.allConferenceRecords),
|
|
includeDocumentBodies: Boolean(options.includeDocBodies),
|
|
mergeDuplicateParticipants: options.mergeDuplicates !== false,
|
|
lateAfterMinutes: parseOptionalNumber(options.lateAfterMinutes),
|
|
earlyBeforeMinutes: parseOptionalNumber(options.earlyBeforeMinutes),
|
|
};
|
|
}
|
|
|
|
function hasCreateOAuth(config: GoogleMeetConfig, options: CreateOptions): boolean {
|
|
return Boolean(
|
|
options.accessToken?.trim() ||
|
|
options.refreshToken?.trim() ||
|
|
config.oauth.accessToken ||
|
|
config.oauth.refreshToken,
|
|
);
|
|
}
|
|
|
|
function writeArtifactsSummary(result: GoogleMeetArtifactsResult): void {
|
|
if (result.input) {
|
|
writeStdoutLine("input: %s", result.input);
|
|
}
|
|
if (result.space) {
|
|
writeStdoutLine("space: %s", result.space.name);
|
|
}
|
|
writeStdoutLine("conference records: %d", result.conferenceRecords.length);
|
|
for (const entry of result.artifacts) {
|
|
writeStdoutLine("");
|
|
writeStdoutLine("record: %s", entry.conferenceRecord.name);
|
|
writeStdoutLine("started: %s", formatOptional(entry.conferenceRecord.startTime));
|
|
writeStdoutLine("ended: %s", formatOptional(entry.conferenceRecord.endTime));
|
|
writeStdoutLine("participants: %d", entry.participants.length);
|
|
writeStdoutLine("recordings: %d", entry.recordings.length);
|
|
writeStdoutLine("transcripts: %d", entry.transcripts.length);
|
|
writeStdoutLine(
|
|
"transcript entries: %d",
|
|
entry.transcriptEntries.reduce((count, transcript) => count + transcript.entries.length, 0),
|
|
);
|
|
writeStdoutLine("smart notes: %d", entry.smartNotes.length);
|
|
if (entry.smartNotesError) {
|
|
writeStdoutLine("smart notes warning: %s", entry.smartNotesError);
|
|
}
|
|
for (const recording of entry.recordings) {
|
|
writeStdoutLine("- recording: %s", recording.name);
|
|
}
|
|
for (const transcript of entry.transcripts) {
|
|
writeStdoutLine("- transcript: %s", transcript.name);
|
|
if (transcript.documentTextError) {
|
|
writeStdoutLine("- transcript document body warning: %s", transcript.documentTextError);
|
|
}
|
|
}
|
|
for (const transcriptEntries of entry.transcriptEntries) {
|
|
if (transcriptEntries.entriesError) {
|
|
writeStdoutLine(
|
|
"- transcript entries warning: %s: %s",
|
|
transcriptEntries.transcript,
|
|
transcriptEntries.entriesError,
|
|
);
|
|
}
|
|
}
|
|
for (const smartNote of entry.smartNotes) {
|
|
writeStdoutLine("- smart note: %s", smartNote.name);
|
|
if (smartNote.documentTextError) {
|
|
writeStdoutLine("- smart note document body warning: %s", smartNote.documentTextError);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function writeAttendanceSummary(result: GoogleMeetAttendanceResult): void {
|
|
if (result.input) {
|
|
writeStdoutLine("input: %s", result.input);
|
|
}
|
|
if (result.space) {
|
|
writeStdoutLine("space: %s", result.space.name);
|
|
}
|
|
writeStdoutLine("conference records: %d", result.conferenceRecords.length);
|
|
writeStdoutLine("attendance rows: %d", result.attendance.length);
|
|
for (const row of result.attendance) {
|
|
const identity = row.displayName || row.user || row.participant;
|
|
writeStdoutLine("");
|
|
writeStdoutLine("participant: %s", identity);
|
|
writeStdoutLine("record: %s", row.conferenceRecord);
|
|
writeStdoutLine("resource: %s", row.participant);
|
|
writeStdoutLine("participants merged: %d", row.participants?.length ?? 1);
|
|
writeStdoutLine("first joined: %s", formatOptional(row.firstJoinTime ?? row.earliestStartTime));
|
|
writeStdoutLine("last left: %s", formatOptional(row.lastLeaveTime ?? row.latestEndTime));
|
|
writeStdoutLine("duration: %s", formatDuration(row.durationMs));
|
|
writeStdoutLine("late: %s", row.late ? formatDuration(row.lateByMs) : "no");
|
|
writeStdoutLine("early leave: %s", row.earlyLeave ? formatDuration(row.earlyLeaveByMs) : "no");
|
|
writeStdoutLine("sessions: %d", row.sessions.length);
|
|
for (const session of row.sessions) {
|
|
writeStdoutLine(
|
|
"- %s: %s -> %s",
|
|
session.name,
|
|
formatOptional(session.startTime),
|
|
formatOptional(session.endTime),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
function writeLatestConferenceRecordSummary(result: GoogleMeetLatestConferenceRecordResult): void {
|
|
writeStdoutLine("input: %s", result.input);
|
|
writeStdoutLine("space: %s", result.space.name);
|
|
if (!result.conferenceRecord) {
|
|
writeStdoutLine("conference record: none");
|
|
return;
|
|
}
|
|
writeStdoutLine("conference record: %s", result.conferenceRecord.name);
|
|
writeStdoutLine("started: %s", formatOptional(result.conferenceRecord.startTime));
|
|
writeStdoutLine("ended: %s", formatOptional(result.conferenceRecord.endTime));
|
|
}
|
|
|
|
function writeCalendarEventsSummary(
|
|
result: Awaited<ReturnType<typeof listGoogleMeetCalendarEvents>>,
|
|
): void {
|
|
writeStdoutLine("calendar: %s", result.calendarId);
|
|
writeStdoutLine("meet events: %d", result.events.length);
|
|
for (const entry of result.events) {
|
|
writeStdoutLine("");
|
|
writeStdoutLine("%s%s", entry.selected ? "* " : "- ", entry.event.summary ?? "untitled");
|
|
writeStdoutLine("meeting uri: %s", entry.meetingUri);
|
|
writeStdoutLine(
|
|
"starts: %s",
|
|
formatOptional(entry.event.start?.dateTime ?? entry.event.start?.date),
|
|
);
|
|
writeStdoutLine("ends: %s", formatOptional(entry.event.end?.dateTime ?? entry.event.end?.date));
|
|
}
|
|
}
|
|
|
|
function pushMarkdownLine(lines: string[], text = ""): void {
|
|
lines.push(text);
|
|
}
|
|
|
|
function formatMarkdownOptional(value: unknown): string {
|
|
return typeof value === "string" && value.trim() ? value : "n/a";
|
|
}
|
|
|
|
function formatMarkdownIdentity(row: GoogleMeetAttendanceResult["attendance"][number]): string {
|
|
return row.displayName || row.user || row.participant;
|
|
}
|
|
|
|
function participantDisplayName(
|
|
entry: GoogleMeetArtifactsResult["artifacts"][number],
|
|
name: string,
|
|
): string {
|
|
const participant = entry.participants.find((candidate) => candidate.name === name);
|
|
if (!participant) {
|
|
return name;
|
|
}
|
|
return (
|
|
participant.signedinUser?.displayName ??
|
|
participant.anonymousUser?.displayName ??
|
|
participant.phoneUser?.displayName ??
|
|
participant.signedinUser?.user ??
|
|
name
|
|
);
|
|
}
|
|
|
|
function renderArtifactsMarkdown(result: GoogleMeetArtifactsResult): string {
|
|
const lines: string[] = ["# Google Meet Artifacts"];
|
|
if (result.input) {
|
|
pushMarkdownLine(lines, `Input: ${result.input}`);
|
|
}
|
|
if (result.space) {
|
|
pushMarkdownLine(lines, `Space: ${result.space.name}`);
|
|
}
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `Conference records: ${result.conferenceRecords.length}`);
|
|
for (const entry of result.artifacts) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `## ${entry.conferenceRecord.name}`);
|
|
pushMarkdownLine(lines, `Started: ${formatMarkdownOptional(entry.conferenceRecord.startTime)}`);
|
|
pushMarkdownLine(lines, `Ended: ${formatMarkdownOptional(entry.conferenceRecord.endTime)}`);
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `Participants: ${entry.participants.length}`);
|
|
pushMarkdownLine(lines, `Recordings: ${entry.recordings.length}`);
|
|
pushMarkdownLine(lines, `Transcripts: ${entry.transcripts.length}`);
|
|
pushMarkdownLine(
|
|
lines,
|
|
`Transcript entries: ${entry.transcriptEntries.reduce(
|
|
(count, transcript) => count + transcript.entries.length,
|
|
0,
|
|
)}`,
|
|
);
|
|
pushMarkdownLine(lines, `Smart notes: ${entry.smartNotes.length}`);
|
|
const warnings = collectGoogleMeetArtifactWarnings({
|
|
conferenceRecords: [entry.conferenceRecord],
|
|
artifacts: [entry],
|
|
});
|
|
if (warnings.length > 0) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, "### Warnings");
|
|
for (const warning of warnings) {
|
|
const resource = warning.resource ? `${warning.resource}: ` : "";
|
|
pushMarkdownLine(lines, `- ${resource}${warning.message}`);
|
|
}
|
|
}
|
|
if (entry.recordings.length > 0) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, "### Recordings");
|
|
for (const recording of entry.recordings) {
|
|
pushMarkdownLine(lines, `- ${recording.name}`);
|
|
}
|
|
}
|
|
if (entry.transcripts.length > 0) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, "### Transcripts");
|
|
for (const transcript of entry.transcripts) {
|
|
pushMarkdownLine(lines, `- ${transcript.name}`);
|
|
if (transcript.documentTextError) {
|
|
pushMarkdownLine(lines, ` - Document body warning: ${transcript.documentTextError}`);
|
|
} else if (transcript.documentText) {
|
|
pushMarkdownLine(lines, ` - Document body: ${transcript.documentText.length} chars`);
|
|
}
|
|
}
|
|
}
|
|
for (const transcriptEntries of entry.transcriptEntries) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `### Transcript Entries: ${transcriptEntries.transcript}`);
|
|
if (transcriptEntries.entriesError) {
|
|
pushMarkdownLine(lines, `Warning: ${transcriptEntries.entriesError}`);
|
|
continue;
|
|
}
|
|
if (transcriptEntries.entries.length === 0) {
|
|
pushMarkdownLine(lines, "_No transcript entries._");
|
|
continue;
|
|
}
|
|
for (const transcriptEntry of transcriptEntries.entries) {
|
|
const times =
|
|
transcriptEntry.startTime || transcriptEntry.endTime
|
|
? ` (${formatMarkdownOptional(transcriptEntry.startTime)} -> ${formatMarkdownOptional(
|
|
transcriptEntry.endTime,
|
|
)})`
|
|
: "";
|
|
const speaker = transcriptEntry.participant
|
|
? `${participantDisplayName(entry, transcriptEntry.participant)}: `
|
|
: "";
|
|
pushMarkdownLine(lines, `- ${speaker}${transcriptEntry.text ?? ""}${times}`);
|
|
}
|
|
}
|
|
if (entry.smartNotes.length > 0) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, "### Smart Notes");
|
|
for (const smartNote of entry.smartNotes) {
|
|
pushMarkdownLine(lines, `- ${smartNote.name}`);
|
|
if (smartNote.documentTextError) {
|
|
pushMarkdownLine(lines, ` - Document body warning: ${smartNote.documentTextError}`);
|
|
} else if (smartNote.documentText) {
|
|
pushMarkdownLine(lines, ` - Document body: ${smartNote.documentText.length} chars`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return `${lines.join("\n")}\n`;
|
|
}
|
|
|
|
function renderAttendanceMarkdown(result: GoogleMeetAttendanceResult): string {
|
|
const lines: string[] = ["# Google Meet Attendance"];
|
|
if (result.input) {
|
|
pushMarkdownLine(lines, `Input: ${result.input}`);
|
|
}
|
|
if (result.space) {
|
|
pushMarkdownLine(lines, `Space: ${result.space.name}`);
|
|
}
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `Conference records: ${result.conferenceRecords.length}`);
|
|
pushMarkdownLine(lines, `Attendance rows: ${result.attendance.length}`);
|
|
for (const row of result.attendance) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `## ${formatMarkdownIdentity(row)}`);
|
|
pushMarkdownLine(lines, `Record: ${row.conferenceRecord}`);
|
|
pushMarkdownLine(lines, `Resource: ${row.participant}`);
|
|
pushMarkdownLine(lines, `Participants merged: ${row.participants?.length ?? 1}`);
|
|
pushMarkdownLine(
|
|
lines,
|
|
`First joined: ${formatMarkdownOptional(row.firstJoinTime ?? row.earliestStartTime)}`,
|
|
);
|
|
pushMarkdownLine(
|
|
lines,
|
|
`Last left: ${formatMarkdownOptional(row.lastLeaveTime ?? row.latestEndTime)}`,
|
|
);
|
|
pushMarkdownLine(lines, `Duration: ${formatDuration(row.durationMs)}`);
|
|
pushMarkdownLine(lines, `Late: ${row.late ? formatDuration(row.lateByMs) : "no"}`);
|
|
pushMarkdownLine(
|
|
lines,
|
|
`Early leave: ${row.earlyLeave ? formatDuration(row.earlyLeaveByMs) : "no"}`,
|
|
);
|
|
pushMarkdownLine(lines, `Sessions: ${row.sessions.length}`);
|
|
for (const session of row.sessions) {
|
|
pushMarkdownLine(
|
|
lines,
|
|
`- ${session.name}: ${formatMarkdownOptional(session.startTime)} -> ${formatMarkdownOptional(
|
|
session.endTime,
|
|
)}`,
|
|
);
|
|
}
|
|
}
|
|
return `${lines.join("\n")}\n`;
|
|
}
|
|
|
|
function csvCell(value: unknown): string {
|
|
const text =
|
|
value === undefined || value === null
|
|
? ""
|
|
: typeof value === "string" || typeof value === "number" || typeof value === "boolean"
|
|
? String(value)
|
|
: JSON.stringify(value);
|
|
return /[",\n]/.test(text) ? `"${text.replaceAll('"', '""')}"` : text;
|
|
}
|
|
|
|
function renderAttendanceCsv(result: GoogleMeetAttendanceResult): string {
|
|
const rows: unknown[][] = [
|
|
[
|
|
"conferenceRecord",
|
|
"displayName",
|
|
"user",
|
|
"participants",
|
|
"firstJoined",
|
|
"lastLeft",
|
|
"durationMs",
|
|
"sessions",
|
|
"late",
|
|
"lateByMs",
|
|
"earlyLeave",
|
|
"earlyLeaveByMs",
|
|
],
|
|
];
|
|
for (const row of result.attendance) {
|
|
rows.push([
|
|
row.conferenceRecord,
|
|
row.displayName ?? "",
|
|
row.user ?? "",
|
|
(row.participants ?? [row.participant]).join(";"),
|
|
row.firstJoinTime ?? row.earliestStartTime ?? "",
|
|
row.lastLeaveTime ?? row.latestEndTime ?? "",
|
|
row.durationMs ?? "",
|
|
row.sessions.length,
|
|
row.late ?? "",
|
|
row.lateByMs ?? "",
|
|
row.earlyLeave ?? "",
|
|
row.earlyLeaveByMs ?? "",
|
|
]);
|
|
}
|
|
return `${rows.map((row) => row.map(csvCell).join(",")).join("\n")}\n`;
|
|
}
|
|
|
|
function renderTranscriptMarkdown(result: GoogleMeetArtifactsResult): string {
|
|
const lines: string[] = ["# Google Meet Transcript"];
|
|
if (result.input) {
|
|
pushMarkdownLine(lines, `Input: ${result.input}`);
|
|
}
|
|
for (const entry of result.artifacts) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `## ${entry.conferenceRecord.name}`);
|
|
if (entry.transcriptEntries.length === 0) {
|
|
pushMarkdownLine(lines, "_No transcript entries._");
|
|
continue;
|
|
}
|
|
for (const transcriptEntries of entry.transcriptEntries) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `### ${transcriptEntries.transcript}`);
|
|
if (transcriptEntries.entriesError) {
|
|
pushMarkdownLine(lines, `Warning: ${transcriptEntries.entriesError}`);
|
|
continue;
|
|
}
|
|
for (const transcriptEntry of transcriptEntries.entries) {
|
|
const speaker = transcriptEntry.participant
|
|
? participantDisplayName(entry, transcriptEntry.participant)
|
|
: "unknown";
|
|
const time = transcriptEntry.startTime ? ` [${transcriptEntry.startTime}]` : "";
|
|
pushMarkdownLine(lines, `- ${speaker}${time}: ${transcriptEntry.text ?? ""}`);
|
|
}
|
|
}
|
|
const docsTranscripts = entry.transcripts.filter((transcript) => transcript.documentText);
|
|
if (docsTranscripts.length > 0) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, "### Transcript Document Bodies");
|
|
for (const transcript of docsTranscripts) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `#### ${transcript.name}`);
|
|
pushMarkdownLine(lines, transcript.documentText?.trim() || "_Empty document body._");
|
|
}
|
|
}
|
|
const smartNotes = entry.smartNotes.filter((smartNote) => smartNote.documentText);
|
|
if (smartNotes.length > 0) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, "### Smart Note Document Bodies");
|
|
for (const smartNote of smartNotes) {
|
|
pushMarkdownLine(lines);
|
|
pushMarkdownLine(lines, `#### ${smartNote.name}`);
|
|
pushMarkdownLine(lines, smartNote.documentText?.trim() || "_Empty document body._");
|
|
}
|
|
}
|
|
}
|
|
return `${lines.join("\n")}\n`;
|
|
}
|
|
|
|
function collectGoogleMeetArtifactWarnings(
|
|
result: GoogleMeetArtifactsResult,
|
|
): GoogleMeetExportWarning[] {
|
|
const warnings: GoogleMeetExportWarning[] = [];
|
|
for (const entry of result.artifacts) {
|
|
const conferenceRecord = entry.conferenceRecord.name;
|
|
if (entry.smartNotesError) {
|
|
warnings.push({
|
|
type: "smart_notes",
|
|
conferenceRecord,
|
|
message: entry.smartNotesError,
|
|
});
|
|
}
|
|
for (const transcriptEntries of entry.transcriptEntries) {
|
|
if (transcriptEntries.entriesError) {
|
|
warnings.push({
|
|
type: "transcript_entries",
|
|
conferenceRecord,
|
|
resource: transcriptEntries.transcript,
|
|
message: transcriptEntries.entriesError,
|
|
});
|
|
}
|
|
}
|
|
for (const transcript of entry.transcripts) {
|
|
if (transcript.documentTextError) {
|
|
warnings.push({
|
|
type: "transcript_document_body",
|
|
conferenceRecord,
|
|
resource: transcript.name,
|
|
message: transcript.documentTextError,
|
|
});
|
|
}
|
|
}
|
|
for (const smartNote of entry.smartNotes) {
|
|
if (smartNote.documentTextError) {
|
|
warnings.push({
|
|
type: "smart_note_document_body",
|
|
conferenceRecord,
|
|
resource: smartNote.name,
|
|
message: smartNote.documentTextError,
|
|
});
|
|
}
|
|
}
|
|
}
|
|
return warnings;
|
|
}
|
|
|
|
export function buildGoogleMeetExportManifest(params: {
|
|
artifacts: GoogleMeetArtifactsResult;
|
|
attendance: GoogleMeetAttendanceResult;
|
|
files: string[];
|
|
request?: GoogleMeetExportRequest;
|
|
tokenSource?: "cached-access-token" | "refresh-token";
|
|
calendarEvent?: GoogleMeetCalendarLookupResult;
|
|
zipFile?: string;
|
|
}): GoogleMeetExportManifest {
|
|
const transcriptEntryCount = params.artifacts.artifacts.reduce(
|
|
(count, entry) =>
|
|
count +
|
|
entry.transcriptEntries.reduce(
|
|
(entryCount, transcript) => entryCount + transcript.entries.length,
|
|
0,
|
|
),
|
|
0,
|
|
);
|
|
const warnings = collectGoogleMeetArtifactWarnings(params.artifacts);
|
|
return {
|
|
generatedAt: new Date().toISOString(),
|
|
...(params.request ? { request: params.request } : {}),
|
|
...(params.tokenSource ? { tokenSource: params.tokenSource } : {}),
|
|
...(params.calendarEvent ? { calendarEvent: params.calendarEvent } : {}),
|
|
inputs: {
|
|
...(params.artifacts.input ? { artifacts: params.artifacts.input } : {}),
|
|
...(params.attendance.input ? { attendance: params.attendance.input } : {}),
|
|
},
|
|
counts: {
|
|
conferenceRecords: params.artifacts.conferenceRecords.length,
|
|
artifacts: params.artifacts.artifacts.length,
|
|
attendanceRows: params.attendance.attendance.length,
|
|
recordings: params.artifacts.artifacts.reduce(
|
|
(count, entry) => count + entry.recordings.length,
|
|
0,
|
|
),
|
|
transcripts: params.artifacts.artifacts.reduce(
|
|
(count, entry) => count + entry.transcripts.length,
|
|
0,
|
|
),
|
|
transcriptEntries: transcriptEntryCount,
|
|
smartNotes: params.artifacts.artifacts.reduce(
|
|
(count, entry) => count + entry.smartNotes.length,
|
|
0,
|
|
),
|
|
warnings: warnings.length,
|
|
},
|
|
conferenceRecords: params.artifacts.conferenceRecords.map((record) => record.name),
|
|
files: params.files,
|
|
...(params.zipFile ? { zipFile: params.zipFile } : {}),
|
|
warnings,
|
|
};
|
|
}
|
|
|
|
export function googleMeetExportFileNames(): string[] {
|
|
return [
|
|
"summary.md",
|
|
"attendance.csv",
|
|
"transcript.md",
|
|
"artifacts.json",
|
|
"attendance.json",
|
|
"manifest.json",
|
|
];
|
|
}
|
|
|
|
function defaultExportDirectory(): string {
|
|
return `google-meet-export-${new Date().toISOString().replace(/[:.]/g, "-")}`;
|
|
}
|
|
|
|
const CRC32_TABLE = new Uint32Array(
|
|
Array.from({ length: 256 }, (_, index) => {
|
|
let value = index;
|
|
for (let bit = 0; bit < 8; bit += 1) {
|
|
value = value & 1 ? 0xedb88320 ^ (value >>> 1) : value >>> 1;
|
|
}
|
|
return value >>> 0;
|
|
}),
|
|
);
|
|
|
|
function crc32(buffer: Buffer): number {
|
|
let value = 0xffffffff;
|
|
for (const byte of buffer) {
|
|
value = CRC32_TABLE[(value ^ byte) & 0xff] ^ (value >>> 8);
|
|
}
|
|
return (value ^ 0xffffffff) >>> 0;
|
|
}
|
|
|
|
function dosDateTime(date = new Date()): { date: number; time: number } {
|
|
return {
|
|
time: (date.getHours() << 11) | (date.getMinutes() << 5) | Math.floor(date.getSeconds() / 2),
|
|
date: ((date.getFullYear() - 1980) << 9) | ((date.getMonth() + 1) << 5) | date.getDate(),
|
|
};
|
|
}
|
|
|
|
function buildZipArchive(files: Array<{ name: string; content: string }>): Buffer {
|
|
const localParts: Buffer[] = [];
|
|
const centralParts: Buffer[] = [];
|
|
let offset = 0;
|
|
const stamp = dosDateTime();
|
|
for (const file of files) {
|
|
const name = Buffer.from(file.name, "utf8");
|
|
const content = Buffer.from(file.content, "utf8");
|
|
const checksum = crc32(content);
|
|
const local = Buffer.alloc(30);
|
|
local.writeUInt32LE(0x04034b50, 0);
|
|
local.writeUInt16LE(20, 4);
|
|
local.writeUInt16LE(0, 6);
|
|
local.writeUInt16LE(0, 8);
|
|
local.writeUInt16LE(stamp.time, 10);
|
|
local.writeUInt16LE(stamp.date, 12);
|
|
local.writeUInt32LE(checksum, 14);
|
|
local.writeUInt32LE(content.length, 18);
|
|
local.writeUInt32LE(content.length, 22);
|
|
local.writeUInt16LE(name.length, 26);
|
|
local.writeUInt16LE(0, 28);
|
|
localParts.push(local, name, content);
|
|
|
|
const central = Buffer.alloc(46);
|
|
central.writeUInt32LE(0x02014b50, 0);
|
|
central.writeUInt16LE(20, 4);
|
|
central.writeUInt16LE(20, 6);
|
|
central.writeUInt16LE(0, 8);
|
|
central.writeUInt16LE(0, 10);
|
|
central.writeUInt16LE(stamp.time, 12);
|
|
central.writeUInt16LE(stamp.date, 14);
|
|
central.writeUInt32LE(checksum, 16);
|
|
central.writeUInt32LE(content.length, 20);
|
|
central.writeUInt32LE(content.length, 24);
|
|
central.writeUInt16LE(name.length, 28);
|
|
central.writeUInt16LE(0, 30);
|
|
central.writeUInt16LE(0, 32);
|
|
central.writeUInt16LE(0, 34);
|
|
central.writeUInt16LE(0, 36);
|
|
central.writeUInt32LE(0, 38);
|
|
central.writeUInt32LE(offset, 42);
|
|
centralParts.push(central, name);
|
|
offset += local.length + name.length + content.length;
|
|
}
|
|
const centralDirectory = Buffer.concat(centralParts);
|
|
const end = Buffer.alloc(22);
|
|
end.writeUInt32LE(0x06054b50, 0);
|
|
end.writeUInt16LE(0, 4);
|
|
end.writeUInt16LE(0, 6);
|
|
end.writeUInt16LE(files.length, 8);
|
|
end.writeUInt16LE(files.length, 10);
|
|
end.writeUInt32LE(centralDirectory.length, 12);
|
|
end.writeUInt32LE(offset, 16);
|
|
end.writeUInt16LE(0, 20);
|
|
return Buffer.concat([...localParts, centralDirectory, end]);
|
|
}
|
|
|
|
export async function writeMeetExportBundle(params: {
|
|
outputDir?: string;
|
|
artifacts: GoogleMeetArtifactsResult;
|
|
attendance: GoogleMeetAttendanceResult;
|
|
zip?: boolean;
|
|
request?: GoogleMeetExportRequest;
|
|
tokenSource?: "cached-access-token" | "refresh-token";
|
|
calendarEvent?: GoogleMeetCalendarLookupResult;
|
|
}): Promise<{ outputDir: string; files: string[]; zipFile?: string }> {
|
|
const outputDir = params.outputDir?.trim() || defaultExportDirectory();
|
|
await mkdir(outputDir, { recursive: true });
|
|
const zipFile = params.zip ? `${outputDir.replace(/\/$/, "")}.zip` : undefined;
|
|
const fileNames = googleMeetExportFileNames();
|
|
const files = [
|
|
{
|
|
name: "summary.md",
|
|
content: `${renderArtifactsMarkdown(params.artifacts)}\n${renderAttendanceMarkdown(params.attendance)}`,
|
|
},
|
|
{ name: "attendance.csv", content: renderAttendanceCsv(params.attendance) },
|
|
{ name: "transcript.md", content: renderTranscriptMarkdown(params.artifacts) },
|
|
{ name: "artifacts.json", content: `${JSON.stringify(params.artifacts, null, 2)}\n` },
|
|
{ name: "attendance.json", content: `${JSON.stringify(params.attendance, null, 2)}\n` },
|
|
{
|
|
name: "manifest.json",
|
|
content: `${JSON.stringify(
|
|
buildGoogleMeetExportManifest({
|
|
artifacts: params.artifacts,
|
|
attendance: params.attendance,
|
|
files: fileNames,
|
|
...(params.request ? { request: params.request } : {}),
|
|
...(params.tokenSource ? { tokenSource: params.tokenSource } : {}),
|
|
...(params.calendarEvent ? { calendarEvent: params.calendarEvent } : {}),
|
|
...(zipFile ? { zipFile } : {}),
|
|
}),
|
|
null,
|
|
2,
|
|
)}\n`,
|
|
},
|
|
];
|
|
for (const file of files) {
|
|
await writeFile(path.join(outputDir, file.name), file.content, "utf8");
|
|
}
|
|
const result: { outputDir: string; files: string[]; zipFile?: string } = {
|
|
outputDir,
|
|
files: files.map((file) => path.join(outputDir, file.name)),
|
|
};
|
|
if (zipFile) {
|
|
await writeFile(zipFile, buildZipArchive(files));
|
|
result.zipFile = zipFile;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
export function registerGoogleMeetCli(params: {
|
|
program: Command;
|
|
config: GoogleMeetConfig;
|
|
ensureRuntime: () => Promise<GoogleMeetRuntime>;
|
|
callGatewayFromCli?: typeof callGatewayFromCli;
|
|
}) {
|
|
const callGateway = params.callGatewayFromCli ?? callGatewayFromCli;
|
|
const operationTimeoutMs = resolveGoogleMeetGatewayOperationTimeoutMs(params.config);
|
|
const root = params.program
|
|
.command("googlemeet")
|
|
.description("Google Meet participant utilities")
|
|
.addHelpText("after", () => `\nDocs: https://docs.openclaw.ai/plugins/google-meet\n`);
|
|
|
|
const auth = root.command("auth").description("Google Meet OAuth helpers");
|
|
|
|
auth
|
|
.command("login")
|
|
.description("Run a PKCE OAuth flow and print refresh-token JSON to store in plugin config")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--manual", "Use copy/paste callback flow instead of localhost callback")
|
|
.option("--json", "Print the token payload as JSON", false)
|
|
.option("--timeout-sec <n>", "Local callback timeout in seconds", "300")
|
|
.action(async (options: OAuthLoginOptions) => {
|
|
const clientId = options.clientId?.trim() || params.config.oauth.clientId;
|
|
const clientSecret = options.clientSecret?.trim() || params.config.oauth.clientSecret;
|
|
if (!clientId) {
|
|
throw new Error(
|
|
"Missing Google Meet OAuth client id. Configure oauth.clientId or pass --client-id.",
|
|
);
|
|
}
|
|
const { verifier, challenge } = createGoogleMeetPkce();
|
|
const state = createGoogleMeetOAuthState();
|
|
const authUrl = buildGoogleMeetAuthUrl({
|
|
clientId,
|
|
challenge,
|
|
state,
|
|
});
|
|
const code = await waitForGoogleMeetAuthCode({
|
|
state,
|
|
manual: Boolean(options.manual),
|
|
timeoutMs: (parseOptionalNumber(options.timeoutSec) ?? 300) * 1000,
|
|
authUrl,
|
|
promptInput,
|
|
writeLine: (message) => writeStdoutLine("%s", message),
|
|
});
|
|
const tokens = await exchangeGoogleMeetAuthCode({
|
|
clientId,
|
|
clientSecret,
|
|
code,
|
|
verifier,
|
|
});
|
|
if (!tokens.refreshToken) {
|
|
throw new Error(
|
|
"Google OAuth did not return a refresh token. Re-run the flow with consent and offline access.",
|
|
);
|
|
}
|
|
const payload = {
|
|
oauth: {
|
|
clientId,
|
|
...(clientSecret ? { clientSecret } : {}),
|
|
refreshToken: tokens.refreshToken,
|
|
accessToken: tokens.accessToken,
|
|
expiresAt: tokens.expiresAt,
|
|
},
|
|
scope: tokens.scope,
|
|
tokenType: tokens.tokenType,
|
|
};
|
|
if (!options.json) {
|
|
writeStdoutLine("Paste this into plugins.entries.google-meet.config:");
|
|
}
|
|
writeStdoutJson(payload);
|
|
});
|
|
|
|
root
|
|
.command("create")
|
|
.description("Create a new Google Meet space and print its meeting URL")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option(
|
|
"--access-type <type>",
|
|
"Google Meet SpaceConfig accessType for API create: OPEN, TRUSTED, or RESTRICTED",
|
|
)
|
|
.option(
|
|
"--entry-point-access <type>",
|
|
"Google Meet SpaceConfig entryPointAccess for API create: ALL or CREATOR_APP_ONLY",
|
|
)
|
|
.option("--no-join", "Only create the meeting URL; do not join it")
|
|
.option("--transport <transport>", "Join transport: chrome, chrome-node, or twilio")
|
|
.option("--mode <mode>", "Join mode: agent, bidi, or transcribe")
|
|
.option("--message <text>", "Realtime speech to trigger after join")
|
|
.option("--dial-in-number <phone>", "Meet dial-in number for Twilio transport")
|
|
.option("--pin <pin>", "Meet phone PIN; # is appended if omitted")
|
|
.option("--dtmf-sequence <sequence>", "Explicit Twilio DTMF sequence")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: CreateOptions) => {
|
|
if (options.join !== false) {
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.create",
|
|
payload: { ...options },
|
|
timeoutMs: operationTimeoutMs,
|
|
});
|
|
if (delegated.ok) {
|
|
const payload = delegated.payload as {
|
|
browser?: { nodeId?: string };
|
|
joined?: boolean;
|
|
join?: { session?: { id?: string } };
|
|
meetingUri?: string;
|
|
source?: string;
|
|
space?: { name?: string; meetingCode?: string };
|
|
tokenSource?: string;
|
|
};
|
|
if (options.json) {
|
|
writeStdoutJson(payload);
|
|
return;
|
|
}
|
|
writeStdoutLine("meeting uri: %s", payload.meetingUri);
|
|
if (payload.space?.name) {
|
|
writeStdoutLine("space: %s", payload.space.name);
|
|
}
|
|
if (payload.space?.meetingCode) {
|
|
writeStdoutLine("meeting code: %s", payload.space.meetingCode);
|
|
}
|
|
if (payload.source) {
|
|
writeStdoutLine("source: %s", payload.source);
|
|
}
|
|
if (payload.browser?.nodeId) {
|
|
writeStdoutLine("node: %s", payload.browser.nodeId);
|
|
}
|
|
if (payload.tokenSource) {
|
|
writeStdoutLine("token source: %s", payload.tokenSource);
|
|
}
|
|
if (payload.joined && payload.join?.session?.id) {
|
|
writeStdoutLine("joined: %s", payload.join.session.id);
|
|
} else {
|
|
writeStdoutLine("joined: no (run `openclaw googlemeet join %s`)", payload.meetingUri);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
if (!hasCreateOAuth(params.config, options)) {
|
|
if (hasCreateSpaceConfigInput(options as Record<string, unknown>)) {
|
|
throw new Error(
|
|
"Google Meet access policy options require OAuth/API room creation. Configure Google Meet OAuth or remove --access-type/--entry-point-access.",
|
|
);
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
const result = await rt.createViaBrowser();
|
|
const join =
|
|
options.join !== false
|
|
? await rt.join({
|
|
url: result.meetingUri,
|
|
transport: options.transport,
|
|
mode: options.mode,
|
|
message: options.message,
|
|
dialInNumber: options.dialInNumber,
|
|
pin: options.pin,
|
|
dtmfSequence: options.dtmfSequence,
|
|
})
|
|
: undefined;
|
|
const payload = {
|
|
source: result.source,
|
|
meetingUri: result.meetingUri,
|
|
joined: Boolean(join),
|
|
...(join ? { join } : {}),
|
|
browser: {
|
|
nodeId: result.nodeId,
|
|
targetId: result.targetId,
|
|
browserUrl: result.browserUrl,
|
|
browserTitle: result.browserTitle,
|
|
},
|
|
};
|
|
if (options.json) {
|
|
writeStdoutJson(payload);
|
|
return;
|
|
}
|
|
writeStdoutLine("meeting uri: %s", result.meetingUri);
|
|
writeStdoutLine("source: browser");
|
|
writeStdoutLine("node: %s", result.nodeId);
|
|
if (join) {
|
|
writeStdoutLine("joined: %s", join.session.id);
|
|
} else {
|
|
writeStdoutLine("joined: no (run `openclaw googlemeet join %s`)", result.meetingUri);
|
|
}
|
|
return;
|
|
}
|
|
const token = await resolveGoogleMeetAccessToken(
|
|
resolveCreateTokenOptions(params.config, options),
|
|
);
|
|
const result = await createGoogleMeetSpace({
|
|
accessToken: token.accessToken,
|
|
config: resolveCreateSpaceConfig(options as Record<string, unknown>),
|
|
});
|
|
const join =
|
|
options.join !== false
|
|
? await (
|
|
await params.ensureRuntime()
|
|
).join({
|
|
url: result.meetingUri,
|
|
transport: options.transport,
|
|
mode: options.mode,
|
|
message: options.message,
|
|
dialInNumber: options.dialInNumber,
|
|
pin: options.pin,
|
|
dtmfSequence: options.dtmfSequence,
|
|
})
|
|
: undefined;
|
|
if (options.json) {
|
|
writeStdoutJson({
|
|
...result,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
joined: Boolean(join),
|
|
...(join ? { join } : {}),
|
|
});
|
|
return;
|
|
}
|
|
writeStdoutLine("meeting uri: %s", result.meetingUri);
|
|
writeStdoutLine("space: %s", result.space.name);
|
|
if (result.space.meetingCode) {
|
|
writeStdoutLine("meeting code: %s", result.space.meetingCode);
|
|
}
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
if (join) {
|
|
writeStdoutLine("joined: %s", join.session.id);
|
|
} else {
|
|
writeStdoutLine("joined: no (run `openclaw googlemeet join %s`)", result.meetingUri);
|
|
}
|
|
});
|
|
|
|
root
|
|
.command("end-active-conference")
|
|
.description("End the active conference for a Google Meet space")
|
|
.argument("[meeting]", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (meeting: string | undefined, options: ResolveSpaceOptions & JsonOptions) => {
|
|
const token = await resolveGoogleMeetAccessToken(
|
|
resolveOAuthTokenOptions(params.config, options),
|
|
);
|
|
const result = await endGoogleMeetActiveConference({
|
|
accessToken: token.accessToken,
|
|
meeting: resolveMeetingInput(params.config, meeting ?? options.meeting),
|
|
});
|
|
if (options.json) {
|
|
writeStdoutJson({
|
|
...result,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
});
|
|
return;
|
|
}
|
|
writeStdoutLine("space: %s", result.space);
|
|
writeStdoutLine("ended: yes");
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
});
|
|
|
|
root
|
|
.command("join")
|
|
.argument("[url]", "Explicit https://meet.google.com/... URL")
|
|
.option("--transport <transport>", "Transport: chrome, chrome-node, or twilio")
|
|
.option("--mode <mode>", "Mode: agent, bidi, or transcribe")
|
|
.option("--message <text>", "Realtime speech to trigger after join")
|
|
.option("--dial-in-number <phone>", "Meet dial-in number for Twilio transport")
|
|
.option("--pin <pin>", "Meet phone PIN; # is appended if omitted")
|
|
.option("--dtmf-sequence <sequence>", "Explicit Twilio DTMF sequence")
|
|
.action(async (url: string | undefined, options: JoinOptions) => {
|
|
const payload = {
|
|
url: resolveMeetingInput(params.config, url),
|
|
transport: options.transport,
|
|
mode: options.mode,
|
|
message: options.message,
|
|
dialInNumber: options.dialInNumber,
|
|
pin: options.pin,
|
|
dtmfSequence: options.dtmfSequence,
|
|
};
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.join",
|
|
payload,
|
|
timeoutMs: operationTimeoutMs,
|
|
});
|
|
if (delegated.ok) {
|
|
const result = delegated.payload as { session?: unknown };
|
|
writeStdoutJson(result.session ?? delegated.payload);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
const result = await rt.join(payload);
|
|
writeStdoutJson(result.session);
|
|
});
|
|
|
|
root
|
|
.command("test-speech")
|
|
.argument("[url]", "Explicit https://meet.google.com/... URL")
|
|
.option("--transport <transport>", "Transport: chrome, chrome-node, or twilio")
|
|
.option("--mode <mode>", "Mode: agent, bidi, or transcribe")
|
|
.option(
|
|
"--message <text>",
|
|
"Realtime speech to trigger",
|
|
"Say exactly: Google Meet speech test complete.",
|
|
)
|
|
.action(async (url: string | undefined, options: JoinOptions) => {
|
|
const payload = {
|
|
url: resolveMeetingInput(params.config, url),
|
|
transport: options.transport,
|
|
mode: options.mode,
|
|
message: options.message,
|
|
};
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.testSpeech",
|
|
payload,
|
|
timeoutMs: operationTimeoutMs,
|
|
});
|
|
if (delegated.ok) {
|
|
writeStdoutJson(delegated.payload);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
writeStdoutJson(await rt.testSpeech(payload));
|
|
});
|
|
|
|
root
|
|
.command("test-listen")
|
|
.argument("[url]", "Explicit https://meet.google.com/... URL")
|
|
.option("--transport <transport>", "Transport: chrome or chrome-node")
|
|
.option("--timeout-ms <ms>", "How long to wait for fresh captions/transcript movement")
|
|
.action(async (url: string | undefined, options: JoinOptions) => {
|
|
const payload = {
|
|
url: resolveMeetingInput(params.config, url),
|
|
transport: options.transport,
|
|
timeoutMs: parsePositiveNumber(options.timeoutMs, "timeout-ms"),
|
|
};
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.testListen",
|
|
payload,
|
|
timeoutMs: operationTimeoutMs,
|
|
});
|
|
if (delegated.ok) {
|
|
writeStdoutJson(delegated.payload);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
writeStdoutJson(await rt.testListen(payload));
|
|
});
|
|
|
|
root
|
|
.command("resolve-space")
|
|
.description("Resolve a Meet URL, meeting code, or spaces/{id} to its canonical space")
|
|
.option("--meeting <value>", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: ResolveSpaceOptions) => {
|
|
const resolved = resolveTokenOptions(params.config, options);
|
|
const token = await resolveGoogleMeetAccessToken(resolved);
|
|
const space = await fetchGoogleMeetSpace({
|
|
accessToken: token.accessToken,
|
|
meeting: resolved.meeting,
|
|
});
|
|
if (options.json) {
|
|
writeStdoutJson(space);
|
|
return;
|
|
}
|
|
writeStdoutLine("input: %s", resolved.meeting);
|
|
writeStdoutLine("space: %s", space.name);
|
|
if (space.meetingCode) {
|
|
writeStdoutLine("meeting code: %s", space.meetingCode);
|
|
}
|
|
if (space.meetingUri) {
|
|
writeStdoutLine("meeting uri: %s", space.meetingUri);
|
|
}
|
|
writeStdoutLine("active conference: %s", space.activeConference ? "yes" : "no");
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
});
|
|
|
|
root
|
|
.command("preflight")
|
|
.description("Validate OAuth + meeting resolution prerequisites for Meet media work")
|
|
.option("--meeting <value>", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: ResolveSpaceOptions) => {
|
|
const resolved = resolveTokenOptions(params.config, options);
|
|
const token = await resolveGoogleMeetAccessToken(resolved);
|
|
const space = await fetchGoogleMeetSpace({
|
|
accessToken: token.accessToken,
|
|
meeting: resolved.meeting,
|
|
});
|
|
const report = buildGoogleMeetPreflightReport({
|
|
input: resolved.meeting,
|
|
space,
|
|
previewAcknowledged: params.config.preview.enrollmentAcknowledged,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
});
|
|
if (options.json) {
|
|
writeStdoutJson(report);
|
|
return;
|
|
}
|
|
writeStdoutLine("input: %s", report.input);
|
|
writeStdoutLine("resolved space: %s", report.resolvedSpaceName);
|
|
if (report.meetingCode) {
|
|
writeStdoutLine("meeting code: %s", report.meetingCode);
|
|
}
|
|
if (report.meetingUri) {
|
|
writeStdoutLine("meeting uri: %s", report.meetingUri);
|
|
}
|
|
writeStdoutLine("active conference: %s", report.hasActiveConference ? "yes" : "no");
|
|
writeStdoutLine("preview acknowledged: %s", report.previewAcknowledged ? "yes" : "no");
|
|
writeStdoutLine("token source: %s", report.tokenSource);
|
|
if (report.blockers.length === 0) {
|
|
writeStdoutLine("blockers: none");
|
|
return;
|
|
}
|
|
writeStdoutLine("blockers:");
|
|
for (const blocker of report.blockers) {
|
|
writeStdoutLine("- %s", blocker);
|
|
}
|
|
});
|
|
|
|
root
|
|
.command("latest")
|
|
.description("Find the latest Meet conference record for a meeting")
|
|
.option("--meeting <value>", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--today", "Find a Meet link on today's calendar")
|
|
.option("--event <query>", "Find a matching calendar event with a Meet link")
|
|
.option("--calendar <id>", "Calendar id for --today or --event", "primary")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: ResolveSpaceOptions) => {
|
|
const token = await resolveGoogleMeetAccessToken(
|
|
resolveOAuthTokenOptions(params.config, options),
|
|
);
|
|
const resolved = await resolveMeetingForToken({
|
|
config: params.config,
|
|
options,
|
|
accessToken: token.accessToken,
|
|
configuredMeeting: options.meeting?.trim(),
|
|
});
|
|
const result = await fetchLatestGoogleMeetConferenceRecord({
|
|
accessToken: token.accessToken,
|
|
meeting: resolved.meeting,
|
|
});
|
|
if (options.json) {
|
|
writeStdoutJson({
|
|
...result,
|
|
...(resolved.calendarEvent ? { calendarEvent: resolved.calendarEvent } : {}),
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
});
|
|
return;
|
|
}
|
|
if (resolved.calendarEvent) {
|
|
writeStdoutLine("calendar event: %s", resolved.calendarEvent.event.summary ?? "untitled");
|
|
writeStdoutLine("calendar meet: %s", resolved.calendarEvent.meetingUri);
|
|
}
|
|
writeLatestConferenceRecordSummary(result);
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
});
|
|
|
|
root
|
|
.command("calendar-events")
|
|
.description("Preview Calendar events with Google Meet links")
|
|
.option("--today", "Find Meet links on today's calendar")
|
|
.option("--event <query>", "Find matching calendar events with Meet links")
|
|
.option("--calendar <id>", "Calendar id for lookup", "primary")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: ResolveSpaceOptions) => {
|
|
const token = await resolveGoogleMeetAccessToken(
|
|
resolveOAuthTokenOptions(params.config, options),
|
|
);
|
|
const window = options.today ? buildGoogleMeetCalendarDayWindow() : {};
|
|
const result = await listGoogleMeetCalendarEvents({
|
|
accessToken: token.accessToken,
|
|
calendarId: options.calendar,
|
|
eventQuery: options.event,
|
|
...window,
|
|
});
|
|
const payload = {
|
|
...result,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
};
|
|
if (options.json) {
|
|
writeStdoutJson(payload);
|
|
return;
|
|
}
|
|
writeCalendarEventsSummary(result);
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
});
|
|
|
|
root
|
|
.command("artifacts")
|
|
.description("List Meet conference records and available participant/artifact metadata")
|
|
.option("--meeting <value>", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--conference-record <name>", "Conference record name or id")
|
|
.option("--today", "Find a Meet link on today's calendar")
|
|
.option("--event <query>", "Find a matching calendar event with a Meet link")
|
|
.option("--calendar <id>", "Calendar id for --today or --event", "primary")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--page-size <n>", "Max resources per Meet API page")
|
|
.option("--all-conference-records", "Fetch every conference record for --meeting")
|
|
.option("--no-transcript-entries", "Skip structured transcript entry lookup")
|
|
.option("--include-doc-bodies", "Export linked transcript and smart-note Google Docs text")
|
|
.option("--format <format>", "Output format: summary or markdown", "summary")
|
|
.option("--output <path>", "Write output to a file instead of stdout")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: MeetArtifactOptions) => {
|
|
const resolved = resolveArtifactTokenOptions(params.config, options);
|
|
const token = await resolveGoogleMeetAccessToken(resolved);
|
|
const meeting = resolved.conferenceRecord
|
|
? resolved.meeting
|
|
: (
|
|
await resolveMeetingForToken({
|
|
config: params.config,
|
|
options,
|
|
accessToken: token.accessToken,
|
|
configuredMeeting: resolved.meeting,
|
|
})
|
|
).meeting;
|
|
const result = await fetchGoogleMeetArtifacts({
|
|
accessToken: token.accessToken,
|
|
meeting,
|
|
conferenceRecord: resolved.conferenceRecord,
|
|
pageSize: resolved.pageSize,
|
|
includeTranscriptEntries: resolved.includeTranscriptEntries,
|
|
allConferenceRecords: resolved.allConferenceRecords,
|
|
includeDocumentBodies: resolved.includeDocumentBodies,
|
|
});
|
|
if (options.json) {
|
|
await writeCliOutput(
|
|
options,
|
|
JSON.stringify(
|
|
{
|
|
...result,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
},
|
|
null,
|
|
2,
|
|
),
|
|
);
|
|
return;
|
|
}
|
|
if (options.format === "markdown") {
|
|
await writeCliOutput(options, renderArtifactsMarkdown(result));
|
|
return;
|
|
}
|
|
if (options.format && options.format !== "summary") {
|
|
throw new Error("Unsupported format. Expected summary or markdown.");
|
|
}
|
|
writeArtifactsSummary(result);
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
});
|
|
|
|
root
|
|
.command("attendance")
|
|
.description("List Meet participants and participant sessions")
|
|
.option("--meeting <value>", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--conference-record <name>", "Conference record name or id")
|
|
.option("--today", "Find a Meet link on today's calendar")
|
|
.option("--event <query>", "Find a matching calendar event with a Meet link")
|
|
.option("--calendar <id>", "Calendar id for --today or --event", "primary")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--page-size <n>", "Max resources per Meet API page")
|
|
.option("--all-conference-records", "Fetch every conference record for --meeting")
|
|
.option("--no-merge-duplicates", "Keep duplicate participant resources as separate rows")
|
|
.option("--late-after-minutes <n>", "Mark participants late after this many minutes", "5")
|
|
.option("--early-before-minutes <n>", "Mark early leavers before this many minutes", "5")
|
|
.option("--format <format>", "Output format: summary, markdown, or csv", "summary")
|
|
.option("--output <path>", "Write output to a file instead of stdout")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: MeetArtifactOptions) => {
|
|
const resolved = resolveArtifactTokenOptions(params.config, options);
|
|
const token = await resolveGoogleMeetAccessToken(resolved);
|
|
const meeting = resolved.conferenceRecord
|
|
? resolved.meeting
|
|
: (
|
|
await resolveMeetingForToken({
|
|
config: params.config,
|
|
options,
|
|
accessToken: token.accessToken,
|
|
configuredMeeting: resolved.meeting,
|
|
})
|
|
).meeting;
|
|
const result = await fetchGoogleMeetAttendance({
|
|
accessToken: token.accessToken,
|
|
meeting,
|
|
conferenceRecord: resolved.conferenceRecord,
|
|
pageSize: resolved.pageSize,
|
|
allConferenceRecords: resolved.allConferenceRecords,
|
|
mergeDuplicateParticipants: resolved.mergeDuplicateParticipants,
|
|
lateAfterMinutes: resolved.lateAfterMinutes,
|
|
earlyBeforeMinutes: resolved.earlyBeforeMinutes,
|
|
});
|
|
if (options.json) {
|
|
await writeCliOutput(
|
|
options,
|
|
JSON.stringify(
|
|
{
|
|
...result,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
},
|
|
null,
|
|
2,
|
|
),
|
|
);
|
|
return;
|
|
}
|
|
if (options.format === "markdown") {
|
|
await writeCliOutput(options, renderAttendanceMarkdown(result));
|
|
return;
|
|
}
|
|
if (options.format === "csv") {
|
|
await writeCliOutput(options, renderAttendanceCsv(result));
|
|
return;
|
|
}
|
|
if (options.format && options.format !== "summary") {
|
|
throw new Error("Unsupported format. Expected summary, markdown, or csv.");
|
|
}
|
|
writeAttendanceSummary(result);
|
|
writeStdoutLine(
|
|
"token source: %s",
|
|
token.refreshed ? "refresh-token" : "cached-access-token",
|
|
);
|
|
});
|
|
|
|
root
|
|
.command("export")
|
|
.description("Write Meet artifacts, attendance, transcript, and raw JSON into a folder")
|
|
.option("--meeting <value>", "Meet URL, meeting code, or spaces/{id}")
|
|
.option("--conference-record <name>", "Conference record name or id")
|
|
.option("--today", "Find a Meet link on today's calendar")
|
|
.option("--event <query>", "Find a matching calendar event with a Meet link")
|
|
.option("--calendar <id>", "Calendar id for --today or --event", "primary")
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--page-size <n>", "Max resources per Meet API page")
|
|
.option("--all-conference-records", "Fetch every conference record for --meeting")
|
|
.option("--no-transcript-entries", "Skip structured transcript entry lookup")
|
|
.option("--include-doc-bodies", "Export linked transcript and smart-note Google Docs text")
|
|
.option("--no-merge-duplicates", "Keep duplicate participant resources as separate rows")
|
|
.option("--late-after-minutes <n>", "Mark participants late after this many minutes", "5")
|
|
.option("--early-before-minutes <n>", "Mark early leavers before this many minutes", "5")
|
|
.option("--output <dir>", "Output directory")
|
|
.option("--zip", "Also write a portable .zip archive")
|
|
.option("--dry-run", "Fetch export data and print the manifest without writing files", false)
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: MeetArtifactOptions) => {
|
|
const resolved = resolveArtifactTokenOptions(params.config, options);
|
|
const token = await resolveGoogleMeetAccessToken(resolved);
|
|
const meetingResult: { meeting?: string; calendarEvent?: GoogleMeetCalendarLookupResult } =
|
|
resolved.conferenceRecord
|
|
? { meeting: resolved.meeting }
|
|
: await resolveMeetingForToken({
|
|
config: params.config,
|
|
options,
|
|
accessToken: token.accessToken,
|
|
configuredMeeting: resolved.meeting,
|
|
});
|
|
const artifacts = await fetchGoogleMeetArtifacts({
|
|
accessToken: token.accessToken,
|
|
meeting: meetingResult.meeting,
|
|
conferenceRecord: resolved.conferenceRecord,
|
|
pageSize: resolved.pageSize,
|
|
includeTranscriptEntries: resolved.includeTranscriptEntries,
|
|
allConferenceRecords: resolved.allConferenceRecords,
|
|
includeDocumentBodies: resolved.includeDocumentBodies,
|
|
});
|
|
const attendance = await fetchGoogleMeetAttendance({
|
|
accessToken: token.accessToken,
|
|
meeting: meetingResult.meeting,
|
|
conferenceRecord: resolved.conferenceRecord,
|
|
pageSize: resolved.pageSize,
|
|
allConferenceRecords: resolved.allConferenceRecords,
|
|
mergeDuplicateParticipants: resolved.mergeDuplicateParticipants,
|
|
lateAfterMinutes: resolved.lateAfterMinutes,
|
|
earlyBeforeMinutes: resolved.earlyBeforeMinutes,
|
|
});
|
|
const resolvedMeeting = meetingResult.meeting ?? resolved.meeting;
|
|
const request: GoogleMeetExportRequest = {
|
|
...(resolvedMeeting ? { meeting: resolvedMeeting } : {}),
|
|
...(resolved.conferenceRecord ? { conferenceRecord: resolved.conferenceRecord } : {}),
|
|
...(meetingResult.calendarEvent?.event.id
|
|
? { calendarEventId: meetingResult.calendarEvent.event.id }
|
|
: {}),
|
|
...(meetingResult.calendarEvent?.event.summary
|
|
? { calendarEventSummary: meetingResult.calendarEvent.event.summary }
|
|
: {}),
|
|
...(options.calendar ? { calendarId: options.calendar } : {}),
|
|
...(resolved.pageSize !== undefined ? { pageSize: resolved.pageSize } : {}),
|
|
includeTranscriptEntries: resolved.includeTranscriptEntries,
|
|
includeDocumentBodies: resolved.includeDocumentBodies,
|
|
allConferenceRecords: resolved.allConferenceRecords,
|
|
mergeDuplicateParticipants: resolved.mergeDuplicateParticipants,
|
|
...(resolved.lateAfterMinutes !== undefined
|
|
? { lateAfterMinutes: resolved.lateAfterMinutes }
|
|
: {}),
|
|
...(resolved.earlyBeforeMinutes !== undefined
|
|
? { earlyBeforeMinutes: resolved.earlyBeforeMinutes }
|
|
: {}),
|
|
};
|
|
if (options.dryRun) {
|
|
writeStdoutJson({
|
|
dryRun: true,
|
|
manifest: buildGoogleMeetExportManifest({
|
|
artifacts,
|
|
attendance,
|
|
files: googleMeetExportFileNames(),
|
|
request,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
...(meetingResult.calendarEvent ? { calendarEvent: meetingResult.calendarEvent } : {}),
|
|
}),
|
|
...(meetingResult.calendarEvent ? { calendarEvent: meetingResult.calendarEvent } : {}),
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
});
|
|
return;
|
|
}
|
|
const bundle = await writeMeetExportBundle({
|
|
outputDir: options.output,
|
|
artifacts,
|
|
attendance,
|
|
zip: Boolean(options.zip),
|
|
request,
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
...(meetingResult.calendarEvent ? { calendarEvent: meetingResult.calendarEvent } : {}),
|
|
});
|
|
const payload = {
|
|
...bundle,
|
|
...(meetingResult.calendarEvent ? { calendarEvent: meetingResult.calendarEvent } : {}),
|
|
tokenSource: token.refreshed ? "refresh-token" : "cached-access-token",
|
|
};
|
|
if (options.json) {
|
|
writeStdoutJson(payload);
|
|
return;
|
|
}
|
|
writeStdoutLine("export: %s", bundle.outputDir);
|
|
for (const file of bundle.files) {
|
|
writeStdoutLine("- %s", file);
|
|
}
|
|
if (bundle.zipFile) {
|
|
writeStdoutLine("zip: %s", bundle.zipFile);
|
|
}
|
|
});
|
|
|
|
root
|
|
.command("status")
|
|
.argument("[session-id]", "Meet session ID")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (sessionId?: string) => {
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.status",
|
|
payload: { sessionId },
|
|
});
|
|
if (delegated.ok) {
|
|
writeStdoutJson(delegated.payload);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
writeStdoutJson(await rt.status(sessionId));
|
|
});
|
|
|
|
root
|
|
.command("doctor")
|
|
.description("Show human-readable Meet session/browser/realtime health")
|
|
.argument("[session-id]", "Meet session ID")
|
|
.option("--oauth", "Verify Google Meet OAuth token refresh without printing secrets", false)
|
|
.option("--meeting <value>", "Also verify spaces.get for a Meet URL, code, or spaces/{id}")
|
|
.option("--create-space", "Also verify spaces.create by creating a throwaway Meet space", false)
|
|
.option("--access-token <token>", "Access token override")
|
|
.option("--refresh-token <token>", "Refresh token override")
|
|
.option("--client-id <id>", "OAuth client id override")
|
|
.option("--client-secret <secret>", "OAuth client secret override")
|
|
.option("--expires-at <ms>", "Cached access token expiry as unix epoch milliseconds")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (sessionId: string | undefined, options: DoctorOptions) => {
|
|
if (options.oauth) {
|
|
const report = await buildOAuthDoctorReport(params.config, options);
|
|
if (options.json) {
|
|
writeStdoutJson(report);
|
|
return;
|
|
}
|
|
writeOAuthDoctorReport(report);
|
|
return;
|
|
}
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.status",
|
|
payload: { sessionId },
|
|
});
|
|
if (delegated.ok) {
|
|
const status = delegated.payload as Awaited<ReturnType<GoogleMeetRuntime["status"]>>;
|
|
if (options.json) {
|
|
writeStdoutJson(status);
|
|
return;
|
|
}
|
|
writeDoctorStatus(status);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
const status = await rt.status(sessionId);
|
|
if (options.json) {
|
|
writeStdoutJson(status);
|
|
return;
|
|
}
|
|
writeDoctorStatus(status);
|
|
});
|
|
|
|
root
|
|
.command("recover-tab")
|
|
.description("Focus and inspect an existing Google Meet tab")
|
|
.argument("[url]", "Optional Meet URL to match")
|
|
.option("--transport <transport>", "Transport to inspect: chrome or chrome-node")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (url: string | undefined, options: RecoverTabOptions) => {
|
|
const rt = await params.ensureRuntime();
|
|
const result = await rt.recoverCurrentTab({ url, transport: options.transport });
|
|
if (options.json) {
|
|
writeStdoutJson(result);
|
|
return;
|
|
}
|
|
writeRecoverCurrentTabResult(result);
|
|
});
|
|
|
|
root
|
|
.command("setup")
|
|
.description("Show Google Meet transport setup status")
|
|
.option("--transport <transport>", "Transport to check: chrome, chrome-node, or twilio")
|
|
.option("--mode <mode>", "Mode to check: agent, bidi, or transcribe")
|
|
.option("--json", "Print JSON output", false)
|
|
.action(async (options: SetupOptions) => {
|
|
const rt = await params.ensureRuntime();
|
|
const status = await rt.setupStatus({ transport: options.transport, mode: options.mode });
|
|
if (options.json) {
|
|
writeStdoutJson(status);
|
|
return;
|
|
}
|
|
writeSetupStatus(status);
|
|
});
|
|
|
|
root
|
|
.command("leave")
|
|
.argument("<session-id>", "Meet session ID")
|
|
.action(async (sessionId: string) => {
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.leave",
|
|
payload: { sessionId },
|
|
});
|
|
if (delegated.ok) {
|
|
const result = delegated.payload as { found?: boolean };
|
|
if (!result.found) {
|
|
throw new Error("session not found");
|
|
}
|
|
writeStdoutLine("left %s", sessionId);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
const result = await rt.leave(sessionId);
|
|
if (!result.found) {
|
|
throw new Error("session not found");
|
|
}
|
|
writeStdoutLine("left %s", sessionId);
|
|
});
|
|
|
|
root
|
|
.command("speak")
|
|
.argument("<session-id>", "Meet session ID")
|
|
.argument("[message]", "Realtime instructions to speak now")
|
|
.action(async (sessionId: string, message?: string) => {
|
|
const delegated = await callGoogleMeetGateway({
|
|
callGateway,
|
|
method: "googlemeet.speak",
|
|
payload: { sessionId, message },
|
|
});
|
|
if (delegated.ok) {
|
|
const result = delegated.payload as Awaited<ReturnType<GoogleMeetRuntime["speak"]>>;
|
|
if (!result.found) {
|
|
throw new Error("session not found");
|
|
}
|
|
if (!result.spoken) {
|
|
throw new Error(
|
|
result.session?.chrome?.health?.speechBlockedMessage ??
|
|
"session has no active realtime audio bridge",
|
|
);
|
|
}
|
|
writeStdoutLine("speaking on %s", sessionId);
|
|
return;
|
|
}
|
|
const rt = await params.ensureRuntime();
|
|
const result = await rt.speak(sessionId, message);
|
|
if (!result.found) {
|
|
throw new Error("session not found");
|
|
}
|
|
if (!result.spoken) {
|
|
throw new Error(
|
|
result.session?.chrome?.health?.speechBlockedMessage ??
|
|
"session has no active realtime audio bridge",
|
|
);
|
|
}
|
|
writeStdoutLine("speaking on %s", sessionId);
|
|
});
|
|
}
|