UNPKG

edge-tts-universal

Version:

Universal text-to-speech library using Microsoft Edge's online TTS service. Works in Node.js and browsers WITHOUT needing Microsoft Edge, Windows, or an API key

355 lines (262 loc) • 10.9 kB
# Edge TTS Universal 🪐 [![API Documentation](https://img.shields.io/badge/API-Documentation-blue)](./API.md) [![npm](https://img.shields.io/npm/v/edge-tts-universal)](https://www.npmjs.com/package/edge-tts-universal) [![JSR](https://jsr.io/badges/@edge-tts/universal)](https://jsr.io/@edge-tts/universal) [![Bundle Size](https://img.shields.io/bundlephobia/minzip/edge-tts-universal)](https://bundlephobia.com/package/edge-tts-universal) [![Downloads](https://img.shields.io/npm/dm/edge-tts-universal)](https://www.npmjs.com/package/edge-tts-universal) [![Tree Shaking](https://img.shields.io/badge/tree%20shaking-āœ“-green)](#bundle-optimization) [![JSR Score](https://jsr.io/badges/@edge-tts/universal/score)](https://jsr.io/@edge-tts/universal) This is a **universal** TypeScript conversion of the Python [`edge-tts`](https://github.com/rany2/edge-tts) library. It allows you to use Microsoft Edge's online text-to-speech service from **Node.js, browsers, and any JavaScript environment**. ## 🌟 Universal Features - **šŸš€ Multiple Entry Points**: Choose between Node.js, browser, or isomorphic APIs - **🌐 Cross-Platform**: Works in Node.js, browsers, Deno, Bun, and edge runtimes - **šŸ“¦ Tree-Shakable**: Import only what you need for optimal bundle size - **šŸ”„ Isomorphic**: Same API works across all environments - **⚔ Zero Dependencies**: Browser builds have no external dependencies - **šŸ›”ļø Type Safe**: Full TypeScript support with comprehensive type definitions This package provides high fidelity to the original Python implementation, replicating the specific headers and WebSocket communication necessary to interact with Microsoft's service. ## Installation ```bash npm install edge-tts-universal # or yarn add edge-tts-universal ``` ## Universal Usage This package provides three usage patterns for maximum compatibility: ### šŸ“¦ Default Import (Node.js + Isomorphic APIs) ```typescript import { EdgeTTS, Communicate, IsomorphicCommunicate, } from 'edge-tts-universal'; ``` ### 🌐 Browser-Only Import (No Node.js dependencies) ```typescript import { EdgeTTS, Communicate } from 'edge-tts-universal/browser'; ``` ### šŸš€ Isomorphic-Only Import (Universal compatibility) ```typescript import { EdgeTTS, Communicate } from 'edge-tts-universal/isomorphic'; ``` ### šŸ‘· Web Worker Import (For background processing) ```typescript import { EdgeTTS, Communicate, postAudioMessage, } from 'edge-tts-universal/webworker'; ``` ### 🌐 CDN Import (No build step required) ```html <!-- Via unpkg --> <script type="module"> import { EdgeTTS } from 'https://unpkg.com/edge-tts-universal/dist/browser.js'; </script> <!-- Via jsdelivr --> <script type="module"> import { EdgeTTS } from 'https://cdn.jsdelivr.net/npm/edge-tts-universal/dist/browser.js'; </script> ``` ## Bundle Optimization Choose the right entry point for optimal bundle size: | Entry Point | Bundle Size\* | Use Case | Dependencies | | ------------------------------- | ------------- | -------------- | --------------- | | `edge-tts-universal` | ~46KB | Node.js apps | All deps | | `edge-tts-universal/browser` | ~30KB | Browser apps | Zero deps | | `edge-tts-universal/isomorphic` | ~36KB | Universal apps | Isomorphic deps | | `edge-tts-universal/webworker` | ~36KB | Web Workers | Isomorphic deps | _\* Minified + Gzipped estimates_ ## Quick Start ### Simple API (Recommended for most use cases) ```typescript import { EdgeTTS } from 'edge-tts-universal'; import fs from 'fs/promises'; // Simple one-shot synthesis const tts = new EdgeTTS('Hello, world!', 'en-US-EmmaMultilingualNeural'); const result = await tts.synthesize(); // Save audio file const audioBuffer = Buffer.from(await result.audio.arrayBuffer()); await fs.writeFile('output.mp3', audioBuffer); ``` ### Advanced Streaming API (For real-time processing) ```typescript import { Communicate } from 'edge-tts-universal'; import fs from 'fs/promises'; const communicate = new Communicate('Hello, world!', { voice: 'en-US-EmmaMultilingualNeural', }); const buffers: Buffer[] = []; for await (const chunk of communicate.stream()) { if (chunk.type === 'audio' && chunk.data) { buffers.push(chunk.data); } } await fs.writeFile('output.mp3', Buffer.concat(buffers)); ``` ### Isomorphic API (Universal - works in Node.js and browsers) ```typescript import { IsomorphicCommunicate } from 'edge-tts-universal'; // Works in both Node.js and browsers (subject to CORS policy) const communicate = new IsomorphicCommunicate('Hello, universal world!', { voice: 'en-US-EmmaMultilingualNeural', }); const audioChunks: Buffer[] = []; for await (const chunk of communicate.stream()) { if (chunk.type === 'audio' && chunk.data) { audioChunks.push(chunk.data); } } // Environment-specific handling const isNode = typeof process !== 'undefined' && process.versions?.node; if (isNode) { // Node.js - save to file const fs = await import('fs/promises'); await fs.writeFile('output.mp3', Buffer.concat(audioChunks)); } else { // Browser - create audio element const audioBlob = new Blob(audioChunks, { type: 'audio/mpeg' }); const audioUrl = URL.createObjectURL(audioBlob); // Use audioUrl with <audio> element } ``` ## Usage Examples <details> <summary><strong>⚔ Simple API Usage (Recommended)</strong></summary> Here's how to use the simple, promise-based API for quick synthesis: ```typescript // examples/simple-api.ts import { EdgeTTS, createVTT, createSRT } from 'edge-tts-universal'; import { promises as fs } from 'fs'; import path from 'path'; const TEXT = 'Hello, world! This is a test of the simple edge-tts API.'; const VOICE = 'en-US-EmmaMultilingualNeural'; const OUTPUT_FILE = path.join(__dirname, 'simple-test.mp3'); async function main() { // Create TTS instance with prosody options const tts = new EdgeTTS(TEXT, VOICE, { rate: '+10%', volume: '+0%', pitch: '+0Hz', }); try { // Synthesize speech (one-shot) const result = await tts.synthesize(); // Save audio file const audioBuffer = Buffer.from(await result.audio.arrayBuffer()); await fs.writeFile(OUTPUT_FILE, audioBuffer); // Generate subtitle files const vttContent = createVTT(result.subtitle); const srtContent = createSRT(result.subtitle); await fs.writeFile('subtitles.vtt', vttContent); await fs.writeFile('subtitles.srt', srtContent); console.log(`Audio saved to ${OUTPUT_FILE}`); console.log(`Generated ${result.subtitle.length} word boundaries`); } catch (error) { console.error('Synthesis failed:', error); } } main().catch(console.error); ``` </details> <details> <summary><strong>šŸ“ Advanced Streaming Usage</strong></summary> Here is an example using the advanced streaming API for real-time processing: ```typescript // examples/streaming.ts import { Communicate } from 'edge-tts-universal'; import { promises as fs } from 'fs'; import path from 'path'; const TEXT = 'Hello, world! This is a test of the new edge-tts Node.js library.'; const VOICE = 'en-US-EmmaMultilingualNeural'; const OUTPUT_FILE = path.join(__dirname, 'test.mp3'); async function main() { const communicate = new Communicate(TEXT, { voice: VOICE }); const buffers: Buffer[] = []; for await (const chunk of communicate.stream()) { if (chunk.type === 'audio' && chunk.data) { buffers.push(chunk.data); } } const finalBuffer = Buffer.concat(buffers); await fs.writeFile(OUTPUT_FILE, finalBuffer); console.log(`Audio saved to ${OUTPUT_FILE}`); } main().catch(console.error); ``` </details> <details> <summary><strong>šŸŽ¤ Listing and Finding Voices</strong></summary> You can list all available voices and filter them by criteria. ```typescript // examples/listVoices.ts import { VoicesManager } from 'edge-tts-universal'; async function main() { const voicesManager = await VoicesManager.create(); // Find all English voices const voices = voicesManager.find({ Language: 'en' }); console.log( 'English voices:', voices.map((v) => v.ShortName) ); // Find female US voices const femaleUsVoices = voicesManager.find({ Gender: 'Female', Locale: 'en-US', }); console.log( 'Female US voices:', femaleUsVoices.map((v) => v.ShortName) ); } main().catch(console.error); ``` </details> <details> <summary><strong>šŸ“ŗ Streaming with Subtitles (WordBoundary events)</strong></summary> The `stream()` method provides `WordBoundary` events for generating subtitles. ```typescript // examples/streaming.ts import { Communicate, SubMaker } from 'edge-tts-universal'; const TEXT = 'This is a test of the streaming functionality, with subtitles.'; const VOICE = 'en-GB-SoniaNeural'; async function main() { const communicate = new Communicate(TEXT, { voice: VOICE }); const subMaker = new SubMaker(); for await (const chunk of communicate.stream()) { if (chunk.type === 'audio' && chunk.data) { // Do something with the audio data, e.g., stream it to a client. console.log(`Received audio chunk of size: ${chunk.data.length}`); } else if (chunk.type === 'WordBoundary') { subMaker.feed(chunk); } } // Get the subtitles in SRT format. const srt = subMaker.getSrt(); console.log('\nGenerated Subtitles (SRT):\n', srt); } main().catch(console.error); ``` </details> ## API Reference šŸ“– **[Complete API Documentation →](./API.md)** The main exports of the package are: **Simple API:** - **`EdgeTTS`** - Simple, promise-based TTS class for one-shot synthesis - **`createVTT`** / **`createSRT`** - Utility functions for subtitle generation **Advanced API:** - **`Communicate`** - Advanced streaming TTS class for real-time processing - **`VoicesManager`** - A class to find and filter voices - **`listVoices`** - A function to get all available voices - **`SubMaker`** - A utility to generate SRT subtitles from `WordBoundary` events **Isomorphic (Universal) API:** - **`IsomorphicCommunicate`** - Universal TTS class that works in Node.js and browsers - **`IsomorphicVoicesManager`** - Universal voice management with environment detection - **`listVoicesIsomorphic`** - Universal voice listing using cross-fetch - **`IsomorphicDRM`** - Cross-platform security token generation **Common:** - **Exception classes** - `NoAudioReceived`, `WebSocketError`, etc. - **TypeScript types** - Complete type definitions for voices, options, and stream chunks All three APIs use the same robust infrastructure including **DRM security handling**, error recovery, proxy support, and all Microsoft Edge authentication features. The isomorphic API provides universal compatibility through environment detection and isomorphic packages. For detailed documentation, examples, and advanced usage patterns, see the [comprehensive API guide](./API.md).