Skip to main content
Glama
MIT License
27,120
19,787
  • Linux
  • Apple
fileRead.ts•2.36 kB
import * as fs from 'node:fs/promises'; import iconv from 'iconv-lite'; import { isBinary } from 'istextorbinary'; import jschardet from 'jschardet'; import { logger } from '../../shared/logger.js'; export type FileSkipReason = 'binary-extension' | 'binary-content' | 'size-limit' | 'encoding-error'; export interface FileReadResult { content: string | null; skippedReason?: FileSkipReason; } /** * Read a file and return its text content * @param filePath Path to the file * @param maxFileSize Maximum file size in bytes * @returns File content as string and skip reason if file was skipped */ export const readRawFile = async (filePath: string, maxFileSize: number): Promise<FileReadResult> => { try { const stats = await fs.stat(filePath); if (stats.size > maxFileSize) { const sizeKB = (stats.size / 1024).toFixed(1); const maxSizeKB = (maxFileSize / 1024).toFixed(1); logger.trace(`File exceeds size limit: ${sizeKB}KB > ${maxSizeKB}KB (${filePath})`); return { content: null, skippedReason: 'size-limit' }; } if (isBinary(filePath)) { logger.debug(`Skipping binary file: ${filePath}`); return { content: null, skippedReason: 'binary-extension' }; } logger.trace(`Reading file: ${filePath}`); const buffer = await fs.readFile(filePath); if (isBinary(null, buffer)) { logger.debug(`Skipping binary file (content check): ${filePath}`); return { content: null, skippedReason: 'binary-content' }; } const { encoding: detectedEncoding, confidence } = jschardet.detect(buffer) ?? {}; const encoding = detectedEncoding && iconv.encodingExists(detectedEncoding) ? detectedEncoding : 'utf-8'; const content = iconv.decode(buffer, encoding, { stripBOM: true }); // Heuristics: U+FFFD indicates decode errors; very low confidence implies unreliable guess. if (content.includes('\uFFFD') || (typeof confidence === 'number' && confidence < 0.2)) { logger.debug( `Skipping file due to encoding errors (${encoding}, confidence=${(confidence ?? 0).toFixed(2)}): ${filePath}`, ); return { content: null, skippedReason: 'encoding-error' }; } return { content }; } catch (error) { logger.warn(`Failed to read file: ${filePath}`, error); return { content: null, skippedReason: 'encoding-error' }; } };

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/yamadashy/repomix'

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