Skip to main content
Glama

Fonoster MCP Server

Official
by fonoster
MIT License
118
7,325
  • Apple
  • Linux
httpBridge.ts3.23 kB
/** * Copyright (C) 2025 by Fonoster Inc (https://fonoster.com) * http://github.com/fonoster/fonoster * * This file is part of Fonoster * * Licensed under the MIT License (the "License"); * you may not use this file except in compliance with * the License. You may obtain a copy of the License at * * https://opensource.org/licenses/MIT * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import fs from "fs"; import path from "path"; import { Readable } from "stream"; import { createUpdateMembershipStatus, IdentityConfig } from "@fonoster/identity"; import { getLogger } from "@fonoster/logger"; import express, { Request, Response } from "express"; import { APP_URL } from "../envs"; const logger = getLogger({ service: "apiserver", filePath: __filename }); const CONTENT_TYPE = "audio/L16;rate=16000;channels=1"; function httpBridge(identityConfig: IdentityConfig, params: { port: number }) { const { port } = params; const app = express(); const streamMap = new Map<string, Readable>(); app.get("/api/sounds/:id", (req: Request, res: Response) => { const idWithoutExtension = req.params.id.split(".")[0]; const stream = streamMap.get(idWithoutExtension); if (!stream) { res.status(404).send(`Stream not found for id: ${req.params.id}`); return; } res.setHeader("content-type", CONTENT_TYPE); stream.on("error", (error) => { logger.error(`error reading file: ${error.message}`); if (!res.headersSent) { res.status(500).send("Error reading file!"); } else { res.end(); } }); stream.on("end", () => { res.end(); }); stream.on("close", () => { res.end(); }); stream.pipe(res); }); app.get( "/api/identity/accept-invite", async (req: Request, res: Response) => { try { await createUpdateMembershipStatus(identityConfig)( req.query.token as string ); res.redirect(APP_URL); } catch (error) { logger.verbose("error updating membership status", error); res.redirect(identityConfig.workspaceInviteFailUrl); } } ); app.get("/api/recordings/:id", (req: Request, res: Response) => { const RECORDINGS_PATH = path.join( "/opt/fonoster/recordings", req.params.id ); const wave = fs.readFileSync(RECORDINGS_PATH); res.setHeader("content-type", "audio/wav"); res.send(wave); }); app.listen(port, () => { logger.info(`HTTP bridge is running on port ${port}`); }); return { addStream: (id: string, stream: Readable) => { streamMap.set(id, stream); }, removeStream: (id: string) => { logger.verbose(`removing stream with id: ${id}`); const stream = streamMap.get(id); if (stream) { stream.destroy(); } streamMap.delete(id); }, getStream: (id: string) => { return streamMap.get(id); } }; } export { httpBridge };

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/fonoster/fonoster'

If you have feedback or need assistance with the MCP directory API, please join our Discord server