Compare commits

...

47 Commits

Author SHA1 Message Date
Hoan Luu Huu
c0fab2880b fix cannot send multipart to aws due to min size (#319) 2024-05-03 07:37:38 -04:00
Hoan Luu Huu
ce2fa392a4 support aws speech by roleArn (#313)
* support aws speech by roleArn

* support 3 types of aws  credentials

* wip

* wip

* update speech util version
2024-05-02 07:57:22 -04:00
Hoan Luu Huu
3b47162d13 Feat/record with pipeline (#318)
* use pipeline for nodejs streams

* use pipeline for nodejs streams
2024-04-30 07:39:24 -04:00
Hoan Luu Huu
b765232d4f api server cannot synthesize text after upgrade latest speech-utils (#317)
* api server cannot synthesize text after upgrade latest speech-utils

* wip

* add testcase for synthesize text

* fix synthesize testcase
2024-04-29 19:48:34 -04:00
Dave Horton
2436bea6ea add support for LCC updateCall with conferenceParticipantState (#296)
* add support for LCC updateCall with conferenceParticipantState

* wip

* wip
2024-04-22 11:06:08 -04:00
Dave Horton
f67abddbd4 bug: attempting to add duplicate dns records on hosted system (#312) 2024-04-19 18:13:27 -04:00
Hoan Luu Huu
39fcb17dec support mod_rimelabs_tts (#310)
* support mod_rimelabs_tts

* update speech utils 0.0.51
2024-04-12 07:25:04 -04:00
Hoan Luu Huu
80418aa7e5 check playht can fetch voices when adding new speech credential (#309)
* check playht can fetch voices when adding new speech credential

* wip

* wip

* wip

* wip
2024-04-12 07:01:13 -04:00
Hoan Luu Huu
b21d10eb3e fetch playht custom voice (#307) 2024-04-09 08:48:18 -04:00
Hoan Luu Huu
7875eb51b9 playht should return list of voice match voice engine configured at speech credentials (#306) 2024-04-09 06:53:17 -04:00
Hoan Luu Huu
e2c1383723 support mod_playht_tts (#304)
* support mod_playht_tts

* wip

* wip

* wip

* wip

* wip

* update speech utils version
2024-04-08 10:21:29 -04:00
Dave Horton
40de2c5945 option_ping was incorrectly removed, adding back (#305) 2024-04-08 08:56:31 -04:00
Dave Horton
3a299bc3ca update to speech utils with azure 1.36.0 (#303) 2024-04-07 17:45:33 -04:00
Dave Horton
70c9407742 update to speech utils with azure 1.36.0 2024-04-07 12:16:55 -04:00
Dave Horton
dba66d58fc back out column addition of -register_use_tls 2024-04-06 13:48:26 -04:00
Dave Horton
0ff3d22faf Revert "feat send options ping for sip gateway (#273)"
This reverts commit a4792a521f.
2024-04-06 13:27:32 -04:00
Hoan Luu Huu
187a428a75 register use tls (#302) 2024-04-04 08:02:29 -04:00
Hoan Luu Huu
a4792a521f feat send options ping for sip gateway (#273)
* feat send options ping for sip gateway

* update upgrade db script to have 8006
2024-03-30 09:14:29 -04:00
Dave Horton
3ac9693735 update speech-utils with fixes for deepgram production api and tts streaming 2024-03-24 08:15:00 -04:00
Dave Horton
3ad54a0e72 update to released deepgram tts voices (#299) 2024-03-13 09:16:12 -04:00
Hoan Luu Huu
bd8fb2f9db remove use_streaming from speech credential (#294)
* remove use_streaming from speech credential

* wip
2024-02-20 08:01:33 -05:00
Dave Horton
32b317ae68 update to latest speech-utils 2024-02-12 21:11:49 -05:00
Hoan Luu Huu
40e8d08727 support deepgram tts onprem (#292)
* support deepgram tts onprem

* wip

* wip

* deepgram disable speech test if api_key is missng
2024-02-12 09:27:13 -05:00
Hoan Luu Huu
256ca440a0 add use_streaming flag for elevenlabs and whisper (#290)
* add use_streaming flag for elevenlabs (not for whisper yet)
---------

Co-authored-by: Dave Horton <daveh@beachdognet.com>
2024-02-12 09:18:49 -05:00
Markus Frindt
68d73345ef Improve Swagger file, add login route, fix swagger linting (#291)
Co-authored-by: Markus Frindt <m.frindt@cognigy.com>
2024-02-06 12:10:31 -05:00
Hoan Luu Huu
54dd72ff66 fetch list of tts voices from provider (#289)
* fetch list of tts voices from provider

* revert serve-integration

* fix

* fix for aws

* fix for aws

* fix for aws

* update speech-utils version
2024-01-25 12:03:02 -05:00
Dave Horton
832a4e8032 update db-helpers 2024-01-17 13:24:51 -05:00
Hoan Luu Huu
33c3b99e2e update paid account to active if it's in deactivated (#287)
* update paid account to active if it's in deactivated

* fix review comment
2024-01-17 09:20:38 -05:00
Hoan Luu Huu
8b2a2e196e Feat/record upload buffer (#285)
* uploader with buffer for google and azure

* wip

* wip

* wip
2024-01-15 09:51:10 -05:00
Hoan Luu Huu
556717a9a4 enable convert raw to mp3 (#284) 2024-01-11 07:47:41 -05:00
Dave Horton
f2c635268f #283: fix for db upgrade script 2024-01-10 10:11:47 -05:00
Hoan Luu Huu
c8999a5929 fix tts stt langs and voices when credential is provided (#282) 2024-01-09 07:31:33 -05:00
Hoan Luu Huu
7e046ac7f3 TTS/STT languages and voices for each provider (#281)
* wip

* wip

* wip

* wip

* wip

* wip

* add testcase
2024-01-08 18:15:40 -05:00
Hoan Luu Huu
997ff05f3c support tts deepgram (#277) 2023-12-26 07:47:19 -05:00
Hoan Luu Huu
55d8fdef1c fix nuance onprem stt does not have data (#276) 2023-12-19 09:12:40 -05:00
Dave Horton
7d355f2fac enable azure stt through a proxy (#275) 2023-12-18 12:37:10 -05:00
Hoan Luu Huu
c6b8ec1b28 fix upload recording (#274) 2023-12-18 08:36:25 -05:00
Hoan Luu Huu
10159a0ba6 fix azure tts test from speech-utils (#272)
* fix azure tts test from speech-utils

* wip
2023-12-06 22:10:55 -05:00
Hoan Luu Huu
7a558c7349 update speech utils version (#270) 2023-12-05 20:48:19 -05:00
Hoan Luu Huu
4dbe7af9db fix list registered user does not have registered_status field (#268) 2023-12-05 07:44:43 -05:00
Hoan Luu Huu
4ec34faa29 support inband dtmf (#261)
* support inband dtmf

* support inband dtmf

* wip
2023-12-02 11:31:07 -05:00
Hoan Luu Huu
e2d6086f9f elevenlabs new model and options (#267) 2023-11-30 14:01:50 -05:00
Hoan Luu Huu
0e056ad296 support getting registered user details (#265)
* support getting registered user details

* add swager

* fix to use new registrar api

* assert sip_realm should be available for registered sip user query

* update mw registra version
2023-11-28 08:44:45 -05:00
Anton Voylenko
1d69457ddc support for llc tag (#262) 2023-11-17 08:48:34 -05:00
Anton Voylenko
dcfe6cc05d swagger: update create call spec (#260) 2023-11-14 10:40:31 -05:00
Hoan Luu Huu
a474c2d4cc let realtimedb-help build redis configuraiton from env variables (#256)
* let realtimedb-help build redis configuraiton from env variables

* update speech-utils version
2023-11-14 08:57:16 -05:00
Hoan Luu Huu
0f244cf6d5 fix create update lcr generate too much request (#259)
* fix create update lcr generate too much request

* wip
2023-11-14 08:07:04 -05:00
50 changed files with 16449 additions and 11011 deletions

10
app.js
View File

@@ -46,18 +46,15 @@ const {
addKey,
retrieveKey,
deleteKey,
incrKey,
JAMBONES_REDIS_SENTINELS
incrKey
} = require('./lib/helpers/realtimedb-helpers');
const {
getTtsVoices,
getTtsSize,
purgeTtsCache,
getAwsAuthToken,
synthAudio
} = require('@jambonz/speech-utils')(JAMBONES_REDIS_SENTINELS || {
host: process.env.JAMBONES_REDIS_HOST,
port: process.env.JAMBONES_REDIS_PORT || 6379
}, logger);
} = require('@jambonz/speech-utils')({}, logger);
const {
lookupAppBySid,
lookupAccountBySid,
@@ -99,6 +96,7 @@ app.locals = {
deleteKey,
getTtsVoices,
getTtsSize,
getAwsAuthToken,
purgeTtsCache,
synthAudio,
lookupAppBySid,

View File

@@ -162,7 +162,7 @@ regex VARCHAR(32) NOT NULL COMMENT 'regex-based pattern match against dialed num
description VARCHAR(1024),
priority INTEGER NOT NULL COMMENT 'lower priority routes are attempted first',
PRIMARY KEY (lcr_route_sid)
) COMMENT='An ordered list of digit patterns in an LCR table. The pat';
) COMMENT='An ordered list of digit patterns in an LCR table. The patterns are tested in sequence until one matches';
CREATE TABLE lcr
(
@@ -173,7 +173,7 @@ default_carrier_set_entry_sid CHAR(36) COMMENT 'default carrier/route to use whe
service_provider_sid CHAR(36),
account_sid CHAR(36),
PRIMARY KEY (lcr_sid)
) COMMENT='An LCR (least cost routing) table that is used by a service ';
) COMMENT='An LCR (least cost routing) table that is used by a service provider or account to make decisions about routing outbound calls when multiple carriers are available.';
CREATE TABLE password_settings
(
@@ -458,6 +458,7 @@ inbound BOOLEAN NOT NULL COMMENT 'if true, whitelist this IP to allow inbound ca
outbound BOOLEAN NOT NULL COMMENT 'if true, include in least-cost routing when placing calls to the PSTN',
voip_carrier_sid CHAR(36) NOT NULL,
is_active BOOLEAN NOT NULL DEFAULT 1,
send_options_ping BOOLEAN NOT NULL DEFAULT 0,
pad_crypto BOOLEAN NOT NULL DEFAULT 0,
protocol ENUM('udp','tcp','tls', 'tls/srtp') DEFAULT 'udp' COMMENT 'Outbound call protocol',
PRIMARY KEY (sip_gateway_sid)
@@ -495,7 +496,7 @@ messaging_hook_sid CHAR(36) COMMENT 'webhook to call for inbound SMS/MMS ',
app_json TEXT,
speech_synthesis_vendor VARCHAR(64) NOT NULL DEFAULT 'google',
speech_synthesis_language VARCHAR(12) NOT NULL DEFAULT 'en-US',
speech_synthesis_voice VARCHAR(64),
speech_synthesis_voice VARCHAR(256),
speech_synthesis_label VARCHAR(64),
speech_recognizer_vendor VARCHAR(64) NOT NULL DEFAULT 'google',
speech_recognizer_language VARCHAR(64) NOT NULL DEFAULT 'en-US',
@@ -503,7 +504,7 @@ speech_recognizer_label VARCHAR(64),
use_for_fallback_speech BOOLEAN DEFAULT false,
fallback_speech_synthesis_vendor VARCHAR(64),
fallback_speech_synthesis_language VARCHAR(12),
fallback_speech_synthesis_voice VARCHAR(64),
fallback_speech_synthesis_voice VARCHAR(256),
fallback_speech_synthesis_label VARCHAR(64),
fallback_speech_recognizer_vendor VARCHAR(64),
fallback_speech_recognizer_language VARCHAR(64),

View File

@@ -551,7 +551,7 @@
</location>
<size>
<width>293.00</width>
<height>540.00</height>
<height>560.00</height>
</size>
<zorder>6</zorder>
<SQLField>
@@ -2332,7 +2332,7 @@
</location>
<size>
<width>281.00</width>
<height>240.00</height>
<height>260.00</height>
</size>
<zorder>7</zorder>
<SQLField>
@@ -2399,10 +2399,18 @@
<notNull><![CDATA[1]]></notNull>
<uid><![CDATA[27D4A5BD-8093-4ADD-B5B5-D546844206F9]]></uid>
</SQLField>
<SQLField>
<name><![CDATA[send_options_ping]]></name>
<type><![CDATA[BOOLEAN]]></type>
<defaultValue><![CDATA[0]]></defaultValue>
<notNull><![CDATA[1]]></notNull>
<uid><![CDATA[E04C19A2-12BF-443F-AB61-96990224A18D]]></uid>
</SQLField>
<SQLField>
<name><![CDATA[pad_crypto]]></name>
<type><![CDATA[BOOLEAN]]></type>
<defaultValue><![CDATA[0]]></defaultValue>
<forcedUnique><![CDATA[0]]></forcedUnique>
<notNull><![CDATA[1]]></notNull>
<uid><![CDATA[C5C0043B-100A-4476-BF01-BE0777AE27C0]]></uid>
</SQLField>
@@ -2560,7 +2568,7 @@
</SQLField>
<SQLField>
<name><![CDATA[speech_synthesis_voice]]></name>
<type><![CDATA[VARCHAR(64)]]></type>
<type><![CDATA[VARCHAR(256)]]></type>
<notNull><![CDATA[0]]></notNull>
<uid><![CDATA[929D66F0-64B9-4D7C-AB4B-24F131E1178F]]></uid>
</SQLField>
@@ -2610,7 +2618,7 @@
</SQLField>
<SQLField>
<name><![CDATA[fallback_speech_synthesis_voice]]></name>
<type><![CDATA[VARCHAR(64)]]></type>
<type><![CDATA[VARCHAR(256)]]></type>
<notNull><![CDATA[0]]></notNull>
<uid><![CDATA[6A0E92C9-32B9-4179-A893-3DADF5DD7728]]></uid>
</SQLField>
@@ -3097,17 +3105,17 @@
<overviewPanelHidden><![CDATA[0]]></overviewPanelHidden>
<pageBoundariesVisible><![CDATA[0]]></pageBoundariesVisible>
<PageGridVisible><![CDATA[0]]></PageGridVisible>
<RightSidebarWidth><![CDATA[1924.000000]]></RightSidebarWidth>
<RightSidebarWidth><![CDATA[1235.000000]]></RightSidebarWidth>
<sidebarIndex><![CDATA[2]]></sidebarIndex>
<snapToGrid><![CDATA[0]]></snapToGrid>
<SourceSidebarWidth><![CDATA[0.000000]]></SourceSidebarWidth>
<SourceSidebarWidth><![CDATA[312.000000]]></SourceSidebarWidth>
<SQLEditorFileFormatVersion><![CDATA[4]]></SQLEditorFileFormatVersion>
<uid><![CDATA[58C99A00-06C9-478C-A667-C63842E088F3]]></uid>
<windowHeight><![CDATA[985.000000]]></windowHeight>
<windowLocationX><![CDATA[-1307.000000]]></windowLocationX>
<windowLocationY><![CDATA[1008.000000]]></windowLocationY>
<windowScrollOrigin><![CDATA[{1.5, 786}]]></windowScrollOrigin>
<windowWidth><![CDATA[2201.000000]]></windowWidth>
<windowHeight><![CDATA[870.000000]]></windowHeight>
<windowLocationX><![CDATA[-1164.000000]]></windowLocationX>
<windowLocationY><![CDATA[1131.000000]]></windowLocationY>
<windowScrollOrigin><![CDATA[{0.5, 0}]]></windowScrollOrigin>
<windowWidth><![CDATA[1512.000000]]></windowWidth>
</SQLDocumentInfo>
<AllowsIndexRenamingOnInsert><![CDATA[1]]></AllowsIndexRenamingOnInsert>
<defaultLabelExpanded><![CDATA[1]]></defaultLabelExpanded>

View File

@@ -88,7 +88,7 @@ const sql = {
'ALTER TABLE user_permissions ADD FOREIGN KEY permission_sid_idxfk (permission_sid) REFERENCES permissions (permission_sid)',
'ALTER TABLE `users` ADD COLUMN `is_active` BOOLEAN NOT NULL default true',
],
8003: [
'8003': [
'SET FOREIGN_KEY_CHECKS=0',
'ALTER TABLE `voip_carriers` ADD COLUMN `register_status` VARCHAR(4096)',
'ALTER TABLE `sbc_addresses` ADD COLUMN `last_updated` DATETIME',
@@ -140,7 +140,7 @@ const sql = {
'ALTER TABLE lcr_carrier_set_entry ADD FOREIGN KEY voip_carrier_sid_idxfk_3 (voip_carrier_sid) REFERENCES voip_carriers (voip_carrier_sid)',
'SET FOREIGN_KEY_CHECKS=1',
],
8004: [
'8004': [
'alter table accounts add column record_all_calls BOOLEAN NOT NULL DEFAULT false',
'alter table accounts add column bucket_credential VARCHAR(8192)',
'alter table accounts add column record_format VARCHAR(16) NOT NULL DEFAULT \'mp3\'',
@@ -160,7 +160,7 @@ const sql = {
'ALTER TABLE clients ADD CONSTRAINT account_sid_idxfk_13 FOREIGN KEY account_sid_idxfk_13 (account_sid) REFERENCES accounts (account_sid)',
'ALTER TABLE sip_gateways ADD COLUMN protocol ENUM(\'udp\',\'tcp\',\'tls\', \'tls/srtp\') DEFAULT \'udp\''
],
8005: [
'8005': [
'DROP INDEX speech_credentials_idx_1 ON speech_credentials',
'ALTER TABLE speech_credentials ADD COLUMN label VARCHAR(64)',
'ALTER TABLE applications ADD COLUMN speech_synthesis_label VARCHAR(64)',
@@ -190,7 +190,12 @@ const sql = {
'ALTER TABLE google_custom_voices ADD FOREIGN KEY speech_credential_sid_idxfk (speech_credential_sid) REFERENCES speech_credentials (speech_credential_sid) ON DELETE CASCADE',
'ALTER TABLE clients ADD COLUMN allow_direct_queue_calling BOOLEAN NOT NULL DEFAULT 1',
'ALTER TABLE clients ADD COLUMN allow_direct_user_calling BOOLEAN NOT NULL DEFAULT 1',
'ALTER TABLE clients ADD COLUMN allow_direct_app_calling BOOLEAN NOT NULL DEFAULT 1'
'ALTER TABLE clients ADD COLUMN allow_direct_app_calling BOOLEAN NOT NULL DEFAULT 1',
],
9000: [
'ALTER TABLE sip_gateways ADD COLUMN send_options_ping BOOLEAN NOT NULL DEFAULT 0',
'ALTER TABLE applications MODIFY COLUMN speech_synthesis_voice VARCHAR(256)',
'ALTER TABLE applications MODIFY COLUMN fallback_speech_synthesis_voice VARCHAR(256)',
]
};
@@ -223,6 +228,7 @@ const doIt = async() => {
if (val < 8003) upgrades.push(...sql['8003']);
if (val < 8004) upgrades.push(...sql['8004']);
if (val < 8005) upgrades.push(...sql['8005']);
if (val < 9000) upgrades.push(...sql['9000']);
// perform all upgrades
logger.info({upgrades}, 'applying schema upgrades..');

View File

@@ -1,29 +1,5 @@
const logger = require('../logger');
const JAMBONES_REDIS_SENTINELS = process.env.JAMBONES_REDIS_SENTINELS ? {
sentinels: process.env.JAMBONES_REDIS_SENTINELS.split(',').map((sentinel) => {
let host, port = 26379;
if (sentinel.includes(':')) {
const arr = sentinel.split(':');
host = arr[0];
port = parseInt(arr[1], 10);
} else {
host = sentinel;
}
return {host, port};
}),
name: process.env.JAMBONES_REDIS_SENTINEL_MASTER_NAME,
...(process.env.JAMBONES_REDIS_SENTINEL_PASSWORD && {
password: process.env.JAMBONES_REDIS_SENTINEL_PASSWORD
}),
...(process.env.JAMBONES_REDIS_SENTINEL_USERNAME && {
username: process.env.JAMBONES_REDIS_SENTINEL_USERNAME
}),
...(process.env.JAMBONES_REDIS_SENTINEL_SENTINAL_PASSWORD && {
sentinelPassword: process.env.JAMBONES_REDIS_SENTINEL_SENTINAL_PASSWORD
}),
} : null;
const {
client,
retrieveCall,
@@ -37,10 +13,7 @@ const {
deleteKey,
incrKey,
client: redisClient,
} = require('@jambonz/realtimedb-helpers')(JAMBONES_REDIS_SENTINELS || {
host: process.env.JAMBONES_REDIS_HOST || 'localhost',
port: process.env.JAMBONES_REDIS_PORT || 6379
}, logger);
} = require('@jambonz/realtimedb-helpers')({}, logger);
module.exports = {
client,
@@ -54,6 +27,5 @@ module.exports = {
retrieveKey,
deleteKey,
redisClient,
incrKey,
JAMBONES_REDIS_SENTINELS
incrKey
};

View File

@@ -199,8 +199,9 @@ class Account extends Model {
debug(r3, 'Account.activateSubscription - replaced old subscription');
/* update account.plan to paid, if it isnt already */
/* update account.is_active to 1, if account is deactivated */
await promisePool.execute(
'UPDATE accounts SET plan_type = \'paid\' WHERE account_sid = ?',
'UPDATE accounts SET plan_type = \'paid\', is_active = 1 WHERE account_sid = ?',
[account_sid]);
return true;
}

View File

@@ -61,6 +61,10 @@ VoipCarrier.fields = [
name: 'requires_register',
type: 'number'
},
{
name: 'register_use_tls',
type: 'number'
},
{
name: 'register_username',
type: 'string'

View File

@@ -1,28 +1,58 @@
const { Writable } = require('stream');
const { BlobServiceClient } = require('@azure/storage-blob');
const { v4: uuidv4 } = require('uuid');
const streamBuffers = require('stream-buffers');
class AzureStorageUploadStream extends Writable {
constructor(logger, opts) {
super(opts);
const blobServiceClient = BlobServiceClient.fromConnectionString(opts.connection_string);
this.blockBlobClient = blobServiceClient.getContainerClient(opts.bucketName).getBlockBlobClient(opts.Key);
this.metadata = opts.metadata;
this.blocks = [];
this.bufferSize = 2 * 1024 * 1024; // Buffer size set to 2MB
this.buffer = new streamBuffers.WritableStreamBuffer({
initialSize: this.bufferSize,
incrementAmount: this.bufferSize
});
}
async _write(chunk, encoding, callback) {
const blockID = uuidv4().replace(/-/g, '');
this.blocks.push(blockID);
try {
await this.blockBlobClient.stageBlock(blockID, chunk, chunk.length);
this.buffer.write(chunk, encoding);
if (this.buffer.size() >= this.bufferSize) {
const blockID = uuidv4().replace(/-/g, '');
this.blocks.push(blockID);
try {
const dataToWrite = this.buffer.getContents();
await this.blockBlobClient.stageBlock(blockID, dataToWrite, dataToWrite.length);
callback();
} catch (error) {
callback(error);
}
} else {
callback();
} catch (error) {
callback(error);
}
}
async _final(callback) {
// Write any remaining data in buffer
if (this.buffer.size() > 0) {
const remainingData = this.buffer.getContents();
const blockID = uuidv4().replace(/-/g, '');
this.blocks.push(blockID);
try {
await this.blockBlobClient.stageBlock(blockID, remainingData, remainingData.length);
} catch (error) {
callback(error);
return;
}
}
try {
await this.blockBlobClient.commitBlockList(this.blocks);
// remove all null/undefined props

View File

@@ -1,5 +1,6 @@
const { Storage } = require('@google-cloud/storage');
const { Writable } = require('stream');
const streamBuffers = require('stream-buffers');
class GoogleStorageUploadStream extends Writable {
@@ -12,18 +13,38 @@ class GoogleStorageUploadStream extends Writable {
this.gcsFile = storage.bucket(opts.bucketName).file(opts.Key);
this.writeStream = this.gcsFile.createWriteStream();
this.bufferSize = 2 * 1024 * 1024; // Buffer size set to 2MB
this.buffer = new streamBuffers.WritableStreamBuffer({
initialSize: this.bufferSize,
incrementAmount: this.bufferSize
});
this.writeStream.on('error', (err) => this.logger.error(err));
this.writeStream.on('finish', () => {
this.logger.info('google storage Upload completed.');
this.logger.info('Google storage Upload completed.');
this._addMetadata();
});
}
_write(chunk, encoding, callback) {
this.writeStream.write(chunk, encoding, callback);
this.buffer.write(chunk, encoding);
// Write to GCS when buffer reaches desired size
if (this.buffer.size() >= this.bufferSize) {
const dataToWrite = this.buffer.getContents();
this.writeStream.write(dataToWrite, callback);
} else {
callback();
}
}
_final(callback) {
// Write any remaining data in the buffer to GCS
if (this.buffer.size() > 0) {
const remainingData = this.buffer.getContents();
this.writeStream.write(remainingData);
}
this.writeStream.end();
this.writeStream.once('finish', callback);
}
@@ -33,7 +54,7 @@ class GoogleStorageUploadStream extends Writable {
await this.gcsFile.setMetadata({metadata: this.metadata});
this.logger.info('Google storage Upload and metadata setting completed.');
} catch (err) {
this.logger.error(err, 'Google storage An error occurred while setting metadata');
this.logger.error(err, 'Google storage An error occurred while setting metadata');
}
}
}

View File

@@ -11,7 +11,7 @@ class S3MultipartUploadStream extends Writable {
super(opts);
this.logger = logger;
this.bucketName = opts.bucketName;
this.objectKey = opts.objectKey;
this.objectKey = opts.Key;
this.uploadId = null;
this.partNumber = 1;
this.multipartETags = [];

View File

@@ -3,6 +3,7 @@ const Websocket = require('ws');
const PCMToMP3Encoder = require('./encoder');
const wav = require('wav');
const { getUploader } = require('./utils');
const { pipeline } = require('stream');
async function upload(logger, socket) {
socket._recvInitialMetadata = false;
@@ -60,22 +61,19 @@ async function upload(logger, socket) {
bitrate: 128
}, logger);
}
const handleError = (err, streamType) => {
logger.error(
{ err },
`Error while streaming for vendor: ${obj.vendor}, pipe: ${streamType}: ${err.message}`
);
};
/* start streaming data */
const duplex = Websocket.createWebSocketStream(socket);
duplex
.on('error', (err) => handleError(err, 'duplex'))
.pipe(encoder)
.on('error', (err) => handleError(err, 'encoder'))
.pipe(uploadStream)
.on('error', (err) => handleError(err, 'uploadStream'));
pipeline(
Websocket.createWebSocketStream(socket),
encoder,
uploadStream,
(error) => {
if (error) {
logger.error({ error }, 'pipeline error, cannot upload data to storage');
socket.close();
}
}
);
} else {
logger.info(`account ${accountSid} does not have any bucket credential, close the socket`);
socket.close();

View File

@@ -5,7 +5,7 @@ const S3MultipartUploadStream = require('./s3-multipart-upload-stream');
const getUploader = (key, metadata, bucket_credential, logger) => {
const uploaderOpts = {
bucketName: bucket_credential.name,
objectKey: key,
Key: key,
metadata
};
try {

View File

@@ -155,6 +155,9 @@ router.get('/:sid/RegisteredSipUsers', async(req, res) => {
if (!result || result.length === 0) {
throw new DbErrorBadRequest(`account not found for sid ${account_sid}`);
}
if (!result[0].sip_realm) {
throw new DbErrorBadRequest('account does not have sip_realm configuration');
}
const users = await registrar.getRegisteredUsersForRealm(result[0].sip_realm);
res.status(200).json(users.map((u) => `${u}@${result[0].sip_realm}`));
} catch (err) {
@@ -162,6 +165,42 @@ router.get('/:sid/RegisteredSipUsers', async(req, res) => {
}
});
router.post('/:sid/RegisteredSipUsers', async(req, res) => {
const {logger, registrar} = req.app.locals;
const users = req.body;
try {
const account_sid = parseAccountSid(req);
await validateRequest(req, account_sid);
const result = await Account.retrieve(account_sid);
if (!result || result.length === 0) {
throw new DbErrorBadRequest(`account not found for sid ${account_sid}`);
}
if (!result[0].sip_realm) {
throw new DbErrorBadRequest('account does not have sip_realm configuration');
}
if (!users || !Array.isArray(users) || users.length === 0) {
return res.status(200).json(await registrar.getRegisteredUsersDetailsForRealm(result[0].sip_realm));
}
const ret = [];
for (const u of users) {
const user = await registrar.query(`${u}@${result[0].sip_realm}`) || {
name: u,
contact: null,
expiryTime: 0,
protocol: null
};
ret.push({
name: u,
...user,
registered_status: user.expiryTime > 0 ? 'active' : 'inactive',
});
}
res.status(200).json(ret);
} catch (err) {
sysError(logger, res, err);
}
});
router.get('/:sid/RegisteredSipUsers/:client', async(req, res) => {
const {logger, registrar, lookupClientByAccountAndUsername} = req.app.locals;
const client = req.params.client;
@@ -224,7 +263,10 @@ function validateUpdateCall(opts) {
'conf_mute_status',
'mute_status',
'sip_request',
'record'
'record',
'tag',
'dtmf',
'conferenceParticipantAction'
]
.reduce((acc, prop) => (opts[prop] ? ++acc : acc), 0);
@@ -266,9 +308,28 @@ function validateUpdateCall(opts) {
if (opts.record && !opts.record.action) {
throw new DbErrorBadRequest('record requires action property');
}
if (opts.dtmf && !opts.dtmf.digit) {
throw new DbErrorBadRequest('invalid dtmf');
}
if ('startCallRecording' === opts.record?.action && !opts.record.siprecServerURL) {
throw new DbErrorBadRequest('record requires siprecServerURL property when starting recording');
}
if (opts.tag && (typeof opts.tag !== 'object' || Array.isArray(opts.tag) || opts.tag === null)) {
throw new DbErrorBadRequest('invalid tag data');
}
if (opts.conferenceParticipantAction) {
if (!['tag', 'untag', 'coach', 'uncoach', 'mute', 'unmute', 'hold', 'unhold']
.includes(opts.conferenceParticipantAction.action)) {
throw new DbErrorBadRequest(
`conferenceParticipantAction invalid action property ${opts.conferenceParticipantAction.action}`);
}
if ('tag' == opts.conferenceParticipantAction.action && !opts.tag) {
throw new DbErrorBadRequest('conferenceParticipantAction requires tag property when action is \'tag\'');
}
if ('coach' == opts.conferenceParticipantAction.action && !opts.tag) {
throw new DbErrorBadRequest('conferenceParticipantAction requires tag property when action is \'coach\'');
}
}
}
function validateTo(to) {

View File

@@ -100,6 +100,93 @@ const preconditions = {
decorate(router, Lcr, ['add', 'update', 'delete'], preconditions);
const validateLcrBatchAdd = async(lcr_sid, body, lookupCarrierBySid) => {
for (const lcr_route of body) {
lcr_route.lcr_sid = lcr_sid;
if (!lcr_route.lcr_carrier_set_entries || lcr_route.lcr_carrier_set_entries.length === 0) {
throw new DbErrorBadRequest('Lcr Route batch process require lcr_carrier_set_entries');
}
for (const entry of lcr_route.lcr_carrier_set_entries) {
// check voip_carrier_sid is exist
if (!entry.voip_carrier_sid) {
throw new DbErrorBadRequest('One of lcr_carrier_set_entries is missing voip_carrier_sid');
}
const carrier = await lookupCarrierBySid(entry.voip_carrier_sid);
if (!carrier) {
throw new DbErrorBadRequest('unknown voip_carrier_sid');
}
}
}
};
const addNewLcrRoute = async(lcr_route) => {
const lcr_sid = lcr_route.lcr_sid;
const lcr_carrier_set_entries = lcr_route.lcr_carrier_set_entries;
delete lcr_route.lcr_carrier_set_entries;
const lcr_route_sid = await LcrRoutes.make(lcr_route);
for (const entry of lcr_carrier_set_entries) {
entry.lcr_route_sid = lcr_route_sid;
const lcr_carrier_set_entry_sid = await LcrCarrierSetEntry.make(entry);
if (lcr_route.priority === 9999) {
// this is default lcr set entry
const [lcr] = await Lcr.retrieve(lcr_sid);
if (lcr) {
lcr.default_carrier_set_entry_sid = lcr_carrier_set_entry_sid;
delete lcr.lcr_sid;
await Lcr.update(lcr_sid, lcr);
}
}
}
};
router.post('/:sid/Routes', async(req, res) => {
const results = await Lcr.retrieve(req.params.sid);
if (results.length === 0) return res.sendStatus(404);
const {logger, lookupCarrierBySid} = req.app.locals;
try {
const body = req.body;
await validateLcrBatchAdd(req.params.sid, body, lookupCarrierBySid);
for (const lcr_route of body) {
await addNewLcrRoute(lcr_route, lookupCarrierBySid);
}
res.sendStatus(204);
} catch (err) {
sysError(logger, res, err);
}
});
router.put('/:sid/Routes', async(req, res) => {
const results = await Lcr.retrieve(req.params.sid);
if (results.length === 0) return res.sendStatus(404);
const {logger, lookupCarrierBySid} = req.app.locals;
try {
const body = req.body;
await validateLcrBatchAdd(req.params.sid, body, lookupCarrierBySid);
for (const lcr_route of body) {
if (lcr_route.lcr_route_sid) {
const lcr_route_sid = lcr_route.lcr_route_sid;
delete lcr_route.lcr_route_sid;
const lcr_carrier_set_entries = lcr_route.lcr_carrier_set_entries;
delete lcr_route.lcr_carrier_set_entries;
await LcrRoutes.update(lcr_route_sid, lcr_route);
for (const entry of lcr_carrier_set_entries) {
const lcr_carrier_set_entry_sid = entry.lcr_carrier_set_entry_sid;
delete entry.lcr_carrier_set_entry_sid;
await LcrCarrierSetEntry.update(lcr_carrier_set_entry_sid, entry);
}
} else {
// Route is not available yet, let create it now
await addNewLcrRoute(lcr_route, lookupCarrierBySid);
}
}
res.sendStatus(204);
} catch (err) {
sysError(logger, res, err);
}
});
router.get('/', async(req, res) => {
const logger = req.app.locals.logger;
try {

View File

@@ -31,6 +31,7 @@ router.post('/:sip_realm', async(req, res) => {
const [sbcs] = await promisePool.query('SELECT ipv4 from sbc_addresses');
if (sbcs.length === 0) throw new Error('no SBC addresses provisioned in the database!');
const ips = sbcs.map((s) => s.ipv4);
const uniqueIps = [...new Set(ips)];
/* retrieve existing dns records */
const [old_recs] = await promisePool.query('SELECT record_id from dns_records WHERE account_sid = ?',
@@ -48,7 +49,7 @@ router.post('/:sip_realm', async(req, res) => {
}
/* add the dns records */
const records = await createDnsRecords(logger, domain, subdomain, ips);
const records = await createDnsRecords(logger, domain, subdomain, uniqueIps);
if (!records) throw new Error(`failure updating dns records for ${sip_realm}`);
const values = records.map((r) => {
return `('${uuid()}', '${account_sid}', '${r.type}', ${r.id})`;

View File

@@ -5,8 +5,11 @@ const SpeechCredential = require('../../models/speech-credential');
const sysError = require('../error');
const {decrypt, encrypt} = require('../../utils/encrypt-decrypt');
const {parseAccountSid, parseServiceProviderSid, parseSpeechCredentialSid} = require('./utils');
const {decryptCredential, testWhisper} = require('../../utils/speech-utils');
const {DbErrorUnprocessableRequest, DbErrorForbidden} = require('../../utils/errors');
const {decryptCredential, testWhisper, testDeepgramTTS,
getLanguagesAndVoicesForVendor,
testPlayHT,
testRimelabs} = require('../../utils/speech-utils');
const {DbErrorUnprocessableRequest, DbErrorForbidden, DbErrorBadRequest} = require('../../utils/errors');
const {
testGoogleTts,
testGoogleStt,
@@ -24,7 +27,6 @@ const {
testElevenlabs,
testAssemblyStt
} = require('../../utils/speech-utils');
const bent = require('bent');
const {promisePool} = require('../../db');
const validateAdd = async(req) => {
@@ -111,11 +113,14 @@ const encryptCredential = (obj) => {
secret_access_key,
aws_region,
api_key,
role_arn,
region,
client_id,
secret,
nuance_tts_uri,
nuance_stt_uri,
deepgram_stt_uri,
deepgram_stt_use_tls,
use_custom_tts,
custom_tts_endpoint,
custom_tts_endpoint_url,
@@ -132,7 +137,10 @@ const encryptCredential = (obj) => {
custom_tts_url,
auth_token = '',
cobalt_server_uri,
model_id
model_id,
user_id,
voice_engine,
options
} = obj;
switch (vendor) {
@@ -148,10 +156,17 @@ const encryptCredential = (obj) => {
return encrypt(service_key);
case 'aws':
assert(access_key_id, 'invalid aws speech credential: access_key_id is required');
assert(secret_access_key, 'invalid aws speech credential: secret_access_key is required');
assert(aws_region, 'invalid aws speech credential: aws_region is required');
const awsData = JSON.stringify({aws_region, access_key_id, secret_access_key});
// AWS polly can work for 3 types of credentials:
// 1/ access_key_id and secret_access_key
// 2/ RoleArn Assume role
// 3/ RoleArn assigned to instance profile where will run this application
const awsData = JSON.stringify(
{
aws_region,
...(access_key_id && {access_key_id}),
...(secret_access_key && {secret_access_key}),
...(role_arn && {role_arn}),
});
return encrypt(awsData);
case 'microsoft':
@@ -184,8 +199,11 @@ const encryptCredential = (obj) => {
return encrypt(nuanceData);
case 'deepgram':
assert(api_key, 'invalid deepgram speech credential: api_key is required');
const deepgramData = JSON.stringify({api_key});
// API key is optional if onprem
if (!deepgram_stt_uri) {
assert(api_key, 'invalid deepgram speech credential: api_key is required');
}
const deepgramData = JSON.stringify({api_key, deepgram_stt_uri, deepgram_stt_use_tls});
return encrypt(deepgramData);
case 'ibm':
@@ -210,9 +228,22 @@ const encryptCredential = (obj) => {
case 'elevenlabs':
assert(api_key, 'invalid elevenLabs speech credential: api_key is required');
assert(model_id, 'invalid elevenLabs speech credential: model_id is required');
const elevenlabsData = JSON.stringify({api_key, model_id});
const elevenlabsData = JSON.stringify({api_key, model_id, options});
return encrypt(elevenlabsData);
case 'playht':
assert(api_key, 'invalid playht speech credential: api_key is required');
assert(user_id, 'invalid playht speech credential: user_id is required');
assert(voice_engine, 'invalid voice_engine speech credential: voice_engine is required');
const playhtData = JSON.stringify({api_key, user_id, voice_engine, options});
return encrypt(playhtData);
case 'rimelabs':
assert(api_key, 'invalid rimelabs speech credential: api_key is required');
assert(model_id, 'invalid rimelabs speech credential: model_id is required');
const rimelabsData = JSON.stringify({api_key, model_id, options});
return encrypt(rimelabsData);
case 'assemblyai':
assert(api_key, 'invalid assemblyai speech credential: api_key is required');
const assemblyaiData = JSON.stringify({api_key});
@@ -411,7 +442,11 @@ router.put('/:sid', async(req, res) => {
custom_stt_url,
custom_tts_url,
cobalt_server_uri,
model_id
model_id,
voice_engine,
options,
deepgram_stt_uri,
deepgram_stt_use_tls,
} = req.body;
const newCred = {
@@ -433,7 +468,11 @@ router.put('/:sid', async(req, res) => {
custom_stt_url,
custom_tts_url,
cobalt_server_uri,
model_id
model_id,
voice_engine,
options,
deepgram_stt_uri,
deepgram_stt_use_tls,
};
logger.info({o, newCred}, 'updating speech credential with this new credential');
obj.credential = encryptCredential(newCred);
@@ -510,12 +549,13 @@ router.get('/:sid/test', async(req, res) => {
}
}
else if (cred.vendor === 'aws') {
const {getTtsVoices, getAwsAuthToken} = req.app.locals;
if (cred.use_for_tts) {
const {getTtsVoices} = req.app.locals;
try {
await testAwsTts(logger, getTtsVoices, {
accessKeyId: credential.access_key_id,
secretAccessKey: credential.secret_access_key,
roleArn: credential.role_arn,
region: credential.aws_region || process.env.AWS_REGION
});
results.tts.status = 'ok';
@@ -527,9 +567,10 @@ router.get('/:sid/test', async(req, res) => {
}
if (cred.use_for_stt) {
try {
await testAwsStt(logger, {
await testAwsStt(logger, getAwsAuthToken, {
accessKeyId: credential.access_key_id,
secretAccessKey: credential.secret_access_key,
roleArn: credential.role_arn,
region: credential.aws_region || process.env.AWS_REGION
});
results.stt.status = 'ok';
@@ -553,7 +594,7 @@ router.get('/:sid/test', async(req, res) => {
} = credential;
if (cred.use_for_tts) {
try {
await testMicrosoftTts(logger, {
await testMicrosoftTts(logger, synthAudio, {
api_key,
region,
use_custom_tts,
@@ -634,7 +675,17 @@ router.get('/:sid/test', async(req, res) => {
}
else if (cred.vendor === 'deepgram') {
const {api_key} = credential;
if (cred.use_for_stt) {
if (cred.use_for_tts) {
try {
await testDeepgramTTS(logger, synthAudio, credential);
results.tts.status = 'ok';
SpeechCredential.ttsTestResult(sid, true);
} catch (err) {
results.tts = {status: 'fail', reason: err.message};
SpeechCredential.ttsTestResult(sid, false);
}
}
if (cred.use_for_stt && api_key) {
try {
await testDeepgramStt(logger, {api_key});
results.stt.status = 'ok';
@@ -702,6 +753,33 @@ router.get('/:sid/test', async(req, res) => {
SpeechCredential.ttsTestResult(sid, false);
}
}
} else if (cred.vendor === 'playht') {
if (cred.use_for_tts) {
try {
await testPlayHT(logger, synthAudio, credential);
results.tts.status = 'ok';
SpeechCredential.ttsTestResult(sid, true);
} catch (err) {
let reason = err.message;
// if error is from bent, let get the body
try {
reason = await err.text();
} catch {}
results.tts = {status: 'fail', reason};
SpeechCredential.ttsTestResult(sid, false);
}
}
} else if (cred.vendor === 'rimelabs') {
if (cred.use_for_tts) {
try {
await testRimelabs(logger, synthAudio, credential);
results.tts.status = 'ok';
SpeechCredential.ttsTestResult(sid, true);
} catch (err) {
results.tts = {status: 'fail', reason: err.message};
SpeechCredential.ttsTestResult(sid, false);
}
}
} else if (cred.vendor === 'assemblyai') {
const {api_key} = credential;
if (cred.use_for_stt) {
@@ -738,89 +816,30 @@ router.get('/:sid/test', async(req, res) => {
* Fetch speech voices and languages
*/
router.post('/voices', async(req, res) => {
const logger = req.app.locals.logger;
const {vendor, label} = req.body;
const account_sid = req.user.account_sid || req.body.account_sid;
const service_provider_sid = req.user.service_provider_sid ||
req.body.service_provider_sid || parseServiceProviderSid(req);
router.get('/speech/supportedLanguagesAndVoices', async(req, res) => {
const {logger, getTtsVoices} = req.app.locals;
try {
res.status(200).json(await getTtsVoices(vendor, label, service_provider_sid, account_sid));
const {vendor, label} = req.query;
if (!vendor) {
throw new DbErrorBadRequest('vendor is required');
}
const account_sid = req.user.account_sid || req.body.account_sid;
const service_provider_sid = req.user.service_provider_sid ||
req.body.service_provider_sid || parseServiceProviderSid(req);
const credentials = await SpeechCredential.getSpeechCredentialsByVendorAndLabel(
service_provider_sid, account_sid, vendor, label);
const tmp = credentials && credentials.length > 0 ? credentials[0] : null;
const cred = tmp ? JSON.parse(decrypt(tmp.credential)) : null;
try {
const data = await getLanguagesAndVoicesForVendor(logger, vendor, cred, getTtsVoices);
res.status(200).json(data);
} catch (err) {
throw new DbErrorUnprocessableRequest(err.message);
}
} catch (err) {
sysError(logger, res, err);
}
});
router.post('/languages', async(req, res) => {
const logger = req.app.locals.logger;
const {vendor, label} = req.body;
const account_sid = req.user.account_sid || req.body.account_sid;
const service_provider_sid = req.user.service_provider_sid ||
req.body.service_provider_sid || parseServiceProviderSid(req);
try {
res.status(200).json(await getTtsLanguages(vendor, label, service_provider_sid, account_sid));
} catch (err) {
sysError(logger, res, err);
}
});
const getTtsVoices = async(vendor, label, service_provider_sid, account_sid) => {
const credentials = await SpeechCredential.getSpeechCredentialsByVendorAndLabel(
service_provider_sid, account_sid, vendor, label);
const tmp = credentials && credentials.length > 0 ? credentials[0] : null;
const cred = tmp ? JSON.parse(decrypt(tmp.credential)) : null;
if (vendor === 'elevenlabs') {
const get = bent('https://api.elevenlabs.io', 'GET', 'json', {
...(cred && {
'xi-api-key' : cred.api_key
})
});
const resp = await get('/v1/voices');
return resp ? resp.voices.map((v) => {
let name = `${v.name}${v.category !== 'premade' ? ` (${v.category})` : ''} -
${v.labels.accent ? ` ${v.labels.accent},` : ''}
${v.labels.description ? ` ${v.labels.description},` : ''}
${v.labels.age ? ` ${v.labels.age},` : ''}
${v.labels.gender ? ` ${v.labels.gender},` : ''}
${v.labels['use case'] ? ` ${v.labels['use case']},` : ''}
`;
const lastIndex = name.lastIndexOf(',');
if (lastIndex !== -1) {
name = name.substring(0, lastIndex);
}
return {
value: v.voice_id,
name
};
}).sort((a, b) => a.name.localeCompare(b.name)) : [];
}
return [];
};
const getTtsLanguages = async(vendor, label, service_provider_sid, account_sid) => {
const credentials = await SpeechCredential.getSpeechCredentialsByVendorAndLabel(
service_provider_sid, account_sid, vendor, label);
const tmp = credentials && credentials.length > 0 ? credentials[0] : null;
const cred = tmp ? JSON.parse(decrypt(tmp.credential)) : null;
if (vendor === 'elevenlabs') {
if (!cred) {
return [];
}
const get = bent('https://api.elevenlabs.io', 'GET', 'json', {
'xi-api-key' : cred.api_key
});
const resp = await get('/v1/models');
if (!resp || resp.length === 0) {
return [];
}
const model = resp.find((m) => m.model_id === cred.model_id);
return model ? model.languages.map((l) => {
return {
value: l.language_id,
name: l.name
};
}).sort((a, b) => a.name.localeCompare(b.name)) : [];
}
};
module.exports = router;

View File

@@ -9,6 +9,8 @@ const {DbErrorBadRequest} = require('../../utils/errors');
const Account = require('../../models/account');
const sysError = require('../error');
const { getSpeechCredential, decryptCredential } = require('../../utils/speech-utils');
const PCMToMP3Encoder = require('../../record/encoder');
const { pipeline } = require('stream');
router.delete('/', async(req, res) => {
const {purgeTtsCache} = req.app.locals;
@@ -38,6 +40,7 @@ router.post('/Synthesize', async(req, res) => {
try {
const accountSid = parseAccountSid(req);
const body = req.body;
const encodingMp3 = req.body.encodingMp3 || false;
if (!body.speech_credential_sid || !body.text || !body.language || !body.voice) {
throw new DbErrorBadRequest('speech_credential_sid, text, language, voice are all required');
}
@@ -67,6 +70,8 @@ router.post('/Synthesize', async(req, res) => {
voice = arr[1];
model = arr[2];
}
} else if (cred.vendor === 'deepgram') {
model = voice;
}
const stats = {
histogram: () => {},
@@ -82,23 +87,44 @@ router.post('/Synthesize', async(req, res) => {
model,
salt,
credentials: cred,
disableTtsCache: false
disableTtsCache: false,
disableTtsStreaming: true
});
const stat = fs.statSync(filePath);
let contentType = 'audio/mpeg';
let readStream = fs.createReadStream(filePath);
if (['nuance', 'nvidia'].includes(cred.vendor) ||
(
process.env.JAMBONES_TTS_TRIM_SILENCE &&
['microsoft', 'azure'].includes(cred.vendor)
)
) {
if (encodingMp3) {
readStream = readStream
.pipe(new PCMToMP3Encoder({
channels: 1,
sampleRate: 8000,
bitRate: 128
}, logger));
} else {
contentType = 'application/octet-stream';
}
}
res.writeHead(200, {
'Content-Type': 'audio/mpeg',
'Content-Length': stat.size,
'Content-Type': contentType,
});
const readStream = fs.createReadStream(filePath);
// We replaced all the event handlers with a simple call to readStream.pipe()
readStream.pipe(res);
pipeline(readStream, res, (err) => {
if (err) {
logger.error('ttscache/Synthesize failed:', err);
if (!res.headersSent) {
res.status(500).end('Server error');
}
}
readStream.on('end', () => {
// Delete the file after it's been read
fs.unlink(filePath, (err) => {
if (err) throw err;
fs.unlink(filePath, (unlinkErr) => {
if (unlinkErr) throw unlinkErr;
logger.info(`${filePath} was deleted`);
});
});

View File

@@ -1,7 +1,7 @@
openapi: 3.0.0
info:
title: jambonz REST API
description: jambonz REST API
title: Jambonz REST API
description: Jambonz REST API specification
contact:
email: daveh@drachtio.org
license:
@@ -382,11 +382,35 @@ paths:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/login:
post:
tags:
- Authentication
summary: login and retrieve a JWT
operationId: login
requestBody:
content:
application/json:
schema:
$ref: '#/components/schemas/Login'
responses:
200:
description: user logged in
content:
application/json:
schema:
$ref: '#/components/schemas/SuccessfulLogin'
500:
description: system error
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/logout:
post:
tags:
- Authentication
summary: log out and deactivate jwt
summary: log out and deactivate the JWT
operationId: logoutUser
responses:
204:
@@ -584,10 +608,9 @@ paths:
content:
application/json:
schema:
type:
array
type: array
items:
$ref: '#/components/schemas/Users'
$ref: '#/components/schemas/UserList'
403:
description: unauthorized
500:
@@ -610,27 +633,13 @@ paths:
- Users
summary: retrieve user information
operationId: getUser
requestBody:
content:
application/json:
schema:
type: object
properties:
name:
type: string
email:
type: string
is_active:
type: boolean
force_change:
type: boolean
scope:
type: string
permissions:
type: array
responses:
204:
200:
description: user information
content:
application/json:
schema:
$ref: '#/components/schemas/UserProfile'
403:
description: user information
content:
@@ -674,6 +683,8 @@ paths:
type: string
permissions:
type: array
items:
type: string
responses:
204:
description: user updated
@@ -712,6 +723,8 @@ paths:
type: string
permissions:
type: array
items:
type: string
old_password:
type: string
description: existing password, which is to be replaced
@@ -998,7 +1011,7 @@ paths:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/AccountTest/:ServiceProviderSid:
/AccountTest/{ServiceProviderSid}:
parameters:
- name: ServiceProviderSid
in: path
@@ -1095,6 +1108,9 @@ paths:
requires_register:
type: boolean
description: wehther this provider requires us to send a REGISTER to them in order to receive calls
register_use_tls:
type: boolean
description: wehther this provider requires us to send a REGISTER use TLS protocol
register_username:
type: string
description: sip username to authenticate with, if registration is required
@@ -1971,7 +1987,7 @@ paths:
tags:
- Service Providers
summary: add a VoiPCarrier to a service provider based on PredefinedCarrier template
operationId: createVoipCarrierFromTemplate
operationId: createVoipCarrierFromTemplateBySP
responses:
201:
description: voip carrier successfully created
@@ -2072,6 +2088,41 @@ paths:
description: credential successfully deleted
404:
description: credential not found
/ServiceProviders/{ServiceProviderSid}/SpeechCredentials/speech/supportedLanguagesAndVoices:
get:
tags:
- Service Providers
summary: get supported languages, voices and models
operationId: supportedLanguagesAndVoices
parameters:
- name: ServiceProviderSid
in: path
required: true
schema:
type: string
format: uuid
- name: vendor
in: query
required: true
schema:
type: string
- name: label
in: query
schema:
type: string
responses:
200:
description: get supported languages, voices and models
content:
application/json:
schema:
$ref: '#/components/schemas/SpeechLanguagesVoices'
500:
description: system error
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/ServiceProviders/{ServiceProviderSid}/SpeechCredentials/{SpeechCredentialSid}/test:
get:
tags:
@@ -2891,7 +2942,7 @@ paths:
tags:
- Accounts
summary: get a specific speech credential
operationId: getSpeechCredential
operationId: getSpeechCredentialByAccount
responses:
200:
description: retrieve speech credentials for a specified account
@@ -2905,7 +2956,7 @@ paths:
tags:
- Accounts
summary: update a speech credential
operationId: updateSpeechCredential
operationId: updateSpeechCredentialByAccount
requestBody:
content:
application/json:
@@ -2926,18 +2977,53 @@ paths:
tags:
- Accounts
summary: delete a speech credential
operationId: deleteSpeechCredential
operationId: deleteSpeechCredentialByAccount
responses:
204:
description: credential successfully deleted
404:
description: credential not found
/Accounts/{AccountSid}/SpeechCredentials/speech/supportedLanguagesAndVoices:
get:
tags:
- Accounts
summary: get supported languages, voices and models
operationId: supportedLanguagesAndVoicesByAccount
parameters:
- name: AccountSid
in: path
required: true
schema:
type: string
format: uuid
- name: vendor
in: query
required: true
schema:
type: string
- name: label
in: query
schema:
type: string
responses:
200:
description: get supported languages, voices and models
content:
application/json:
schema:
$ref: '#/components/schemas/SpeechLanguagesVoices'
500:
description: system error
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/Accounts/{AccountSid}/SpeechCredentials/{SpeechCredentialSid}/test:
get:
tags:
- Accounts
summary: test a speech credential
operationId: testSpeechCredential
operationId: testSpeechCredentialByAccount
parameters:
- name: AccountSid
in: path
@@ -3183,7 +3269,7 @@ paths:
tags:
- Service Providers
summary: retrieve pcap for a call
operationId: getRecentCallTrace
operationId: getRecentCallTraceBySP
responses:
200:
description: retrieve sip trace data
@@ -3269,7 +3355,7 @@ paths:
tags:
- Service Providers
summary: retrieve recent calls for an account
operationId: listRecentCalls
operationId: listRecentCallsBySP
responses:
200:
description: retrieve recent call records for a specified account
@@ -3370,7 +3456,7 @@ paths:
tags:
- Service Providers
summary: retrieve sip trace detail for a call
operationId: getRecentCallTrace
operationId: getRecentCallTraceByCallId
responses:
200:
description: retrieve sip trace data
@@ -3397,7 +3483,7 @@ paths:
tags:
- Accounts
summary: retrieve pcap for a call
operationId: getRecentCallTrace
operationId: getRecentCallTraceByAccount
responses:
200:
description: retrieve sip trace data
@@ -3583,7 +3669,7 @@ paths:
tags:
- Accounts
summary: retrieve alerts for an account
operationId: listAlerts
operationId: listAlertsByAccount
responses:
200:
description: retrieve alerts for a specified account
@@ -3799,7 +3885,7 @@ paths:
/Accounts/{AccountSid}/Calls:
post:
tags:
- Accounts
- Accounts
summary: create a call
operationId: createCall
parameters:
@@ -3858,6 +3944,10 @@ paths:
type: object
description: The customer SIP headers to associate with the call
example: {"X-Custom-Header": "Hello"}
sipRequestWithinDialogHook:
type: string
description: The sip indialog hook to receive session messages
example: '/customHook'
responses:
201:
description: call successfully created
@@ -4062,6 +4152,22 @@ paths:
type: string
siprecServerURL:
type: string
conferenceParticipantAction:
type: object
properties:
action:
type: string
enum:
- tag
- untag
- coach
- uncoach
- mute
- unmute
- hold
- unhold
tag:
type: string
responses:
200:
description: Accepted
@@ -4178,7 +4284,7 @@ paths:
tags:
- Accounts
summary: retrieve online sip users for an account
operationId: listQueues
operationId: listRegisteredSipUsers
responses:
200:
description: retrieve online sip users for an account
@@ -4188,8 +4294,35 @@ paths:
type: array
items:
type: string
post:
tags:
- Accounts
summary: retrieve online sip users for an account by list of sip username
operationId: listRegisteredSipUsersByUsername
requestBody:
content:
application/json:
schema:
type: array
items:
type: string
responses:
200:
description: retrieve online sip users for an account
content:
application/json:
schema:
type: array
items:
$ref: '#/components/schemas/RegisteredClient'
/Accounts/{AccountSid}/RegisteredSipUsers/{Client}:
parameters:
- name: AccountSid
in: path
required: true
schema:
type: string
format: uuid
- name: Client
in: path
required: true
@@ -4210,6 +4343,13 @@ paths:
schema:
$ref: '#/components/schemas/RegisteredClient'
/Accounts/{AccountSid}/TtsCache/Synthesize:
parameters:
- name: AccountSid
in: path
required: true
schema:
type: string
format: uuid
post:
tags:
- Accounts
@@ -4237,6 +4377,10 @@ paths:
type: string
description: voice ID
example: en-US-Standard-C
encodingMp3:
type: boolean
description: convert audio to mp3.
example: true
required:
- speech_credential_sid
- text
@@ -4415,6 +4559,69 @@ paths:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/Lcrs/{LcrSid}/Routes:
parameters:
- name: LcrSid
in: path
required: true
style: simple
explode: false
schema:
type: string
post:
tags:
- Lcrs
summary: Create least cost routing routes and carrier set entries
operationId: createLeastCostRoutingRoutesAndCarrierEntries
requestBody:
content:
application/json:
schema:
$ref: '#/components/schemas/LcrRoutes'
responses:
204:
description: least cost routing routes and carrier set entries created
400:
description: bad request
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
404:
description: least cost routing not found
500:
description: system error
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
put:
tags:
- Lcrs
summary: update least cost routing routes and carrier set entries
operationId: updateLeastCostRoutingRoutesAndCarrierEntries
requestBody:
content:
application/json:
schema:
$ref: '#/components/schemas/LcrRoutes'
responses:
204:
description: least cost routing ruoutes and carrier entries updated
400:
description: bad request
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
404:
description: least cost routing not found
500:
description: system error
content:
application/json:
schema:
$ref: '#/components/schemas/GeneralError'
/LcrRoutes:
post:
tags:
@@ -4881,17 +5088,32 @@ components:
scheme: bearer
bearerFormat: token
schemas:
SuccessfulLogin:
type: object
required:
- username
- password
properties:
token:
type: string
user_sid:
type: string
scope:
type: string
force_change:
type: boolean
Login:
type: object
properties:
user_sid:
username:
type: string
api_token:
type: string
change_password:
type: boolean
password:
type: string
required:
- user_sid
- username
- password
SuccessfulApiKeyAdd:
type: object
required:
@@ -5832,6 +6054,19 @@ components:
- lcr_route_sid
- voip_carrier_sid
- priority
LcrRouteAndCarrierEntries:
allOf:
- $ref: '#/components/schemas/LcrRoute'
- type: object
properties:
lcr_carrier_set_entries:
type: array
items:
$ref: '#/components/schemas/LcrCarrierSetEntry'
LcrRoutes:
type: array
items:
$ref: '#/components/schemas/LcrRouteAndCarrierEntries'
GoogleCustomVoice:
type: object
properties:
@@ -5886,6 +6121,70 @@ components:
- name
- reported_usage
- model
TtsModel:
type: object
properties:
name:
type: string
example: Turbo v2
value:
type: string
example: eleven_turbo_v2
LanguageVoice:
type: object
properties:
name:
type: string
example: Standard-A (Female)
value:
type: string
example: ar-XA-Standard-A
LanguageVoices:
type: object
properties:
name:
type: string
example: English (US)
value:
type: string
example: en-US
voices:
type: array
items:
$ref: '#/components/schemas/LanguageVoice'
SpeechLanguagesVoices:
type: object
properties:
tts:
type: array
items:
$ref: '#/components/schemas/LanguageVoices'
stt:
type: array
items:
$ref: '#/components/schemas/LanguageVoice'
ttsModel:
type: array
items:
$ref: '#/components/schemas/TtsModel'
UserList:
type: object
properties:
name:
type: string
email:
type: string
is_active:
type: boolean
force_change:
type: boolean
scope:
type: string
permissions:
type: array
items:
type: string
security:
- bearerAuth: []

View File

@@ -0,0 +1,22 @@
module.exports = [
{ name: 'Global English', value: 'en' },
{ name: 'Australian English', value: 'en_au' },
{ name: 'British English', value: 'en_uk' },
{ name: 'US English', value: 'en_us' },
{ name: 'Spanish', value: 'es' },
{ name: 'French', value: 'fr' },
{ name: 'German', value: 'de' },
{ name: 'Italian', value: 'it' },
{ name: 'Portuguese', value: 'pt' },
{ name: 'Dutch', value: 'nl' },
{ name: 'Hindi', value: 'hi' },
{ name: 'Japanese', value: 'ja' },
{ name: 'Chinese', value: 'zh' },
{ name: 'Finnish', value: 'fi' },
{ name: 'Korean', value: 'ko' },
{ name: 'Polish', value: 'pl' },
{ name: 'Russian', value: 'ru' },
{ name: 'Turkish', value: 'tr' },
{ name: 'Ukrainian', value: 'uk' },
{ name: 'Vietnamese', value: 'vi' },
];

View File

@@ -0,0 +1,10 @@
module.exports = [
{ name: 'Australian English', value: 'en-AU' },
{ name: 'British English', value: 'en-GB' },
{ name: 'US English', value: 'en-US' },
{ name: 'French', value: 'fr-FR' },
{ name: 'Canadian French', value: 'fr-CA' },
{ name: 'German', value: 'de-DE' },
{ name: 'Italian', value: 'it-IT' },
{ name: 'US Spanish', value: 'es-US' },
];

View File

@@ -0,0 +1,26 @@
module.exports = [
{
name: 'English US',
value: 'en_US-8khz',
},
{
name: 'English UK',
value: 'en_UK-8khz',
},
{
name: 'Spanish',
value: 'es_xx-8khz',
},
{
name: 'French',
value: 'fr_fr-8khz',
},
{
name: 'Russian',
value: 'ru_ru-8khz',
},
{
name: 'Portuguese',
value: 'pt_br-8khz',
},
];

View File

@@ -0,0 +1,138 @@
module.exports = [
{
name: 'Chinese - general',
value: 'zh',
},
{
name: 'Chinese (China)',
value: 'zh-CN',
},
{
name: 'Chinese (Taiwan)',
value: 'zh-TW',
},
{
name: 'Dutch - general',
value: 'nl',
},
{
name: 'English - general',
value: 'en',
},
{
name: 'English (Australia)',
value: 'en-AU',
},
{
name: 'English (United Kingdom)',
value: 'en-GB',
},
{
name: 'English (India)',
value: 'en-IN',
},
{
name: 'English (New Zealand)',
value: 'en-NZ',
},
{
name: 'English (United States)',
value: 'en-US',
},
{
name: 'French - general',
value: 'fr',
},
{
name: 'French (Canada)',
value: 'fr-CA',
},
{
name: 'German - general',
value: 'de',
},
{
name: 'Hindi - general',
value: 'hi',
},
{
name: 'Hindi (Roman Script)',
value: 'hi-Latin',
},
{
name: 'Indonesian - general',
value: 'in',
},
{
name: 'Italian - general',
value: 'it',
},
{
name: 'Japanese - general',
value: 'ja',
},
{
name: 'Korean - general',
value: 'ko',
},
{
name: 'Norwegian - general',
value: 'no',
},
{
name: 'Polish - general',
value: 'pl',
},
{
name: 'Portuguese - general',
value: 'pt',
},
{
name: 'Portuguese (Brazil)',
value: 'pt-BR',
},
{
name: 'Portuguese (Portugal)',
value: 'pt-PT',
},
{
name: 'Russian - general',
value: 'ru',
},
{
name: 'Spanish - general',
value: 'es',
},
{
name: 'Spanish (Latin America)',
value: 'es-419',
},
{
name: 'Swedish - general',
value: 'sv',
},
{
name: 'Turkish - general',
value: 'tr',
},
{
name: 'Ukrainian - general',
value: 'uk',
},
{
name: 'Flemish - general',
value: 'nl-BE',
},
{
name: 'Danish - general',
value: 'da',
},
{
name: 'Tamil - general',
value: 'ta',
},
{
name: 'Tamasheq - general',
value: 'taq',
},
];

View File

@@ -0,0 +1,130 @@
module.exports = [
{ name: 'Afrikaans (South Africa)', value: 'af-ZA' },
{ name: 'Albanian (Albania)', value: 'sq-AL' },
{ name: 'Amharic (Ethiopia)', value: 'am-ET' },
{ name: 'Arabic (Algeria)', value: 'ar-DZ' },
{ name: 'Arabic (Bahrain)', value: 'ar-BH' },
{ name: 'Arabic (Egypt)', value: 'ar-EG' },
{ name: 'Arabic (Iraq)', value: 'ar-IQ' },
{ name: 'Arabic (Israel)', value: 'ar-IL' },
{ name: 'Arabic (Jordan)', value: 'ar-JO' },
{ name: 'Arabic (Kuwait)', value: 'ar-KW' },
{ name: 'Arabic (Lebanon)', value: 'ar-LB' },
{ name: 'Arabic (Morocco)', value: 'ar-MA' },
{ name: 'Arabic (Oman)', value: 'ar-OM' },
{ name: 'Arabic (Qatar)', value: 'ar-QA' },
{ name: 'Arabic (Saudi Arabia)', value: 'ar-SA' },
{ name: 'Arabic (State of Palestine)', value: 'ar-PS' },
{ name: 'Arabic (Tunisia)', value: 'ar-TN' },
{ name: 'Arabic (United Arab Emirates)', value: 'ar-AE' },
{ name: 'Armenian (Armenia)', value: 'hy-AM' },
{ name: 'Azerbaijani (Azerbaijan)', value: 'az-AZ' },
{ name: 'Basque (Spain)', value: 'eu-ES' },
{ name: 'Bengali (Bangladesh)', value: 'bn-BD' },
{ name: 'Bengali (India)', value: 'bn-IN' },
{ name: 'Bulgarian (Bulgaria)', value: 'bg-BG' },
{ name: 'Burmese (Myanmar)', value: 'my-MM' },
{ name: 'Catalan (Spain)', value: 'ca-ES' },
{ name: 'Chinese, Cantonese (Traditional, Hong Kong)', value: 'yue-Hant-HK' },
{ name: 'Chinese, Mandarin (Simplified, China)', value: 'zh' },
{ name: 'Chinese, Mandarin (Simplified, Hong Kong)', value: 'zh-HK' },
{ name: 'Chinese, Mandarin (Simplified, Taiwan)', value: 'zh-TW' },
{ name: 'Croatian (Croatia)', value: 'hr-HR' },
{ name: 'Czech (Czech Republic)', value: 'cs-CZ' },
{ name: 'Danish (Denmark)', value: 'da-DK' },
{ name: 'Dutch (Belgium)', value: 'nl-BE' },
{ name: 'Dutch (Netherlands)', value: 'nl-NL' },
{ name: 'English (Australia)', value: 'en-AU' },
{ name: 'English (Canada)', value: 'en-CA' },
{ name: 'English (Ghana)', value: 'en-GH' },
{ name: 'English (India)', value: 'en-IN' },
{ name: 'English (Ireland)', value: 'en-IE' },
{ name: 'English (Kenya)', value: 'en-KE' },
{ name: 'English (New Zealand)', value: 'en-NZ' },
{ name: 'English (Nigeria)', value: 'en-NG' },
{ name: 'English (Philippines)', value: 'en-PH' },
{ name: 'English (Singapore)', value: 'en-SG' },
{ name: 'English (South Africa)', value: 'en-ZA' },
{ name: 'English (Tanzania)', value: 'en-TZ' },
{ name: 'English (United Kingdom)', value: 'en-GB' },
{ name: 'English (United States)', value: 'en-US' },
{ name: 'Estonian (Estonia)', value: 'et-EE' },
{ name: 'Filipino (Philippines)', value: 'fil-PH' },
{ name: 'Finnish (Finland)', value: 'fi-FI' },
{ name: 'French (Canada)', value: 'fr-CA' },
{ name: 'French (France)', value: 'fr-FR' },
{ name: 'Galician (Spain)', value: 'gl-ES' },
{ name: 'Georgian (Georgia)', value: 'ka-GE' },
{ name: 'German (Germany)', value: 'de-DE' },
{ name: 'Greek (Greece)', value: 'el-GR' },
{ name: 'Gujarati (India)', value: 'gu-IN' },
{ name: 'Hebrew (Israel)', value: 'he-IL' },
{ name: 'Hindi (India)', value: 'hi-IN' },
{ name: 'Hungarian (Hungary)', value: 'hu-HU' },
{ name: 'Icelandic (Iceland)', value: 'is-IS' },
{ name: 'Indonesian (Indonesia)', value: 'id-ID' },
{ name: 'Italian (Italy)', value: 'it-IT' },
{ name: 'Japanese (Japan)', value: 'ja-JP' },
{ name: 'Javanese (Indonesia)', value: 'jv-ID' },
{ name: 'Kannada (India)', value: 'kn-IN' },
{ name: 'Khmer (Cambodia)', value: 'km-KH' },
{ name: 'Korean (South Korea)', value: 'ko-KR' },
{ name: 'Lao (Laos)', value: 'lo-LA' },
{ name: 'Latvian (Latvia)', value: 'lv-LV' },
{ name: 'Lithuanian (Lithuania)', value: 'lt-LT' },
{ name: 'Macedonian (North Macedonia)', value: 'mk-MK' },
{ name: 'Malay (Malaysia)', value: 'ms-MY' },
{ name: 'Malayalam (India)', value: 'ml-IN' },
{ name: 'Marathi (India)', value: 'mr-IN' },
{ name: 'Mongolian (Mongolia)', value: 'mn-MN' },
{ name: 'Nepali (Nepal)', value: 'ne-NP' },
{ name: 'Norwegian Bokmål (Norway)', value: 'nb-NO' },
{ name: 'Persian (Iran)', value: 'fa-IR' },
{ name: 'Polish (Poland)', value: 'pl-PL' },
{ name: 'Portuguese (Brazil)', value: 'pt-BR' },
{ name: 'Portuguese (Portugal)', value: 'pt-PT' },
{ name: 'Punjabi (Gurmukhi, India)', value: 'pa-guru-IN' },
{ name: 'Romanian (Romania)', value: 'ro-RO' },
{ name: 'Russian (Russia)', value: 'ru-RU' },
{ name: 'Serbian (Serbia)', value: 'sr-RS' },
{ name: 'Sinhala (Sri Lanka)', value: 'si-LK' },
{ name: 'Slovak (Slovakia)', value: 'sk-SK' },
{ name: 'Slovenian (Slovenia)', value: 'sl-SI' },
{ name: 'Spanish (Argentina)', value: 'es-AR' },
{ name: 'Spanish (Bolivia)', value: 'es-BO' },
{ name: 'Spanish (Chile)', value: 'es-CL' },
{ name: 'Spanish (Colombia)', value: 'es-CO' },
{ name: 'Spanish (Costa Rica)', value: 'es-CR' },
{ name: 'Spanish (Dominican Republic)', value: 'es-DO' },
{ name: 'Spanish (Ecuador)', value: 'es-EC' },
{ name: 'Spanish (El Salvador)', value: 'es-SV' },
{ name: 'Spanish (Guatemala)', value: 'es-GT' },
{ name: 'Spanish (Honduras)', value: 'es-HN' },
{ name: 'Spanish (Mexico)', value: 'es-MX' },
{ name: 'Spanish (Nicaragua)', value: 'es-NI' },
{ name: 'Spanish (Panama)', value: 'es-PA' },
{ name: 'Spanish (Paraguay)', value: 'es-PY' },
{ name: 'Spanish (Peru)', value: 'es-PE' },
{ name: 'Spanish (Puerto Rico)', value: 'es-PR' },
{ name: 'Spanish (Spain)', value: 'es-ES' },
{ name: 'Spanish (United States)', value: 'es-US' },
{ name: 'Spanish (Uruguay)', value: 'es-UY' },
{ name: 'Spanish (Venezuela)', value: 'es-VE' },
{ name: 'Sundanese (Indonesia)', value: 'su-ID' },
{ name: 'Swahili (Kenya)', value: 'sw-KE' },
{ name: 'Swahili (Tanzania)', value: 'sw-TZ' },
{ name: 'Swedish (Sweden)', value: 'sv-SE' },
{ name: 'Tamil (India)', value: 'ta-IN' },
{ name: 'Tamil (Malaysia)', value: 'ta-MY' },
{ name: 'Tamil (Singapore)', value: 'ta-SG' },
{ name: 'Tamil (Sri Lanka)', value: 'ta-LK' },
{ name: 'Telugu (India)', value: 'te-IN' },
{ name: 'Thai (Thailand)', value: 'th-TH' },
{ name: 'Turkish (Turkey)', value: 'tr-TR' },
{ name: 'Ukrainian (Ukraine)', value: 'uk-UA' },
{ name: 'Urdu (India)', value: 'ur-IN' },
{ name: 'Urdu (Pakistan)', value: 'ur-PK' },
{ name: 'Uzbek (Uzbekistan)', value: 'uz-UZ' },
{ name: 'Vietnamese (Vietnam)', value: 'vi-VN' },
{ name: 'Zulu (South Africa)', value: 'zu-ZA' },
];

View File

@@ -0,0 +1,82 @@
module.exports = [
{
name: 'Arabic (Modern Standard)',
value: 'ar-MS_Telephony',
},
{
name: 'Chinese (Mandarin)',
value: 'zh-CN_Telephony',
},
{
name: 'Czech ',
value: 'cs-CZ_Telephony',
},
{
name: 'Dutch (Belgian)',
value: 'nl-BE_Telephony',
},
{
name: 'Dutch (Netherlands)',
value: 'nl-NL_Telephony',
},
{
name: 'English (all supported dialects)',
value: 'en-WW_Medical_Telephony',
},
{
name: 'English (Australian)',
value: 'en-AU_Telephony',
},
{
name: 'English (Indian)',
value: 'en-IN_Telephony',
},
{
name: 'English (United Kingdom)',
value: 'en-GB_Telephony',
},
{
name: 'English (United States)',
value: 'en-US_Telephony',
},
{
name: 'French (Canadian)',
value: 'fr-CA_Telephony',
},
{
name: 'French (France)',
value: 'fr-FR_Telephony',
},
{
name: 'German',
value: 'de-DE_Telephony',
},
{
name: 'Hindi (Indian)',
value: 'hi-IN_Telephony',
},
{
name: 'Italian',
value: 'it-IT_Telephony',
},
{
name: 'Korean',
value: 'ko-KR_Telephony',
},
{
name: 'Portuguese (Brazilian)',
value: 'pt-BR_Telephony',
},
{
name: 'Spanish (Mexican)',
value: 'es-LA_Telephony',
},
{
name: 'Spanish (Castilian)',
value: 'es-ES_Telephony',
},
{
name: 'Swedish ',
value: 'sv-SE_Telephony',
},
];

View File

@@ -0,0 +1,490 @@
module.exports = [
{
name: 'Afrikaans (South Africa)',
value: 'af-ZA',
},
{
name: 'Amharic (Ethiopia)',
value: 'am-ET',
},
{
name: 'Arabic (Algeria)',
value: 'ar-DZ',
},
{
name: 'Arabic (Bahrain)',
value: 'ar-BH',
},
{
name: 'Arabic (Egypt)',
value: 'ar-EG',
},
{
name: 'Arabic (Iraq)',
value: 'ar-IQ',
},
{
name: 'Arabic (Israel)',
value: 'ar-IL',
},
{
name: 'Arabic (Jordan)',
value: 'ar-JO',
},
{
name: 'Arabic (Kuwait)',
value: 'ar-KW',
},
{
name: 'Arabic (Lebanon)',
value: 'ar-LB',
},
{
name: 'Arabic (Libya)',
value: 'ar-LY',
},
{
name: 'Arabic (Morocco)',
value: 'ar-MA',
},
{
name: 'Arabic (Oman)',
value: 'ar-OM',
},
{
name: 'Arabic (Qatar)',
value: 'ar-QA',
},
{
name: 'Arabic (Saudi Arabia)',
value: 'ar-SA',
},
{
name: 'Arabic (Palestinian Authority)',
value: 'ar-PS',
},
{
name: 'Arabic (Syria)',
value: 'ar-SY',
},
{
name: 'Arabic (Tunisia)',
value: 'ar-TN',
},
{
name: 'Arabic (United Arab Emirates)',
value: 'ar-AE',
},
{
name: 'Arabic (Yemen)',
value: 'ar-YE',
},
{
name: 'Bulgarian (Bulgaria)',
value: 'bg-BG',
},
{
name: 'Bengali (India)',
value: 'bn-IN',
},
{
name: 'Catalan (Spain)',
value: 'ca-ES',
},
{
name: 'Chinese (Cantonese, Traditional)',
value: 'zh-HK',
},
{
name: 'Chinese (Mandarin, Simplified)',
value: 'zh-CN',
},
{
name: 'Chinese (Taiwanese Mandarin)',
value: 'zh-TW',
},
{
name: 'Croatian (Croatia)',
value: 'hr-HR',
},
{
name: 'Czech (Czech)',
value: 'cs-CZ',
},
{
name: 'Danish (Denmark)',
value: 'da-DK',
},
{
name: 'Dutch (Netherlands)',
value: 'nl-NL',
},
{
name: 'Dutch (Belgium)',
value: 'nl-BE',
},
{
name: 'English (Australia)',
value: 'en-AU',
},
{
name: 'English (Canada)',
value: 'en-CA',
},
{
name: 'English (Ghana)',
value: 'en-GH',
},
{
name: 'English (Hong Kong)',
value: 'en-HK',
},
{
name: 'English (India)',
value: 'en-IN',
},
{
name: 'English (Ireland)',
value: 'en-IE',
},
{
name: 'English (Kenya)',
value: 'en-KE',
},
{
name: 'English (New Zealand)',
value: 'en-NZ',
},
{
name: 'English (Nigeria)',
value: 'en-NG',
},
{
name: 'English (Philippines)',
value: 'en-PH',
},
{
name: 'English (Singapore)',
value: 'en-SG',
},
{
name: 'English (South Africa)',
value: 'en-ZA',
},
{
name: 'English (Tanzania)',
value: 'en-TZ',
},
{
name: 'English (United Kingdom)',
value: 'en-GB',
},
{
name: 'English (United States)',
value: 'en-US',
},
{
name: 'Estonian(Estonia)',
value: 'et-EE',
},
{
name: 'Filipino (Philippines)',
value: 'fil-PH',
},
{
name: 'Finnish (Finland)',
value: 'fi-FI',
},
{
name: 'French (Belgium)',
value: 'fr-BE',
},
{
name: 'French (Canada)',
value: 'fr-CA',
},
{
name: 'French (France)',
value: 'fr-FR',
},
{
name: 'French (Switzerland)',
value: 'fr-CH',
},
{
name: 'German (Austria)',
value: 'de-AT',
},
{
name: 'German (Switzerland)',
value: 'de-CH',
},
{
name: 'German (Germany)',
value: 'de-DE',
},
{
name: 'Greek (Greece)',
value: 'el-GR',
},
{
name: 'Gujarati (Indian)',
value: 'gu-IN',
},
{
name: 'Hebrew (Israel)',
value: 'he-IL',
},
{
name: 'Hindi (India)',
value: 'hi-IN',
},
{
name: 'Hungarian (Hungary)',
value: 'hu-HU',
},
{
name: 'Indonesian (Indonesia)',
value: 'id-ID',
},
{
name: 'Icelandic (Iceland)',
value: 'is-IS',
},
{
name: 'Irish (Ireland)',
value: 'ga-IE',
},
{
name: 'Italian (Italy)',
value: 'it-IT',
},
{
name: 'Japanese (Japan)',
value: 'ja-JP',
},
{
name: 'Javanese (Indonesia)',
value: 'jv-ID',
},
{
name: 'Kannada (India)',
value: 'kn-IN',
},
{
name: 'Khmer (Cambodia)',
value: 'km-KH',
},
{
name: 'Korean (Korea)',
value: 'ko-KR',
},
{
name: 'Latvian (Latvia)',
value: 'lv-LV',
},
{
name: 'Lao (Laos)',
value: 'lo-LA',
},
{
name: 'Lithuanian (Lithuania)',
value: 'lt-LT',
},
{
name: 'Malay (Malaysia)',
value: 'ms-MY',
},
{
name: 'Macedonian (North Macedonia)',
value: 'mk-MK',
},
{
name: 'Maltese (Malta)',
value: 'mt-MT',
},
{
name: 'Marathi (India)',
value: 'mr-IN',
},
{
name: 'Burmese (Myanmar)',
value: 'my-MM',
},
{
name: 'Norwegian (Bokmål, Norway)',
value: 'nb-NO',
},
{
name: 'Persian (Iran)',
value: 'fa-IR',
},
{
name: 'Polish (Poland)',
value: 'pl-PL',
},
{
name: 'Portuguese (Brazil)',
value: 'pt-BR',
},
{
name: 'Portuguese (Portugal)',
value: 'pt-PT',
},
{
name: 'Romanian (Romania)',
value: 'ro-RO',
},
{
name: 'Russian (Russia)',
value: 'ru-RU',
},
{
name: 'Slovak (Slovakia)',
value: 'sk-SK',
},
{
name: 'Slovenian (Slovenia)',
value: 'sl-SI',
},
{
name: 'Spanish (Argentina)',
value: 'es-AR',
},
{
name: 'Spanish (Bolivia)',
value: 'es-BO',
},
{
name: 'Spanish (Chile)',
value: 'es-CL',
},
{
name: 'Spanish (Colombia)',
value: 'es-CO',
},
{
name: 'Spanish (Costa Rica)',
value: 'es-CR',
},
{
name: 'Spanish (Cuba)',
value: 'es-CU',
},
{
name: 'Spanish (Dominican Republic)',
value: 'es-DO',
},
{
name: 'Spanish (Ecuador)',
value: 'es-EC',
},
{
name: 'Spanish (El Salvador)',
value: 'es-SV',
},
{
name: 'Spanish (Equatorial Guinea)',
value: 'es-GQ',
},
{
name: 'Spanish (Guatemala)',
value: 'es-GT',
},
{
name: 'Spanish (Honduras)',
value: 'es-HN',
},
{
name: 'Spanish (Mexico)',
value: 'es-MX',
},
{
name: 'Spanish (Nicaragua)',
value: 'es-NI',
},
{
name: 'Spanish (Panama)',
value: 'es-PA',
},
{
name: 'Spanish (Paraguay)',
value: 'es-PY',
},
{
name: 'Spanish (Peru)',
value: 'es-PE',
},
{
name: 'Spanish (Puerto Rico)',
value: 'es-PR',
},
{
name: 'Spanish (Spain)',
value: 'es-ES',
},
{
name: 'Spanish (Uruguay)',
value: 'es-UY',
},
{
name: 'Spanish (USA)',
value: 'es-US',
},
{
name: 'Spanish (Venezuela)',
value: 'es-VE',
},
{
name: 'Swahili (Kenya)',
value: 'sw-KE',
},
{
name: 'Swahili (Tanzania)',
value: 'sw-TZ',
},
{
name: 'Sinhala (Sri Lanka)',
value: 'si-LK',
},
{
name: 'Swedish (Sweden)',
value: 'sv-SE',
},
{
name: 'Serbian (Serbia)',
value: 'sr-RS',
},
{
name: 'Tamil (India)',
value: 'ta-IN',
},
{
name: 'Telugu (India)',
value: 'te-IN',
},
{
name: 'Thai (Thailand)',
value: 'th-TH',
},
{
name: 'Turkish (Turkey)',
value: 'tr-TR',
},
{
name: 'Ukrainian (Ukraine)',
value: 'uk-UA',
},
{
name: 'Uzbek (Uzbekistan)',
value: 'uz-UZ',
},
{
name: 'Zulu (South Africa)',
value: 'zu-ZA',
},
{
name: 'Vietnamese (Vietnam)',
value: 'vi-VN',
},
];

View File

@@ -0,0 +1,207 @@
module.exports = [
{
name: 'Arabic (Worldwide)',
value: 'ar-WW',
valueMix: 'ara-XWW',
},
{
name: 'Catalan (Spain)',
value: 'ca-ES',
valueMix: 'cat-ESP',
},
{
name: 'Croatian (Croatia)',
value: 'hr-HR',
valueMix: 'hrv-HRV',
},
{
name: 'Czech (Czech Republic)',
value: 'cs-CZ',
valueMix: 'ces-CZE',
},
{
name: 'Danish (Denmark)',
value: 'da-DK',
valueMix: 'dan-DNK',
},
{
name: 'Dutch (Netherlands)',
value: 'nl-NL',
valueMix: 'nld-NLD',
},
{
name: 'English (Australia)',
value: 'en-AU',
valueMix: 'eng-AUS',
},
{
name: 'English (United States)',
value: 'en-US',
valueMix: 'eng-USA',
},
{
name: 'English (India)',
value: 'en-IN',
valueMix: 'eng-IND',
},
{
name: 'English (United Kingdom)',
value: 'en-GB',
valueMix: 'eng-GBR',
},
{
name: 'Finnish (Finland)',
value: 'fi-FI',
valueMix: 'fin-FIN',
},
{
name: 'French (Canada)',
value: 'fr-CA',
valueMix: 'fra-CAN',
},
{
name: 'French (France)',
value: 'fr-FR',
valueMix: 'fra-FRA',
},
{
name: 'German (Germany)',
value: 'de-DE',
valueMix: 'deu-DEU',
},
{
name: 'Greek (Greece)',
value: 'el-GR',
valueMix: 'ell-GRC',
},
{
name: 'Hebrew (Israel)',
value: 'he-IL',
valueMix: 'heb-ISR',
},
{
name: 'Hindi (India)',
value: 'hi-IN',
valueMix: 'hin-IND',
},
{
name: 'Hungarian (Hungary)',
value: 'hu-HU',
valueMix: 'hun-HUN',
},
{
name: 'Indonesian (Indonesia)',
value: 'id-ID',
valueMix: 'ind-IDN',
},
{
name: 'Italian (Italy)',
value: 'it-IT',
valueMix: 'ita-ITA',
},
{
name: 'Japanese (Japan)',
value: 'ja-JP',
valueMix: 'jpn-JPN',
},
{
name: 'Korean (South Korea)',
value: 'ko-KR',
valueMix: 'kor-KOR',
},
{
name: 'Malay (Malaysia)',
value: 'ms-MY',
valueMix: 'zlm-MYS',
},
{
name: 'Norwegian (Norway)',
value: 'no-NO',
valueMix: 'nor-NOR',
},
{
name: 'Polish (Poland)',
value: 'pl-PL',
valueMix: 'pol-POL',
},
{
name: 'Portuguese (Brazil)',
value: 'pt-BR',
valueMix: 'por-BRA',
},
{
name: 'Portuguese (Portugal)',
value: 'pt-PT',
valueMix: 'por-PRT',
},
{
name: 'Romanian (Romania)',
value: 'ro-RO',
valueMix: 'ron-ROU',
},
{
name: 'Russian (Russia)',
value: 'ru-RU',
valueMix: 'rus-RUS',
},
{
name: 'Shanghainese (China)',
value: 'zh-WU',
valueMix: 'wuu-CHN',
},
{
name: 'Mandarin (China)',
value: 'zh-CN',
valueMix: 'cmn-CHN',
},
{
name: 'Slovak (Slovakia)',
value: 'sk-SK',
valueMix: 'slk-SVK',
},
{
name: 'Spanish (Spain)',
value: 'es-ES',
valueMix: 'spa-ESP',
},
{
name: 'Spanish (Latin America)',
value: 'es-US',
valueMix: 'spa-XLA',
},
{
name: 'Swedish (Sweden)',
value: 'sv-SE',
valueMix: 'swe-SWE',
},
{
name: 'Thai (Thailand)',
value: 'th-TH',
valueMix: 'tha-THA',
},
{
name: 'Cantonese (Hong Kong)',
value: 'cn-HK',
valueMix: 'yue-CHS',
},
{
name: 'Mandarin (Taiwan)',
value: 'zh-TW',
valueMix: 'cmn-TWN',
},
{
name: 'Turkish (Turkey)',
value: 'tr-TR',
valueMix: 'tur-TUR',
},
{
name: 'Ukrainian (Ukraine)',
value: 'uk-UA',
valueMix: 'ukr-UKR',
},
{
name: 'Vietnamese (Vietnam)',
value: 'vi-VN',
valueMix: 'vie-VNM',
},
];

View File

@@ -0,0 +1,58 @@
module.exports = [
{
name: 'Arabic',
value: 'ar-AR',
},
{
name: 'English',
value: 'en-US',
},
{
name: 'English - GB',
value: 'en-GB',
},
{
name: 'Spanish - US',
value: 'es-US',
},
{
name: 'Spanish',
value: 'es-ES',
},
{
name: 'German',
value: 'de-DE',
},
{
name: 'French',
value: 'fr-FR',
},
{
name: 'Hindi',
value: 'hi-IN',
},
{
name: 'Russian',
value: 'ru-RU',
},
{
name: 'Korean',
value: 'ko-KR',
},
{
name: 'Brazilian-Portuguese',
value: 'pt-BR',
},
{
name: 'Japanese',
value: 'ja-JP',
},
{
name: 'Italian',
value: 'it-IT',
},
{
name: 'Mandarin',
value: 'zh-CN',
},
];

View File

@@ -0,0 +1,6 @@
module.exports = [
{
name: 'English (United States)',
value: 'en-US',
},
];

View File

@@ -0,0 +1,213 @@
module.exports = [
{
value: 'arb',
name: 'Arabic',
voices: [{ value: 'Zeina', name: 'Zeina (Female)' }],
},
{
value: 'cmn-CN',
name: 'Chinese, Mandarin',
voices: [{ value: 'Zhiyu', name: 'Zhiyu (Female)' }],
},
{
value: 'da-DK',
name: 'Danish',
voices: [
{ value: 'Naja', name: 'Naja (Female)' },
{ value: 'Mads', name: 'Mads (Male)' },
],
},
{
value: 'nl-NL',
name: 'Dutch',
voices: [
{ value: 'Lotte', name: 'Lotte (Female)' },
{ value: 'Ruben', name: 'Ruben (Male)' },
],
},
{
value: 'en-AU',
name: 'English (Australian)',
voices: [
{ value: 'Nicole', name: 'Nicole (Female)' },
{ value: 'Russell', name: 'Russell (Male)' },
],
},
{
value: 'en-GB',
name: 'English (British)',
voices: [
{ value: 'Amy', name: 'Amy (Female)' },
{ value: 'Emma', name: 'Emma (Female)' },
{ value: 'Brian', name: 'Brian (Male)' },
],
},
{
value: 'en-IN',
name: 'English (Indian)',
voices: [
{ value: 'Aditi', name: 'Aditi (Female)' },
{ value: 'Raveena', name: 'Raveena (Female)' },
],
},
{
value: 'en-US',
name: 'English (US)',
voices: [
{ value: 'Joanna', name: 'Joanna (Female)' },
{ value: 'Kendra', name: 'Kendra (Female)' },
{ value: 'Kimberly', name: 'Kimberly (Female)' },
{ value: 'Ivy', name: 'Ivy (Female child)' },
{ value: 'Salli', name: 'Salli (Female)' },
{ value: 'Joey', name: 'Joey (Male)' },
{ value: 'Matthew', name: 'Matthew (Male)' },
{ value: 'Justin', name: 'Justin (Male child)' },
],
},
{
value: 'en-GB-WLS',
name: 'English (Welsh)',
voices: [{ value: 'Geraint', name: 'Geraint (Male)' }],
},
{
value: 'fr-FR',
name: 'French',
voices: [
{ value: 'Celine', name: 'Céline (Female)' },
{ value: 'Lea', name: 'Léa (Female)' },
{ value: 'Mathieu', name: 'Mathieu (Male)' },
],
},
{
value: 'fr-CA',
name: 'French (Canadian)',
voices: [{ value: 'Chantal', name: 'Chantal (Female)' }],
},
{
value: 'de-DE',
name: 'German',
voices: [
{ value: 'Marlene', name: 'Marlene (Female)' },
{ value: 'Vicki', name: 'Vicki (Female)' },
{ value: 'Hans', name: 'Hans (Male)' },
],
},
{
value: 'hi-IN',
name: 'Hindi',
voices: [{ value: 'Aditi', name: 'Aditi (Female)' }],
},
{
value: 'is-IS',
name: 'Icelandic',
voices: [
{ value: 'Dora', name: 'Dóra (Female)' },
{ value: 'Karl', name: 'Karl (Male)' },
],
},
{
value: 'it-IT',
name: 'Italian',
voices: [
{ value: 'Carla', name: 'Carla (Female)' },
{ value: 'Bianca', name: 'Bianca (Female)' },
{ value: 'Giorgio', name: 'Giorgio (Male)' },
],
},
{
value: 'ja-JP',
name: 'Japanese',
voices: [
{ value: 'Mizuki', name: 'Mizuki (Female)' },
{ value: 'Takumi', name: 'Takumi (Male)' },
],
},
{
value: 'ko-KR',
name: 'Korean',
voices: [{ value: 'Seoyeon', name: 'Seoyeon (Female)' }],
},
{
value: 'nb-NO',
name: 'Norwegian',
voices: [{ value: 'Liv', name: 'Liv (Female)' }],
},
{
value: 'pl-PL',
name: 'Polish',
voices: [
{ value: 'Ewa', name: 'Ewa (Female)' },
{ value: 'Maja', name: 'Maja (Female)' },
{ value: 'Jacek', name: 'Jacek (Male)' },
{ value: 'Jan', name: 'Jan (Male)' },
],
},
{
value: 'pt-BR',
name: 'Portuguese (Brazilian)',
voices: [
{ value: 'Camila', name: 'Camila (Female)' },
{ value: 'Vitoria', name: 'Vitória (Female)' },
{ value: 'Ricardo', name: 'Ricardo (Male)' },
],
},
{
value: 'pt-PT',
name: 'Portuguese (European)',
voices: [
{ value: 'Ines', name: 'Inês (Female)' },
{ value: 'Cristiano', name: 'Cristiano (Male)' },
],
},
{
value: 'ro-RO',
name: 'Romanian',
voices: [{ value: 'Carmen', name: 'Carmen (Female)' }],
},
{
value: 'ru-RU',
name: 'Russian',
voices: [
{ value: 'Tatyana', name: 'Tatyana (Female)' },
{ value: 'Maxim', name: 'Maxim (Male)' },
],
},
{
value: 'es-ES',
name: 'Spanish (European)',
voices: [
{ value: 'Conchita', name: 'Conchita (Female)' },
{ value: 'Lucia', name: 'Lucia (Female)' },
{ value: 'Enrique', name: 'Enrique (Male)' },
],
},
{
value: 'es-MX',
name: 'Spanish (Mexican)',
voices: [{ value: 'Mia', name: 'Mia (Female)' }],
},
{
value: 'es-US',
name: 'Spanish (US)',
voices: [
{ value: 'Lupe', name: 'Lupe (Female)' },
{ value: 'Penelope', name: 'Penélope (Female)' },
{ value: 'Miguel', name: 'Miguel (Male)' },
],
},
{
value: 'sv-SE',
name: 'Swedish',
voices: [{ value: 'Astrid', name: 'Astrid (Female)' }],
},
{
value: 'tr-TR',
name: 'Turkish',
voices: [{ value: 'Filiz', name: 'Filiz (Female)' }],
},
{
value: 'cy-GB',
name: 'Welsh',
voices: [{ value: 'Gwyneth', name: 'Gwyneth (Female)' }],
},
];

View File

@@ -0,0 +1,192 @@
module.exports = [
{
value: 'ar',
name: 'Arabic',
voices: [
{
value: 'pNInz6obpgDQGcFmaJgB',
name: 'Adam - american, deep, middle aged, male, narration',
},
{
value: 'ErXwobaYiN019PkySvjV',
name: 'Antoni - american, well-rounded, young, male, narration',
},
{
value: 'VR6AewLTigWG4xSOukaG',
name: 'Arnold - american, crisp, middle aged, male, narration',
},
{
value: 'EXAVITQu4vr4xnSDxMaL',
name: 'Bella - american, soft, young, female, narration',
},
{
value: 'N2lVS1w4EtoT3dr4eOWO',
name: 'Callum - american, hoarse, middle aged, male, video games',
},
{
value: 'IKne3meq5aSn9XLyUdCD',
name: 'Charlie - australian, casual, middle aged, male, conversational',
},
{
value: 'XB0fDUnXU5powFXDhCwa',
name: 'Charlotte - english-swedish, seductive, middle aged, female, video games',
},
{
value: '2EiwWnXFnvU5JabPnv8n',
name: 'Clyde - american, war veteran, middle aged, male, video games',
},
{
value: 'onwK4e9ZLuTAKqWW03F9',
name: 'Daniel - british, deep, middle aged, male, news presenter',
},
{
value: 'CYw3kZ02Hs0563khs1Fj',
name: 'Dave - british-essex, conversational, young, male, video games',
},
{
value: 'AZnzlk1XvdvUeBnXmlld',
name: 'Domi - american, strong, young, female, narration',
},
{
value: 'ThT5KcBeYPX3keUQqHPh',
name: "Dorothy - british, pleasant, young, female, children's stories",
},
{
value: 'MF3mGyEYCl7XYWbV9V6O',
name: 'Elli - american, emotional, young, female, narration',
},
{
value: 'LcfcDJNUP1GQjkzn1xUU',
name: 'Emily - american, calm, young, female, meditation',
},
{
value: 'g5CIjZEefAph4nQFvHAz',
name: 'Ethan - american, undefined, young, male, ASMR',
},
{
value: 'D38z5RcWu1voky8WS1ja',
name: 'Fin - irish, sailor, old, male, video games',
},
{
value: 'jsCqWAovK2LkecY7zXl4',
name: 'Freya - american, undefined, young, female, undefined',
},
{
value: 'jBpfuIE2acCO8z3wKNLl',
name: 'Gigi - american, childlish, young, female, animation',
},
{
value: 'zcAOhNBS3c14rBihAFp1',
name: 'Giovanni - english-italian, foreigner, young, male, audiobook',
},
{
value: 'z9fAnlkpzviPz146aGWa',
name: 'Glinda - american, witch, middle aged, female, video games',
},
{
value: 'oWAxZDx7w5VEj9dCyTzz',
name: 'Grace - american-southern, undefined, young, female, audiobook ',
},
{
value: 'SOYHLrjzK2X1ezoPC6cr',
name: 'Harry - american, anxious, young, male, video games',
},
{
value: 'ZQe5CZNOzWyzPSCn5a3c',
name: 'James - australian, calm , old, male, news',
},
{
value: 'bVMeCyTHy58xNoL34h3p',
name: 'Jeremy - american-irish, excited, young, male, narration',
},
{
value: 't0jbNlBVZ17f02VDIeMI',
name: 'Jessie - american, raspy , old, male, video games',
},
{
value: 'Zlb1dXrM653N07WRdFW3',
name: 'Joseph - british, undefined, middle aged, male, news',
},
{
value: 'TxGEqnHWrfWFTfGW9XjX',
name: 'Josh - american, deep, young, male, narration',
},
{
value: 'TX3LPaxmHKxFdv7VOQHJ',
name: 'Liam - american, undefined, young, male, narration',
},
{
value: 'XrExE9yKIg1WjnnlVkGX',
name: 'Matilda - american, warm, young, female, audiobook',
},
{
value: 'Yko7PKHZNXotIFUBG7I9',
name: 'Matthew - british, undefined, middle aged, male, audiobook',
},
{
value: 'flq6f7yk4E4fJM5XTYuZ',
name: 'Michael - american, undefined, old, male, audiobook',
},
{
value: 'zrHiDhphv9ZnVXBqCLjz',
name: 'Mimi - english-swedish, childish, young, female, animation',
},
{
value: 'piTKgcLEGmPE4e6mEKli',
name: 'Nicole - american, whisper, young, female, audiobook',
},
{
value: 'ODq5zmih8GrVes37Dizd',
name: 'Patrick - american, shouty, middle aged, male, video games',
},
{
value: '21m00Tcm4TlvDq8ikWAM',
name: 'Rachel - american, calm, young, female, narration',
},
{
value: 'wViXBPUzp2ZZixB1xQuM',
name: 'Ryan - american, soldier, middle aged, male, audiobook',
},
{
value: 'yoZ06aMxZJJ28mfd3POQ',
name: 'Sam - american, raspy, young, male, narration',
},
{
value: 'pMsXgVXv3BLzUgSXRplE',
name: 'Serena - american, pleasant, middle aged, female, interactive',
},
{
value: 'GBv7mTt0atIp3Br8iCZE',
name: 'Thomas - american, calm, young, male, meditation',
},
],
},
{ value: 'bg', name: 'Bulgarian', voices: [] },
{ value: 'zh', name: 'Chinese', voices: [] },
{ value: 'hr', name: 'Croatian', voices: [] },
{ value: 'cs', name: 'Czech', voices: [] },
{ value: 'da', name: 'Danish', voices: [] },
{ value: 'nl', name: 'Dutch', voices: [] },
{ value: 'en', name: 'English', voices: [] },
{ value: 'fil', name: 'Filipino', voices: [] },
{ value: 'fi', name: 'Finnish', voices: [] },
{ value: 'fr', name: 'French', voices: [] },
{ value: 'de', name: 'German', voices: [] },
{ value: 'el', name: 'Greek', voices: [] },
{ value: 'hi', name: 'Hindi', voices: [] },
{ value: 'id', name: 'Indonesian', voices: [] },
{ value: 'it', name: 'Italian', voices: [] },
{ value: 'ja', name: 'Japanese', voices: [] },
{ value: 'ko', name: 'Korean', voices: [] },
{ value: 'ms', name: 'Malay', voices: [] },
{ value: 'pl', name: 'Polish', voices: [] },
{ value: 'pt', name: 'Portuguese', voices: [] },
{ value: 'ro', name: 'Romanian', voices: [] },
{ value: 'ru', name: 'Russian', voices: [] },
{ value: 'sk', name: 'Slovak', voices: [] },
{ value: 'es', name: 'Spanish', voices: [] },
{ value: 'sv', name: 'Swedish', voices: [] },
{ value: 'ta', name: 'Tamil', voices: [] },
{ value: 'tr', name: 'Turkish', voices: [] },
{ value: 'uk', name: 'Ukrainian', voices: [] },
];

View File

@@ -0,0 +1,796 @@
module.exports = [
{
value: 'ar-XA',
name: 'Arabic',
voices: [
{ value: 'ar-XA-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ar-XA-Standard-B', name: 'Standard-B (Male)' },
{ value: 'ar-XA-Standard-C', name: 'Standard-C (Male)' },
{ value: 'ar-XA-Standard-D', name: 'Standard-D (Female)' },
{ value: 'ar-XA-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ar-XA-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'ar-XA-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'ar-XA-Wavenet-D', name: 'Wavenet-D (Female)' },
],
},
{
value: 'af-ZA',
name: 'Afrikaans (South Africa)',
voices: [{ value: 'af-ZA-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'bn-IN',
name: 'Bengali (India)',
voices: [
{ value: 'bn-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'bn-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'bn-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'bn-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
],
},
{
value: 'bg-BG',
name: 'Bulgarian (Bulgaria)',
voices: [{ value: 'bg-BG-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'ca-ES',
name: 'Catalan (Spain)',
voices: [{ value: 'ca-ES-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'cs-CZ',
name: 'Czech (Czech Republic)',
voices: [
{ value: 'cs-CZ-Standard-A', name: 'Standard-A (Female)' },
{ value: 'cs-CZ-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'da-DK',
name: 'Danish (Denmark)',
voices: [
{ value: 'da-DK-Standard-A', name: 'Standard-A (Female)' },
{ value: 'da-DK-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'da-DK-Neural2-D', name: 'Neural2-D (Female)' },
{ value: 'da-DK-Neural2-F', name: 'Neural2-F (Male)' },
{ value: 'da-DK-Standard-C', name: 'Standard-C (Male)' },
{ value: 'da-DK-Standard-D', name: 'Standard-D (Female)' },
{ value: 'da-DK-Standard-E', name: 'Standard-E (Female)' },
{ value: 'da-DK-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'da-DK-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'da-DK-Wavenet-E', name: 'Wavenet-E (Female)' },
],
},
{
value: 'eu-ES',
name: 'Basque (Spain)',
voices: [{ value: 'eu-ES-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'nl-NL',
name: 'Dutch (Netherlands)',
voices: [
{ value: 'nl-NL-Standard-A', name: 'Standard-A (Female)' },
{ value: 'nl-NL-Standard-B', name: 'Standard-B (Male)' },
{ value: 'nl-NL-Standard-C', name: 'Standard-C (Male)' },
{ value: 'nl-NL-Standard-D', name: 'Standard-D (Female)' },
{ value: 'nl-NL-Standard-E', name: 'Standard-E (Female)' },
{ value: 'nl-NL-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'nl-NL-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'nl-NL-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'nl-NL-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'nl-NL-Wavenet-E', name: 'Wavenet-E (Female)' },
],
},
{
value: 'en-AU',
name: 'English (Australia)',
voices: [
{ value: 'en-AU-Standard-A', name: 'Standard-A (Female)' },
{ value: 'en-AU-Standard-B', name: 'Standard-B (Male)' },
{ value: 'en-AU-Standard-C', name: 'Standard-C (Female)' },
{ value: 'en-AU-Standard-D', name: 'Standard-D (Male)' },
{ value: 'en-AU-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'en-AU-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'en-AU-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'en-AU-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'en-AU-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'en-AU-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'en-AU-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'en-AU-Neural2-D', name: 'Neural2-D (Male)' },
{ value: 'en-AU-Polyglot-1', name: 'Polyglot-1 (Male)' },
{ value: 'en-AU-News-E', name: 'News-E (Female)' },
{ value: 'en-AU-News-F', name: 'News-F (Female)' },
{ value: 'en-AU-News-G', name: 'News-G (Male)' },
],
},
{
value: 'en-IN',
name: 'English (India)',
voices: [
{ value: 'en-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'en-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'en-IN-Standard-C', name: 'Standard-C (Male)' },
{ value: 'en-IN-Standard-D', name: 'Standard-D (Female)' },
{ value: 'en-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'en-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'en-IN-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'en-IN-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'en-IN-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'en-IN-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'en-IN-Neural2-C', name: 'Neural2-C (Male)' },
{ value: 'en-IN-Neural2-D', name: 'Neural2-D (Female)' },
],
},
{
value: 'en-GB',
name: 'English (UK)',
voices: [
{ value: 'en-GB-Standard-A', name: 'Standard-A (Female)' },
{ value: 'en-GB-Standard-B', name: 'Standard-B (Male)' },
{ value: 'en-GB-Standard-C', name: 'Standard-C (Female)' },
{ value: 'en-GB-Standard-D', name: 'Standard-D (Male)' },
{ value: 'en-GB-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'en-GB-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'en-GB-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'en-GB-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'en-GB-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'en-GB-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'en-GB-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'en-GB-Neural2-D', name: 'Neural2-D (Male)' },
{ value: 'en-GB-Neural2-F', name: 'Neural2-F (Female)' },
{ value: 'en-GB-News-G', name: 'News-G (Female)' },
{ value: 'en-GB-News-H', name: 'News-H (Female)' },
{ value: 'en-GB-News-I', name: 'News-I (Female)' },
{ value: 'en-GB-News-J', name: 'News-J (Male)' },
{ value: 'en-GB-News-K', name: 'News-K (Male)' },
{ value: 'en-GB-News-L', name: 'News-L (Male)' },
{ value: 'en-GB-News-M', name: 'News-M (Male)' },
{ value: 'en-GB-Studio-B', name: 'Studio-B (Male)' },
{ value: 'en-GB-Studio-C', name: 'Studio-C (Female)' },
{ value: 'en-GB-Wavenet-F', name: 'Wavenet-F (Female)' },
{ value: 'en-GB-Standard-F', name: 'Standard-F (Female)' },
],
},
{
value: 'en-US',
name: 'English (US)',
voices: [
{ value: 'en-US-Standard-B', name: 'Standard-B (Male)' },
{ value: 'en-US-Standard-C', name: 'Standard-C (Female)' },
{ value: 'en-US-Standard-D', name: 'Standard-D (Male)' },
{ value: 'en-US-Standard-E', name: 'Standard-E (Female)' },
{ value: 'en-US-Wavenet-A', name: 'Wavenet-A (Male)' },
{ value: 'en-US-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'en-US-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'en-US-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'en-US-Wavenet-E', name: 'Wavenet-E (Female)' },
{ value: 'en-US-Wavenet-F', name: 'Wavenet-F (Female)' },
{ value: 'en-US-Neural2-A', name: 'Neural2-A (Male)' },
{ value: 'en-US-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'en-US-Neural2-D', name: 'Neural2-D (Male)' },
{ value: 'en-US-Neural2-E', name: 'Neural2-E (Female)' },
{ value: 'en-US-Neural2-F', name: 'Neural2-F (Female)' },
{ value: 'en-US-Neural2-G', name: 'Neural2-G (Female)' },
{ value: 'en-US-Neural2-H', name: 'Neural2-H (Female)' },
{ value: 'en-US-Neural2-I', name: 'Neural2-I (Male)' },
{ value: 'en-US-Neural2-J', name: 'Neural2-J (Male)' },
{ value: 'en-US-Studio-M', name: 'Studio-M (Male)' },
{ value: 'en-US-Studio-O', name: 'Studio-M (Female)' },
{ value: 'en-US-Polyglot-1', name: 'Polyglot-1 (Male)' },
{ value: 'en-US-News-K', name: 'News-K (Female)' },
{ value: 'en-US-News-L', name: 'News-L (Female)' },
{ value: 'en-US-News-M', name: 'News-M (Male)' },
{ value: 'en-US-News-N', name: 'News-N (Male)' },
{ value: 'en-US-Standard-A', name: 'Standard-A (Male)' },
{ value: 'en-US-Standard-F', name: 'Standard-F (Female)' },
{ value: 'en-US-Standard-G', name: 'Standard-G (Female)' },
{ value: 'en-US-Standard-H', name: 'Standard-H (Female)' },
{ value: 'en-US-Standard-I', name: 'Standard-I (Male)' },
{ value: 'en-US-Standard-J', name: 'Standard-J (Male)' },
{ value: 'en-US-Wavenet-G', name: 'Wavenet-G (Female)' },
{ value: 'en-US-Wavenet-H', name: 'Wavenet-H (Female)' },
{ value: 'en-US-Wavenet-I', name: 'Wavenet-I (Male)' },
{ value: 'en-US-Wavenet-J', name: 'Wavenet-J (Male)' },
],
},
{
value: 'fil-PH',
name: 'Filipino (Philippines)',
voices: [
{ value: 'fil-PH-Standard-A', name: 'Standard-A (Female)' },
{ value: 'fil-PH-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'fil-ph-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'fil-ph-Neural2-D', name: 'Neural2-A (Male)' },
{ value: 'fil-PH-Standard-B', name: 'Standard-B (Female)' },
{ value: 'fil-PH-Standard-C', name: 'Standard-C (Male)' },
{ value: 'fil-PH-Standard-D', name: 'Standard-D (Male)' },
{ value: 'fil-PH-Wavenet-B', name: 'Wavenet-B (Female)' },
{ value: 'fil-PH-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'fil-PH-Wavenet-D', name: 'Wavenet-D (Male)' },
],
},
{
value: 'fi-FI',
name: 'Finnish (Finland)',
voices: [
{ value: 'fi-FI-Standard-A', name: 'Standard-A (Female)' },
{ value: 'fi-FI-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'fr-CA',
name: 'French (Canada)',
voices: [
{ value: 'fr-CA-Standard-A', name: 'Standard-A (Female)' },
{ value: 'fr-CA-Standard-B', name: 'Standard-B (Male)' },
{ value: 'fr-CA-Standard-C', name: 'Standard-C (Female)' },
{ value: 'fr-CA-Standard-D', name: 'Standard-D (Male)' },
{ value: 'fr-CA-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'fr-CA-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'fr-CA-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'fr-CA-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'fr-CA-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'fr-CA-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'fr-CA-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'fr-CA-Neural2-D', name: 'Neural2-D (Male)' },
],
},
{
value: 'fr-FR',
name: 'French (France)',
voices: [
{ value: 'fr-FR-Standard-A', name: 'Standard-A (Female)' },
{ value: 'fr-FR-Standard-B', name: 'Standard-B (Male)' },
{ value: 'fr-FR-Standard-C', name: 'Standard-C (Female)' },
{ value: 'fr-FR-Standard-D', name: 'Standard-D (Male)' },
{ value: 'fr-FR-Standard-E', name: 'Standard-E (Female)' },
{ value: 'fr-FR-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'fr-FR-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'fr-FR-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'fr-FR-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'fr-FR-Wavenet-E', name: 'Wavenet-E (Female)' },
{ value: 'fr-FR-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'fr-FR-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'fr-FR-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'fr-FR-Neural2-D', name: 'Neural2-D (Male)' },
{ value: 'fr-FR-Neural2-E', name: 'Neural2-E (Female)' },
{ value: 'fr-FR-Polyglot-1', name: 'Polyglot-1 (Male)' },
{ value: 'fr-FR-Studio-A', name: 'Studio-A (Female)' },
{ value: 'fr-FR-Studio-D', name: 'Studio-D (Male)' },
],
},
{
value: 'de-DE',
name: 'German (Germany)',
voices: [
{ value: 'de-DE-Standard-A', name: 'Standard-A (Female)' },
{ value: 'de-DE-Standard-B', name: 'Standard-B (Male)' },
{ value: 'de-DE-Standard-E', name: 'Standard-E (Male)' },
{ value: 'de-DE-Standard-F', name: 'Standard-F (Female)' },
{ value: 'de-DE-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'de-DE-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'de-DE-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'de-DE-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'de-DE-Wavenet-E', name: 'Wavenet-E (Male)' },
{ value: 'de-DE-Wavenet-F', name: 'Wavenet-F (Female)' },
{ value: 'de-DE-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'de-DE-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'de-DE-Neural2-D', name: 'Neural2-D (Male)' },
{ value: 'de-DE-Neural2-F', name: 'Neural2-F (Female)' },
{ value: 'de-DE-Polyglot-1', name: 'Polyglot-1 (Male)' },
{ value: 'de-DE-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'de-DE-Standard-C', name: 'Standard-C (Female)' },
{ value: 'de-DE-Standard-D', name: 'Standard-D (Male)' },
{ value: 'de-DE-Studio-B', name: 'Studio-B (Male)' },
{ value: 'de-DE-Studio-C', name: 'Studio-C (Female)' },
],
},
{
value: 'el-GR',
name: 'Greek (Greece)',
voices: [
{ value: 'el-GR-Standard-A', name: 'Standard-A (Female)' },
{ value: 'el-GR-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'gl-ES',
name: 'Galician (Spain)',
voices: [{ value: 'gl-ES-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'gu-IN',
name: 'Gujarati (India)',
voices: [
{ value: 'gu-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'gu-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'gu-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'gu-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
],
},
{
value: 'he-IL',
name: 'Hebrew (Israel)',
voices: [
{ value: 'he-IL-Standard-A', name: 'Standard-A (Female)' },
{ value: 'he-IL-Standard-B', name: 'Standard-B (Male)' },
{ value: 'he-IL-Standard-C', name: 'Standard-C (Female)' },
{ value: 'he-IL-Standard-D', name: 'Standard-D (Male)' },
{ value: 'he-IL-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'he-IL-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'he-IL-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'he-IL-Wavenet-D', name: 'Wavenet-D (Male)' },
],
},
{
value: 'hi-IN',
name: 'Hindi (India)',
voices: [
{ value: 'hi-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'hi-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'hi-IN-Standard-C', name: 'Standard-C (Male)' },
{ value: 'hi-IN-Standard-D', name: 'Standard-D (Female)' },
{ value: 'hi-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'hi-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'hi-IN-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'hi-IN-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'hi-IN-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'hi-IN-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'hi-IN-Neural2-C', name: 'Neural2-C (Male)' },
{ value: 'hi-IN-Neural2-D', name: 'Neural2-D (Female)' },
],
},
{
value: 'hu-HU',
name: 'Hungarian (Hungary)',
voices: [
{ value: 'hu-HU-Standard-A', name: 'Standard-A (Female)' },
{ value: 'hu-HU-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'is-IS',
name: 'Icelandic (Iceland)',
voices: [{ value: 'is-IS-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'id-ID',
name: 'Indonesian (Indonesia)',
voices: [
{ value: 'id-ID-Standard-A', name: 'Standard-A (Female)' },
{ value: 'id-ID-Standard-B', name: 'Standard-B (Male)' },
{ value: 'id-ID-Standard-C', name: 'Standard-C (Male)' },
{ value: 'id-ID-Standard-D', name: 'Standard-D (Female)' },
{ value: 'id-ID-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'id-ID-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'id-ID-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'id-ID-Wavenet-D', name: 'Wavenet-D (Female)' },
],
},
{
value: 'it-IT',
name: 'Italian (Italy)',
voices: [
{ value: 'it-IT-Standard-A', name: 'Standard-A (Female)' },
{ value: 'it-IT-Standard-B', name: 'Standard-B (Female)' },
{ value: 'it-IT-Standard-C', name: 'Standard-C (Male)' },
{ value: 'it-IT-Standard-D', name: 'Standard-D (Male)' },
{ value: 'it-IT-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'it-IT-Wavenet-B', name: 'Wavenet-B (Female)' },
{ value: 'it-IT-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'it-IT-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'it-IT-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'it-IT-Neural2-C', name: 'Neural2-C (Male)' },
],
},
{
value: 'ja-JP',
name: 'Japanese (Japan)',
voices: [
{ value: 'ja-JP-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ja-JP-Standard-B', name: 'Standard-B (Female)' },
{ value: 'ja-JP-Standard-C', name: 'Standard-C (Male)' },
{ value: 'ja-JP-Standard-D', name: 'Standard-D (Male)' },
{ value: 'ja-JP-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ja-JP-Wavenet-B', name: 'Wavenet-B (Female)' },
{ value: 'ja-JP-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'ja-JP-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'ja-JP-Neural2-B', name: 'Neural2-B (Female)' },
{ value: 'ja-JP-Neural2-C', name: 'Neural2-C (Male)' },
{ value: 'ja-JP-Neural2-D', name: 'Neural2-D (Male)' },
],
},
{
value: 'kn-IN',
name: 'Kannada (India)',
voices: [
{ value: 'kn-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'kn-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'kn-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'kn-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
],
},
{
value: 'ko-KR',
name: 'Korean (South Korea)',
voices: [
{ value: 'ko-KR-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ko-KR-Standard-B', name: 'Standard-B (Female)' },
{ value: 'ko-KR-Standard-C', name: 'Standard-C (Male)' },
{ value: 'ko-KR-Standard-D', name: 'Standard-D (Male)' },
{ value: 'ko-KR-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ko-KR-Wavenet-B', name: 'Wavenet-B (Female)' },
{ value: 'ko-KR-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'ko-KR-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'ko-KR-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'ko-KR-Neural2-B', name: 'Neural2-B (Female)' },
{ value: 'ko-KR-Neural2-C', name: 'Neural2-C (Male)' },
],
},
{
value: 'lv-LV',
name: 'Latvian (Latvia)',
voices: [{ value: 'lv-LV-Standard-A', name: 'Standard-A (Male)' }],
},
{
value: 'lt-LT',
name: 'Lithuanian (Lithuania)',
voices: [{ value: 'lt-LT-Standard-A', name: 'Standard-A (Male)' }],
},
{
value: 'cmn-CN',
name: 'Mandarin Chinese',
voices: [
{ value: 'cmn-CN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'cmn-CN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'cmn-CN-Standard-C', name: 'Standard-C (Male)' },
{ value: 'cmn-CN-Standard-D', name: 'Standard-D (Female)' },
{ value: 'cmn-CN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'cmn-CN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'cmn-CN-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'cmn-CN-Wavenet-D', name: 'Wavenet-D (Female)' },
],
},
{
value: 'cmn-TW',
name: 'Mandarin Chinese (Traditional)',
voices: [
{ value: 'cmn-TW-Standard-A-Alpha', name: 'Standard-A-Alpha (Female)' },
{ value: 'cmn-TW-Standard-B-Alpha', name: 'Standard-B-Alpha (Male)' },
{ value: 'cmn-TW-Standard-C-Alpha', name: 'Standard-C-Alpha (Male)' },
{ value: 'cmn-TW-Wavenet-A-Alpha', name: 'Wavenet-A-Alpha (Female)' },
{ value: 'cmn-TW-Wavenet-B-Alpha', name: 'Wavenet-B-Alpha (Male)' },
{ value: 'cmn-TW-Wavenet-C-Alpha', name: 'Wavenet-C-Alpha (Male)' },
],
},
{
value: 'ms-MY',
name: 'Malay (Malaysia)',
voices: [
{ value: 'ms-MY-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ms-MY-Standard-B', name: 'Standard-B (Male)' },
{ value: 'ms-MY-Standard-C', name: 'Standard-C (Female)' },
{ value: 'ms-MY-Standard-D', name: 'Standard-D (Male)' },
{ value: 'ms-MY-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ms-MY-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'ms-MY-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'ms-MY-Wavenet-D', name: 'Wavenet-D (Male)' },
],
},
{
value: 'ml-IN',
name: 'Malayalam (India)',
voices: [
{ value: 'ml-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ml-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'ml-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ml-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'ml-IN-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'ml-IN-Wavenet-D', name: 'Wavenet-D (Male)' },
],
},
{
value: 'mr-IN',
name: 'Marathi (India)',
voices: [
{ value: 'mr-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'mr-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'mr-IN-Standard-C', name: 'Standard-C (Female)' },
{ value: 'mr-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'mr-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'mr-IN-Wavenet-C', name: 'Wavenet-C (Female)' },
],
},
{
value: 'nb-NO',
name: 'Norwegian (Norway)',
voices: [
{ value: 'nb-NO-Standard-A', name: 'Standard-A (Female)' },
{ value: 'nb-NO-Standard-B', name: 'Standard-B (Male)' },
{ value: 'nb-NO-Standard-C', name: 'Standard-C (Female)' },
{ value: 'nb-NO-Standard-D', name: 'Standard-D (Male)' },
{ value: 'nb-no-Standard-E', name: 'Standard-E (Female)' },
{ value: 'nb-NO-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'nb-NO-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'nb-NO-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'nb-NO-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'nb-no-Wavenet-E', name: 'Wavenet-E (Female)' },
],
},
{
value: 'nl-BE',
name: 'Dutch (Belgium)',
voices: [
{ value: 'nl-BE-Standard-A', name: 'Standard-A (Female)' },
{ value: 'nl-BE-Standard-B', name: 'Standard-B (Male)' },
{ value: 'nl-BE-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'nl-BE-Wavenet-B', name: 'Wavenet-B (Male)' },
],
},
{
value: 'pl-PL',
name: 'Polish (Poland)',
voices: [
{ value: 'pl-PL-Standard-A', name: 'Standard-A (Female)' },
{ value: 'pl-PL-Standard-B', name: 'Standard-B (Male)' },
{ value: 'pl-PL-Standard-C', name: 'Standard-C (Male)' },
{ value: 'pl-PL-Standard-D', name: 'Standard-D (Female)' },
{ value: 'pl-PL-Standard-E', name: 'Standard-E (Female)' },
{ value: 'pl-PL-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'pl-PL-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'pl-PL-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'pl-PL-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'pl-PL-Wavenet-E', name: 'Wavenet-E (Female)' },
],
},
{
value: 'pa-IN',
name: 'Punjabi (India)',
voices: [
{ value: 'pa-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'pa-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'pa-IN-Standard-C', name: 'Standard-C (Female)' },
{ value: 'pa-IN-Standard-D', name: 'Standard-D (Male)' },
{ value: 'pa-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'pa-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'pa-IN-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'pa-IN-Wavenet-D', name: 'Wavenet-D (Male)' },
],
},
{
value: 'pt-BR',
name: 'Portuguese (Brazil)',
voices: [
{ value: 'pt-BR-Standard-A', name: 'Standard-A (Female)' },
{ value: 'pt-BR-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'pt-BR-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'pt-BR-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'pt-BR-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'pt-BR-Standard-B', name: 'Standard-B (Male)' },
{ value: 'pt-BR-Standard-C', name: 'Standard-C (Female)' },
{ value: 'pt-BR-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'pt-BR-Wavenet-C', name: 'Wavenet-C (Female)' },
],
},
{
value: 'pt-PT',
name: 'Portuguese (Portugal)',
voices: [
{ value: 'pt-PT-Standard-A', name: 'Standard-A (Female)' },
{ value: 'pt-PT-Standard-B', name: 'Standard-B (Male)' },
{ value: 'pt-PT-Standard-C', name: 'Standard-C (Male)' },
{ value: 'pt-PT-Standard-D', name: 'Standard-D (Female)' },
{ value: 'pt-PT-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'pt-PT-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'pt-PT-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'pt-PT-Wavenet-D', name: 'Wavenet-D (Female)' },
],
},
{
value: 'ro-RO',
name: 'Romanian (Romania)',
voices: [
{ value: 'ro-RO-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ro-RO-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'ru-RU',
name: 'Russian (Russia)',
voices: [
{ value: 'ru-RU-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ru-RU-Standard-B', name: 'Standard-B (Male)' },
{ value: 'ru-RU-Standard-C', name: 'Standard-C (Female)' },
{ value: 'ru-RU-Standard-D', name: 'Standard-D (Male)' },
{ value: 'ru-RU-Standard-E', name: 'Standard-E (Female)' },
{ value: 'ru-RU-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ru-RU-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'ru-RU-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'ru-RU-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'ru-RU-Wavenet-E', name: 'Wavenet-E (Female)' },
],
},
{
value: 'sk-SK',
name: 'Slovak (Slovakia)',
voices: [
{ value: 'sk-SK-Standard-A', name: 'Standard-A (Female)' },
{ value: 'sk-SK-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'sr-RS',
name: 'Serbian (Cyrillic)',
voices: [{ value: 'sr-RS-Standard-A', name: 'Standard-A (Female)' }],
},
{
value: 'es-ES',
name: 'Spanish (Spain)',
voices: [
{ value: 'es-ES-Standard-A', name: 'Standard-A (Female)' },
{ value: 'es-ES-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'es-ES-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'es-ES-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'es-ES-Neural2-D', name: 'Neural2-D (Female)' },
{ value: 'es-ES-Neural2-E', name: 'Neural2-E (Female)' },
{ value: 'es-ES-Neural2-F', name: 'Neural2-F (Male)' },
{ value: 'es-ES-Polyglot-1', name: 'Polyglot-1 (Male)' },
{ value: 'es-ES-Standard-B', name: 'Standard-B (Male)' },
{ value: 'es-ES-Standard-C', name: 'Standard-C (Female)' },
{ value: 'es-ES-Standard-D', name: 'Standard-D (Female)' },
{ value: 'es-ES-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'es-ES-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'es-ES-Wavenet-D', name: 'Wavenet-D (Female)' },
],
},
{
value: 'es-US',
name: 'Spanish (US)',
voices: [
{ value: 'es-US-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'es-US-Neural2-B', name: 'Neural2-B (Male)' },
{ value: 'es-US-Neural2-C', name: 'Neural2-C (Male)' },
{ value: 'es-US-Studio-B', name: 'Studio-B (Male)' },
{ value: 'es-US-Polyglot-1', name: 'Polyglot-1 (Male)' },
{ value: 'es-US-News-D', name: 'News-D (Male)' },
{ value: 'es-US-News-E', name: 'News-E (Male)' },
{ value: 'es-US-News-F', name: 'News-F (Female)' },
{ value: 'es-US-News-G', name: 'News-G (Female)' },
{ value: 'es-US-Standard-A', name: 'Standard-A (Female)' },
{ value: 'es-US-Standard-B', name: 'Standard-B (Male)' },
{ value: 'es-US-Standard-C', name: 'Standard-C (Male)' },
{ value: 'es-US-Studio-B', name: 'Studio-B (Male)' },
{ value: 'es-US-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'es-US-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'es-US-Wavenet-C', name: 'Wavenet-C (Male)' },
],
},
{
value: 'sv-SE',
name: 'Swedish (Sweden)',
voices: [
{ value: 'sv-SE-Standard-A', name: 'Standard-A (Female)' },
{ value: 'sv-SE-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'sv-SE-Standard-B', name: 'Standard-B (Female)' },
{ value: 'sv-SE-Standard-C', name: 'Standard-C (Female)' },
{ value: 'sv-SE-Standard-D', name: 'Standard-D (Male)' },
{ value: 'sv-SE-Standard-E', name: 'Standard-E (Male)' },
{ value: 'sv-SE-Wavenet-B', name: 'Wavenet-B (Female)' },
{ value: 'sv-SE-Wavenet-C', name: 'Wavenet-C (Male)' },
{ value: 'sv-SE-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'sv-SE-Wavenet-E', name: 'Wavenet-E (Male)' },
],
},
{
value: 'ta-IN',
name: 'Tamil (India)',
voices: [
{ value: 'ta-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'ta-IN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'ta-IN-Standard-C', name: 'Standard-C (Female)' },
{ value: 'ta-IN-Standard-D', name: 'Standard-D (Male)' },
{ value: 'ta-IN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'ta-IN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'ta-IN-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'ta-IN-Wavenet-D', name: 'Wavenet-D (Male)' },
],
},
{
value: 'te-IN',
name: 'Telugu (India)',
voices: [
{ value: 'te-IN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'te-IN-Standard-B', name: 'Standard-B (Male)' },
],
},
{
value: 'tr-TR',
name: 'Turkish (Turkey)',
voices: [
{ value: 'tr-TR-Standard-A', name: 'Standard-A (Female)' },
{ value: 'tr-TR-Standard-B', name: 'Standard-B (Male)' },
{ value: 'tr-TR-Standard-C', name: 'Standard-C (Female)' },
{ value: 'tr-TR-Standard-D', name: 'Standard-D (Female)' },
{ value: 'tr-TR-Standard-E', name: 'Standard-E (Male)' },
{ value: 'tr-TR-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'tr-TR-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'tr-TR-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'tr-TR-Wavenet-D', name: 'Wavenet-D (Female)' },
{ value: 'tr-TR-Wavenet-E', name: 'Wavenet-E (Male)' },
],
},
{
value: 'uk-UA',
name: 'Ukrainian (Ukraine)',
voices: [
{ value: 'uk-UA-Standard-A', name: 'Standard-A (Female)' },
{ value: 'uk-UA-Wavenet-A', name: 'Wavenet-A (Female)' },
],
},
{
value: 'th-TH',
name: 'Thai (Thailand)',
voices: [
{ value: 'th-TH-Neural2-C', name: 'Neural2-C (Female)' },
{ value: 'th-TH-Standard-A', name: 'Standard-A (Female)' },
],
},
{
value: 'vi-VN',
name: 'Vietnamese (Vietnam)',
voices: [
{ value: 'vi-VN-Standard-A', name: 'Standard-A (Female)' },
{ value: 'vi-VN-Standard-B', name: 'Standard-B (Male)' },
{ value: 'vi-VN-Standard-C', name: 'Standard-C (Female)' },
{ value: 'vi-VN-Standard-D', name: 'Standard-D (Male)' },
{ value: 'vi-VN-Wavenet-A', name: 'Wavenet-A (Female)' },
{ value: 'vi-VN-Wavenet-B', name: 'Wavenet-B (Male)' },
{ value: 'vi-VN-Wavenet-C', name: 'Wavenet-C (Female)' },
{ value: 'vi-VN-Wavenet-D', name: 'Wavenet-D (Male)' },
{ value: 'vi-VN-Neural2-A', name: 'Neural2-A (Female)' },
{ value: 'vi-VN-Neural2-D', name: 'Neural2-D (Male)' },
],
},
{
value: 'yue-HK',
name: 'Chinese (Hong Kong)',
voices: [
{ value: 'yue-HK-Standard-A', name: 'Standard-A (Female)' },
{ value: 'yue-HK-Standard-B', name: 'Standard-B (Male)' },
{ value: 'yue-HK-Standard-C', name: 'Standard-C (Female)' },
{ value: 'yue-HK-Standard-D', name: 'Standard-D (Male)' },
],
},
];

View File

@@ -0,0 +1,167 @@
module.exports = [
{
value: 'de-DE',
name: 'German (Germany)',
voices: [
{ value: 'de-DE_DieterVoice', name: 'Dieter (Male): Standard German' },
{
value: 'de-DE_DieterV2Voice',
name: 'Dieter 2 (Male): Standard German',
},
{
value: 'de-DE_DieterV3Voice',
name: 'Dieter 3 (Male): Standard German',
},
{ value: 'de-DE_ErikaV3Voice', name: 'Erika (Female): Standard German' },
{ value: 'de-DE_BirgitVoice', name: 'Brigit (Female): Standard German' },
{
value: 'de-DE_BirgitV2Voice',
name: 'Brigit 2 (Female): Standard German',
},
{
value: 'de-DE_BirgitV3Voice',
name: 'Brigit 3 (Female): Standard German',
},
],
},
{
value: 'en-US',
name: 'English (US)',
voices: [
{
value: 'en-US_MichaelExpressive',
name: 'Michael (Male): American English - Expressive',
},
{ value: 'en-US_MichaelVoice', name: 'Michael (Male): American English' },
{
value: 'en-US_MichaelV2Voice',
name: 'Michael 2 (Male): American English',
},
{
value: 'en-US_MichaelV3Voice',
name: 'Michael 3 (Male): American English',
},
{ value: 'en-US_HenryV3Voice', name: 'Henry (Male): American English' },
{ value: 'en-US_EmilyV3Voice', name: 'Emily (Female): American English' },
{
value: 'en-US_OliviaV3Voice',
name: 'Olivia (Female): American English',
},
{
value: 'en-US_AllisonExpressive',
name: 'Allison (Female): American English - Expressive',
},
{
value: 'en-US_AllisonVoice',
name: 'Allison (Female): American English',
},
{
value: 'en-US_AllisonV2Voice',
name: 'Allison 2 (Female): American English',
},
{
value: 'en-US_AllisonV3Voice',
name: 'Allison 3 (Female): American English',
},
{
value: 'en-US_LisaExpressive',
name: 'Lisa (Female): American English - Expressive',
},
{ value: 'en-US_LisaVoice', name: 'Lisa (Female): American English' },
{ value: 'en-US_LisaV2Voice', name: 'Lisa 2 (Female): American English' },
{ value: 'en-US_LisaV3Voice', name: 'Lisa 3 (Female): American English' },
{ value: 'en-US_KevinV3Voice', name: 'Kevin (Male): American English' },
{
value: 'en-US_EmmaExpressive',
name: 'Emma (Female): American English - Expressive',
},
],
},
{
value: 'en-GB',
name: 'English (GB)',
voices: [
{ value: 'en-GB_JamesV3Voice', name: 'James (Male)' },
{ value: 'en-GB_KateVoice', name: 'Kate (Female)' },
{ value: 'en-GB_KateV3Voice', name: 'Kate 2 (Female)' },
{ value: 'en-GB_CharlotteV3Voice', name: 'Charlotte (Female)' },
],
},
{
value: 'es-US',
name: 'Spanish (North America)',
voices: [
{
value: 'es-US_SofiaVoice',
name: 'Sofia (Female): North American Spanish',
},
{
value: 'es-US_SofiaV3Voice',
name: 'Sofia 2 (Female): North American Spanish',
},
],
},
{
value: 'es-LA',
name: 'Spanish (Latin America)',
voices: [
{
value: 'es-LA_SofiaVoice',
name: 'Sofia (Female): Latin American Spanish',
},
{
value: 'es-LA_SofiaV3Voice',
name: 'Sofia 2 (Female): Latin American Spanish',
},
],
},
{
value: 'es-ES',
name: 'Spanish (Castilian)',
voices: [
{ value: 'es-ES_LauraVoice', name: 'Laura (Female)' },
{ value: 'es-ES_LauraV3Voice', name: 'Laura 2 (Female)' },
{ value: 'es-ES_EnriqueVoice', name: 'Enrique (Male)' },
{ value: 'es-ES_EnriqueV3Voice', name: 'Enrique 2 (Male)' },
],
},
{
value: 'fr-FR',
name: 'French (FR)',
voices: [
{ value: 'fr-FR_NicolasV3Voice', name: 'Nicolas (Male)' },
{ value: 'fr-FR_ReneeVoice', name: 'Renee (Female)' },
{ value: 'fr-FR_ReneeV3Voice', name: 'Renee 2 (Female)' },
],
},
{
value: 'fr-CA',
name: 'French (CA)',
voices: [{ value: 'fr-CA_LouiseV3Voice', name: 'Louise (Female)' }],
},
{
value: 'it-IT',
name: 'Italian',
voices: [
{ value: 'it-IT_FrancescaVoice', name: 'Francesca (Female)' },
{ value: 'it-IT_FrancescaV2Voice', name: 'Francesca 2 (Female)' },
{ value: 'it-IT_FrancescaV3Voice', name: 'Francesca 3 (Female)' },
],
},
{
value: 'pt-BR',
name: 'Portuguese (Brazil)',
voices: [
{ value: 'pt-BR_IsabelaVoice', name: 'Isabela (Female)' },
{ value: 'pt-BR_IsabelaV3Voice', name: 'Isabela 2 (Female)' },
],
},
{
value: 'ja-JP',
name: 'Japanese',
voices: [
{ value: 'ja-JP_EmiVoice', name: 'Emi (Female)' },
{ value: 'ja-JP_EmiV3Voice', name: 'Emi 2 (Female)' },
],
},
];

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,28 @@
const TtsAwsLanguagesVoiceRaw = require('./tts-microsoft-raw');
const languagesVoices = [];
TtsAwsLanguagesVoiceRaw.forEach((data) => {
const lang = languagesVoices.find((l) => {
return l.value === data.Locale;
});
if (!lang) {
languagesVoices.push({
value: data.Locale,
name: data.LocaleName,
voices: TtsAwsLanguagesVoiceRaw
.filter((d) => {
return d.Locale === data.Locale;
})
.map((d) => {
return {
value: d.ShortName,
name: `${d.DisplayName} (${d.Gender})`,
};
}),
});
}
});
module.exports = languagesVoices;

View File

@@ -0,0 +1,14 @@
module.exports = [
{ name: 'Asteria English (US) Female', value: 'aura-asteria-en' },
{ name: 'Luna English (US) Female', value: 'aura-luna-en' },
{ name: 'Stella English (US) Female', value: 'aura-stella-en' },
{ name: 'Stella English (UK) Female', value: 'aura-athena-en' },
{ name: 'Hera English (US) Female', value: 'aura-hera-en' },
{ name: 'Orion English (US) Male', value: 'aura-orion-en' },
{ name: 'Arcas English (US) Male', value: 'aura-arcas-en' },
{ name: 'Perseus English (US) Male', value: 'aura-perseus-en' },
{ name: 'Angus English (Ireland) Male', value: 'aura-angus-en' },
{ name: 'Orpheus English (US) Male', value: 'aura-orpheus-en' },
{ name: 'Helios English (UK) Male', value: 'aura-helios-en' },
{ name: 'Zeus English (US) Male', value: 'aura-zeus-en' },
];

View File

@@ -0,0 +1,8 @@
module.exports = [
{ name: 'Turbo v2', value: 'eleven_turbo_v2' },
{ name: 'Multilingual v2', value: 'eleven_multilingual_v2' },
{ name: 'Multilingual v1', value: 'eleven_multilingual_v1' },
{ name: 'English v1', value: 'eleven_monolingual_v1' },
{ name: 'English v2', value: 'eleven_english_sts_v2' },
];

View File

@@ -0,0 +1,6 @@
module.exports = [
{ name: 'PlayHT2.0-turbo', value: 'PlayHT2.0-turbo' },
{ name: 'PlayHT2.0', value: 'PlayHT2.0' },
{ name: 'PlayHT1.0', value: 'PlayHT1.0' },
];

View File

@@ -0,0 +1,5 @@
module.exports = [
{ name: 'Mist', value: 'mist' },
{ name: 'V1', value: 'v1' },
];

View File

@@ -0,0 +1,5 @@
module.exports = [
{ name: 'TTS-1', value: 'tts-1' },
{ name: 'TTS-1-HD', value: 'tts-1-hd' },
];

View File

@@ -0,0 +1,958 @@
module.exports = [
{
value: 'ar-WW',
name: 'Arabic (Worldwide)',
voices: [
{
value: 'Laila - standard',
name: 'Laila (standard)',
model: 'standard',
},
{
value: 'Tarik - standard',
name: 'Tarik (standard)',
model: 'standard',
},
{
value: 'Miriam - standard',
name: 'Miriam (standard)',
model: 'standard',
},
],
},
{
value: 'eu-ES',
name: 'Basque (Spain)',
voices: [{ value: 'Miren', name: 'Miren (standard)', model: 'standard' }],
},
{
value: 'bn-IN',
name: 'Bengali (India)',
voices: [
{ value: 'Paya - standard', name: 'Paya (standard)', model: 'standard' },
],
},
{
value: 'bho-IN',
name: 'Bhojpuri (India)',
voices: [
{ value: 'Jaya - standard', name: 'Jaya (standard)', model: 'standard' },
],
},
{
value: 'bg-BG',
name: 'Bulgarian (Bulgaria)',
voices: [
{
value: 'Daria - standard',
name: 'Daria (standard)',
model: 'standard',
},
],
},
{
value: 'yue-HK',
name: 'Cantonese (Hong Kong)',
voices: [
{
value: 'Sinji-Ml - standard',
name: 'Sinji-Ml (standard)',
model: 'standard',
},
],
},
{
value: 'ca-ES',
name: 'Catalan (Spain)',
voices: [
{
value: 'Jordi - standard',
name: 'Jordi (standard)',
model: 'standard',
},
{
value: 'Montserrat - standard',
name: 'Montserrat (standard)',
model: 'enhanced',
},
],
},
{
value: 'yue-HK',
name: 'Croatian (Croatia)',
voices: [
{ value: 'Lana - standard', name: 'Lana (standard)', model: 'standard' },
],
},
{
value: 'cs-CZ',
name: 'Czech (Czech Republic)',
voices: [
{
value: 'Iveta - standard',
name: 'Iveta (standard)',
model: 'standard',
},
{
value: 'Zuzana - standard',
name: 'Zuzana (standard)',
model: 'standard',
},
{
value: 'Zuzana-ml - enhanced',
name: 'Zuzana (enhanced)',
model: 'enhanced',
},
],
},
{
value: 'da-DK',
name: 'Danish (Denmark)',
voices: [
{
value: 'Magnus - standard',
name: 'Magnus (standard)',
model: 'standard',
},
{ value: 'Sara - standard', name: 'Sara (standard)', model: 'standard' },
],
},
{
value: 'nl-BE',
name: 'Dutch (Belgium)',
voices: [
{
value: 'Ellen - standard',
name: 'Ellen (standard)',
model: 'standard',
},
],
},
{
value: 'nl-NL',
name: 'Dutch (Belgium)',
voices: [
{
value: 'Claire-Ml - standard',
name: 'Claire-Ml (standard)',
model: 'standard',
},
{
value: 'Xander - standard',
name: 'Xander (standard)',
model: 'standard',
},
],
},
{
value: 'en-AU',
name: 'English (Australia)',
voices: [
{
value: 'Karen - standard',
name: 'Karen (standard)',
model: 'standard',
},
{ value: 'Lee - standard', name: 'Lee (standard)', model: 'standard' },
{
value: 'Matilda - enhanced',
name: 'Matilda (enhanced)',
model: 'enhanced',
},
],
},
{
value: 'en-IN',
name: 'English (India)',
voices: [
{
value: 'Isha-Ml - enhanced',
name: 'Isha-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Rishi - standard',
name: 'Rishi (standard)',
model: 'standard',
},
{
value: 'Rishi-Ml - standard',
name: 'Rishi-Ml (standard)',
model: 'standard',
},
{
value: 'Sangeeta - standard',
name: 'Sangeeta (standard)',
model: 'standard',
},
{
value: 'Veena - standard',
name: 'Veena (standard)',
model: 'standard',
},
],
},
{
value: 'en-IE',
name: 'English (Ireland)',
voices: [
{
value: 'Moira - standard',
name: 'Moira (standard)',
model: 'standard',
},
],
},
{
value: 'en-SC',
name: 'English (Scotland)',
voices: [
{
value: 'Fiona - standard',
name: 'Fiona (standard)',
model: 'standard',
},
],
},
{
value: 'en-ZA',
name: 'English (South Africa)',
voices: [
{
value: 'Tessa - standard',
name: 'Tessa (standard)',
model: 'standard',
},
],
},
{
value: 'en-GB',
name: 'English (United Kingdom)',
voices: [
{
value: 'Daniel - standard',
name: 'Daniel (standard)',
model: 'standard',
},
{ value: 'Kate - standard', name: 'Kate (standard)', model: 'standard' },
{
value: 'Malcolm - standard',
name: 'Malcolm (standard)',
model: 'standard',
},
{
value: 'Oliver - standard',
name: 'Oliver (standard)',
model: 'standard',
},
{
value: 'Serena - enhanced',
name: 'Serena (enhanced)',
model: 'enhanced',
},
{
value: 'Simon - standard',
name: 'Simon (standard)',
model: 'standard',
},
{
value: 'Stephanie - standard',
name: 'Stephanie (standard)',
model: 'standard',
},
],
},
{
value: 'en-US',
name: 'English (United States)',
voices: [
{
value: 'Allison - standard',
name: 'Allison (standard)',
model: 'standard',
},
{
value: 'Ava-Ml - enhanced',
name: 'Ava-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Chloe - standard',
name: 'Chloe (standard)',
model: 'standard',
},
{ value: 'Evan - enhanced', name: 'Evan (enhanced)', model: 'enhanced' },
{
value: 'Nathan - enhanced',
name: 'Nathan (enhanced)',
model: 'enhanced',
},
{
value: 'Evelyn - standard',
name: 'Evelyn (standard)',
model: 'standard',
},
{
value: 'Nolan - standard',
name: 'Nolan (standard)',
model: 'standard',
},
{
value: 'Samantha - standard',
name: 'Samantha (standard)',
model: 'standard',
},
{
value: 'Susan - standard',
name: 'Susan (standard)',
model: 'standard',
},
{ value: 'Tom - standard', name: 'Tom (standard)', model: 'standard' },
{
value: 'Zoe-Ml - enhanced',
name: 'Zoe-Ml (enhanced)',
model: 'enhanced',
},
],
},
{
value: 'fi-FI',
name: 'Finnish (Finland)',
voices: [
{ value: 'Onni - standard', name: 'Onni (standard)', model: 'standard' },
{ value: 'Satu - standard', name: 'Satu (standard)', model: 'standard' },
],
},
{
value: 'fr-BE',
name: 'French (Belgium)',
voices: [
{ value: 'Aude - standard', name: 'Aude (standard)', model: 'standard' },
],
},
{
value: 'fr-CA',
name: 'French (Canada)',
voices: [
{
value: 'Amelie-Ml - enhanced',
name: 'Amelie-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Chantal - standard',
name: 'Chantal (standard)',
model: 'standard',
},
{
value: 'Nicolas - standard',
name: 'Nicolas (standard)',
model: 'standard',
},
],
},
{
value: 'fr-FR',
name: 'French (France)',
voices: [
{
value: 'Audrey-Ml - enhanced',
name: 'Audrey-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Aurelie - standard',
name: 'Aurelie (standard)',
model: 'standard',
},
{
value: 'Thomas - standard',
name: 'Thomas (standard)',
model: 'standard',
},
],
},
{
value: 'gl-ES',
name: 'Galician (Spain)',
voices: [
{
value: 'Carmela - standard',
name: 'Carmela (standard)',
model: 'standard',
},
],
},
{
value: 'de-DE',
name: 'German (Germany)',
voices: [
{
value: 'Anna-Ml - enhanced',
name: 'Anna-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Markus - standard',
name: 'Markus (standard)',
model: 'standard',
},
{
value: 'Petra-Ml - enhanced',
name: 'Petra-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Viktor - standard',
name: 'Viktor (standard)',
model: 'standard',
},
{
value: 'Yannick - standard',
name: 'Yannick (standard)',
model: 'standard',
},
],
},
{
value: 'el-GR',
name: 'Greek (Greece)',
voices: [
{
value: 'Melina - standard',
name: 'Melina (standard)',
model: 'standard',
},
{
value: 'Nikos - standard',
name: 'Nikos (standard)',
model: 'standard',
},
],
},
{
value: 'he-IL',
name: 'Hebrew (Israel)',
voices: [
{
value: 'Carmit - standard',
name: 'Carmit (standard)',
model: 'standard',
},
],
},
{
value: 'hi-IN',
name: 'Hindi (India)',
voices: [
{
value: 'Kiyara-Ml - enhanced',
name: 'Kiyara-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Lekha - standard',
name: 'Lekha (standard)',
model: 'standard',
},
{ value: 'neel - standard', name: 'Neel (standard)', model: 'standard' },
{
value: 'Neel-Ml - standard',
name: 'Neel-Ml (standard)',
model: 'standard',
},
],
},
{
value: 'hu-HU',
name: 'Hungarian (Hungary)',
voices: [
{
value: 'Mariska - standard',
name: 'Mariska (standard)',
model: 'standard',
},
],
},
{
value: 'id-ID',
name: 'Indonesian (Indonesia)',
voices: [
{
value: 'Damayanti - standard',
name: 'Damayanti (standard)',
model: 'standard',
},
],
},
{
value: 'it-IT',
name: 'Italian (Italy)',
voices: [
{ value: 'Emma - enhanced', name: 'Emma (enhanced)', model: 'enhanced' },
{
value: 'Federica-Ml - standard',
name: 'Federica-Ml (standard)',
model: 'standard',
},
{ value: 'Luca - standard', name: 'Luca (standard)', model: 'standard' },
{
value: 'Neel-Ml - standard',
name: 'Neel-Ml (standard)',
model: 'standard',
},
{
value: 'Paola - standard',
name: 'Paola (standard)',
model: 'standard',
},
],
},
{
value: 'ja-JP',
name: 'Japanese (Japan)',
voices: [
{
value: 'Ayane - standard',
name: 'Ayane (standard)',
model: 'standard',
},
{
value: 'Daisuke - standard',
name: 'Daisuke (standard)',
model: 'standard',
},
{
value: 'Ichiro - standard',
name: 'Ichiro (standard)',
model: 'standard',
},
{
value: 'Koharu - standard',
name: 'Koharu (standard)',
model: 'standard',
},
{
value: 'Kyoko - standard',
name: 'Kyoko (standard)',
model: 'standard',
},
{
value: 'Mizuki - standard',
name: 'Mizuki (standard)',
model: 'standard',
},
{
value: 'Otoya - standard',
name: 'Otoya (standard)',
model: 'standard',
},
{
value: 'Sakura - standard',
name: 'Sakura (standard)',
model: 'standard',
},
{
value: 'Seiji - standard',
name: 'Seiji (standard)',
model: 'standard',
},
],
},
{
value: 'kn-IN',
name: 'Kannada (India)',
voices: [
{
value: 'Alpana - standard',
name: 'Alpana (standard)',
model: 'standard',
},
],
},
{
value: 'ko-KR',
name: 'Korean (South Korea)',
voices: [
{ value: 'Jina - enhanced', name: 'Jina (enhanced)', model: 'enhanced' },
{ value: 'Sora - standard', name: 'Sora (standard)', model: 'standard' },
{ value: 'Yuna - standard', name: 'Yuna (standard)', model: 'standard' },
{
value: 'Yuna-Ml - enhanced',
name: 'Yuna-Ml (enhanced)',
model: 'enhanced',
},
],
},
{
value: 'zlm-MY',
name: 'Malay (Malaysia)',
voices: [
{
value: 'Amira - standard',
name: 'Amira (standard)',
model: 'standard',
},
],
},
{
value: 'zh-CN',
name: 'Mandarin (China)',
voices: [
{
value: 'Lili-Ml - enhanced',
name: 'Lili-Ml (enhanced)',
model: 'enhanced',
},
{
value: 'Binbin-Ml - standard',
name: 'Binbin-Ml (standard)',
model: 'standard',
},
{
value: 'Lilian-Ml - standard',
name: 'Lilian-Ml (standard)',
model: 'standard',
},
{
value: 'Lisheng-Ml - standard',
name: 'Lisheng-Ml (standard)',
model: 'standard',
},
{
value: 'Tiantian-Ml - standard',
name: 'Tiantian-Ml (standard)',
model: 'standard',
},
{
value: 'Tingting-Ml - standard',
name: 'Tingting-Ml (standard)',
model: 'standard',
},
],
},
{
value: 'cmn-TW',
name: 'Mandarin (Taiwan)',
voices: [
{
value: 'Meijia-Ml - standard',
name: 'Meijia-Ml (standard)',
model: 'standard',
},
],
},
{
value: 'mr-IN',
name: 'Marathi (India)',
voices: [
{
value: 'Ananya - standard',
name: 'Ananya (standard)',
model: 'standard',
},
],
},
{
value: 'nb-NO',
name: 'Norwegian Bokmål (Norway)',
voices: [
{
value: 'Henrik - standard',
name: 'Henrik (standard)',
model: 'standard',
},
{ value: 'Nora - standard', name: 'Nora (standard)', model: 'standard' },
],
},
{
value: 'pl-PL',
name: 'Polish (Poland)',
voices: [
{ value: 'Ewa - enhanced', name: 'Ewa (enhanced)', model: 'enhanced' },
{
value: 'Krzysztof - standard',
name: 'Krzysztof (standard)',
model: 'standard',
},
{
value: 'Zosia - standard',
name: 'Zosia (standard)',
model: 'standard',
},
],
},
{
value: 'pt-BR',
name: 'Portuguese (Brazil)',
voices: [
{
value: 'luciana - enhanced',
name: 'Luciana (enhanced)',
model: 'enhanced',
},
{
value: 'Fernanda - standard',
name: 'Fernanda (standard)',
model: 'standard',
},
{
value: 'Felipe - standard',
name: 'Felipe (standard)',
model: 'standard',
},
],
},
{
value: 'pt-PT',
name: 'Portuguese (Portugal)',
voices: [
{
value: 'Catarina - standard',
name: 'Catarina (standard)',
model: 'standard',
},
{
value: 'Joana - standard',
name: 'Joana (standard)',
model: 'standard',
},
{
value: 'Joaquim - standard',
name: 'Joaquim (standard)',
model: 'standard',
},
],
},
{
value: 'ro-RO',
name: 'Romanian (Romania)',
voices: [
{
value: 'Ioana - standard',
name: 'Ioana (standard)',
model: 'standard',
},
],
},
{
value: 'ru-RU',
name: 'Russian (Russia)',
voices: [
{
value: 'Katya - standard',
name: 'Katya (standard)',
model: 'standard',
},
{
value: 'Katya-Ml - standard',
name: 'Katya-Ml (standard)',
model: 'standard',
},
{
value: 'Milena - standard',
name: 'Milena (standard)',
model: 'standard',
},
{ value: 'yuri - standard', name: 'Yuri (standard)', model: 'standard' },
],
},
{
value: 'sk-SK',
name: 'Slovak (Slovakia)',
voices: [
{
value: 'Laura - standard',
name: 'Laura (standard)',
model: 'standard',
},
],
},
{
value: 'es-AR',
name: 'Spanish (Argentina)',
voices: [
{
value: 'Diego - standard',
name: 'Diego (standard)',
model: 'standard',
},
{
value: 'Isabela - standard',
name: 'Isabela (standard)',
model: 'standard',
},
],
},
{
value: 'es-CL',
name: 'Spanish (Chile)',
voices: [
{
value: 'Francisca - standard',
name: 'Francisca (standard)',
model: 'standard',
},
],
},
{
value: 'es-CO',
name: 'Spanish (Colombia)',
voices: [
{
value: 'Carlos - standard',
name: 'Carlos (standard)',
model: 'standard',
},
{
value: 'Soledad - standard',
name: 'Soledad (standard)',
model: 'standard',
},
{
value: 'Ximena - standard',
name: 'Ximena (standard)',
model: 'standard',
},
],
},
{
value: 'es-MX',
name: 'Spanish (Mexico)',
voices: [
{
value: 'Angelica - standard',
name: 'Angelica (standard)',
model: 'standard',
},
{
value: 'Javier - standard',
name: 'Javier (standard)',
model: 'standard',
},
{ value: 'Juan - standard', name: 'Juan (standard)', model: 'standard' },
{
value: 'Paulina-Ml - enhanced',
name: 'Paulina-Ml (enhanced)',
model: 'enhanced',
},
],
},
{
value: 'es-ES',
name: 'Spanish (Spain)',
voices: [
{
value: 'Jorge - standard',
name: 'Jorge (standard)',
model: 'standard',
},
{
value: 'Marisol-Ml - standard',
name: 'Marisol-Ml (standard)',
model: 'standard',
},
{
value: 'Monica-Ml - standard',
name: 'Monica-Ml (standard)',
model: 'standard',
},
],
},
{
value: 'sv-SE',
name: 'Swedish (Sweden)',
voices: [
{ value: 'Alva - standard', name: 'Alva (standard)', model: 'standard' },
{
value: 'Klara - standard',
name: 'Klara (standard)',
model: 'standard',
},
{
value: 'Oskar - standard',
name: 'Oskar (standard)',
model: 'standard',
},
],
},
{
value: 'ta-IN',
name: 'Tamil (India)',
voices: [
{ value: 'Vani - standard', name: 'Vani (standard)', model: 'standard' },
],
},
{
value: 'te-IN',
name: 'Telugu (India)',
voices: [
{
value: 'Geeta - standard',
name: 'Geeta (standard)',
model: 'standard',
},
],
},
{
value: 'th-TH',
name: 'Thai (Thailand)',
voices: [
{
value: 'Kanya - enhanced',
name: 'Kanya (enhanced)',
model: 'enhanced',
},
{
value: 'Narisa - standard',
name: 'Narisa (standard)',
model: 'standard',
},
],
},
{
value: 'tr-TR',
name: 'Turkish (Turkey)',
voices: [
{
value: 'Cem-Ml - standard',
name: 'Cem-Ml (standard)',
model: 'standard',
},
{
value: 'Yelda - standard',
name: 'Yelda (standard)',
model: 'standard',
},
],
},
{
value: 'uk-UA',
name: 'Ukrainian (Ukraine)',
voices: [
{
value: 'Lesya - standard',
name: 'Lesya (standard)',
model: 'standard',
},
],
},
{
value: 'va-ES',
name: 'Valencian (Spain)',
voices: [
{
value: 'Empar - standard',
name: 'Empar (standard)',
model: 'standard',
},
],
},
{
value: 'vi-VN',
name: 'Vietnamese (Vietnam)',
voices: [
{ value: 'Linh - standard', name: 'Linh (standard)', model: 'standard' },
],
},
];

View File

@@ -0,0 +1,16 @@
module.exports = [
{
value: 'en-US',
name: 'English',
voices: [
{
value: 'English-US.Female-1',
name: 'Female',
},
{
value: 'English-US.Male-1',
name: 'Male',
},
],
},
];

View File

@@ -0,0 +1,710 @@
module.exports = [
{
value: 'en-US',
name: 'English (US)',
voices: [
{
value:
's3://mockingbird-prod/abigail_vo_6661b91f-4012-44e3-ad12-589fbdee9948/voices/speaker/manifest.json',
name: 'Abigail - american, female, narrative, smooth',
},
{
value: 'abram',
name: 'Abram - british, old, male, low, narrative, slow, round',
},
{
value: 'adolfo',
name: 'Adolfo - american, adult, male, neutral, narrative, fast, thick',
},
{
value: 'adrian',
name: 'Adrian - american, old, male, neutral, narrative, fast, thick',
},
{
value: 'ahmed',
name: 'Logan - british, old, male, neutral, narrative, neutral, thick',
},
{
value: 'alex',
name: 'Alex - british, adult, male, high, narrative, slow, thick',
},
{
value: 'alexander',
name: 'Alexander - british, old, male, high, narrative, fast, thick',
},
{
value: 'alfonso',
name: 'Alfonso - american, adult, male, neutral, videos, neutral, gravelly',
},
{
value: 'alphonso',
name: 'Alphonso - american, adult, female, low, videos, neutral, smooth',
},
{
value: 'amado',
name: 'Amado - american, old, male, low, narrative, fast, smooth',
},
{
value: 'anny',
name: 'Anny - american, youth, female, neutral, narrative, neutral, thick',
},
{
value: 'anthony',
name: 'Anthony - american, adult, male, neutral, training, slow, thick',
},
{
value: 'spencer',
name: 'April - british, adult, female, neutral, narrative, slow, smooth',
},
{
value: 'victor',
name: 'Ariana - american, youth, female, high, videos, fast, thick',
},
{
value: 'arthur',
name: 'Arthur - british, adult, male, neutral, narrative, neutral, smooth',
},
{
value: 'aubrey',
name: 'Aubrey - british, adult, male, neutral, videos, neutral, smooth',
},
{
value: 'hipolito',
name: 'Audrey - american, adult, female, low, narrative, slow, round',
},
{
value: 'aurora',
name: 'Aurora - british, adult, female, low, training, slow, round',
},
{
value: 'axel',
name: 'Axel - american, adult, male, neutral, narrative, fast, thick',
},
{
value:
's3://mockingbird-prod/ayla_vo_commercials_d66900d5-69f5-476f-9bd6-8eab2936dda3/voices/speaker/manifest.json',
name: 'Ayla (Advertising) - american, female, advertising',
},
{
value:
's3://mockingbird-prod/ayla_vo_expressive_16095e08-b9e8-429b-947c-47a75e41053b/voices/speaker/manifest.json',
name: 'Ayla (Expressive) - american, female, narrative',
},
{
value:
's3://mockingbird-prod/ayla_vo_meditation_d11dd9da-b5f1-4709-95a6-e6d5dc77614a/voices/speaker/manifest.json',
name: 'Ayla (Meditation) - american, female, meditation',
},
{
value:
's3://mockingbird-prod/ayla_vo_narrative_d8199dfd-b50f-40c7-9d99-e203ba5f4152/voices/speaker/manifest.json',
name: 'Ayla (Narrative) - american, female, narrative',
},
{
value:
's3://mockingbird-prod/ayla_vo_training_e6751ca5-e47c-4c4b-ad05-d3a194417600/voices/speaker/manifest.json',
name: 'Ayla (Training) - american, female, training',
},
{
value: 'benton',
name: 'Benton - american, old, male, high, videos, fast, smooth',
},
{
value: 'bertram',
name: 'Bertram - british, adult, male, low, narrative, neutral, gravelly',
},
{
value: 'bill',
name: 'Harper - american, adult, female, high, videos, fast, smooth',
},
{
// eslint-disable-next-line max-len
value:'s3://mockingbird-prod/nathan_drake_carmelo_pampillonio_7d540ad6-7d32-41f6-8d53-2584901aa03d/voices/speaker/manifest.json',
name: 'Billy - american, male, gaming',
},
{
value: 'blaine',
name: 'Blaine - british, adult, male, high, narrative, neutral, thick',
},
{
value: 'booker',
name: 'Booker - british, youth, male, neutral, narrative, neutral, round',
},
{
value: 'bret',
name: 'Bret - american, adult, female, neutral, narrative, slow, smooth',
},
{
value: 'bruce',
name: 'Bruce - british, adult, male, high, training, fast, thick',
},
{
value: 'bryan',
name: 'Bryan - american, adult, male, low, videos, fast, gravelly',
},
{
value: 'carlo',
name: 'Carlo - british, adult, male, neutral, advertising, neutral, smooth',
},
{
value: 'carter',
name: 'Carter - american, adult, male, neutral, narrative, neutral, thick',
},
{
value: 'charles',
name: 'Charles - american, adult, male, neutral, narrative, neutral, round',
},
{
value: 'charlotte',
name: 'Charlotte - canadian, adult, female, low, narrative, neutral, smooth',
},
{
value:
's3://voice-cloning-zero-shot/028a32d4-6a79-4ca3-a303-d6559843114b/chris/manifest.json',
name: 'Chris - american, adult, male,',
},
{
value: 'chuck',
name: 'Chuck - british, adult, male, neutral, videos, slow, round',
},
{
value: 'clark',
name: 'Clark - british, old, male, neutral, narrative, slow, smooth',
},
{
value: 'clifton',
name: 'Clifton - american, old, male, high, narrative, neutral, gravelly',
},
{
value: 'hayden',
name: 'Cooper - american, adult, male, neutral, narrative, neutral, round',
},
{
value: 'daisy',
name: 'Daisy - british, adult, female, low, narrative, neutral, gravelly',
},
{
value: 'dane',
name: 'Dane - american, adult, male, neutral, videos, neutral, round',
},
{
value: 'daniel',
name: 'Daniel - canadian, adult, male, low, narrative, neutral, smooth',
},
{
value: 'darnell',
name: 'Darnell - american, youth, male, neutral, narrative, neutral, smooth',
},
{
value: 'daron',
name: 'Daron - american, old, male, low, narrative, slow, round',
},
{
value: 'darrell',
name: 'Darrell - british, adult, male, neutral, advertising, neutral, thick',
},
{
value: 's3://peregrine-voices/a10/manifest.json',
name: 'Davis - american, adult, male,',
},
{
value: 'ignacio',
name: 'Delilah - american, adult, female, neutral, narrative, slow, smooth',
},
{
value: 'denis',
name: 'Eleanor - british, adult, female, neutral, advertising, neutral, smooth',
},
{
value: 'dick',
name: 'Dick - american, adult, male, neutral, training, fast, smooth',
},
{
value: 'domenic',
name: 'Domenic - british, adult, male, high, videos, neutral, thick',
},
{
value: 's3://peregrine-voices/donna_meditation_saad/manifest.json',
name: 'Donna (Meditation) - american, female, meditation',
},
{
value: 's3://peregrine-voices/donna_parrot_saad/manifest.json',
name: 'Donna (Narrative) - american, female, narrative',
},
{
value: 'donovan',
name: 'Donovan - american, adult, male, low, narrative, neutral, smooth',
},
{
value: 'dudley',
name: 'Dudley - american, old, male, low, narrative, fast, smooth',
},
{
value: 'dylan',
name: 'Dylan - british, old, male, high, gaming, slow, smooth',
},
{
value: 'earle',
name: 'Earle - british, adult, male, high, narrative, neutral, gravelly',
},
{
value: 'efren',
name: 'Efren - american, adult, male, neutral, training, slow, thick',
},
{
value: 'denis',
name: 'Eleanor - british, adult, female, neutral, advertising, neutral, smooth',
},
{
value: 'elijah',
name: 'Elijah - american, old, male, neutral, training, neutral, gravelly',
},
{
value: 'ellie',
name: 'Ellie - american, adult, female, low, training, slow, smooth',
},
{
value: 'erasmo',
name: 'Erasmo - american, old, male, low, training, fast, smooth',
},
{
value: 's3://peregrine-voices/evelyn 2 saad parrot/manifest.json',
name: 'Evelyn - american, adult, female, low, videos, neutral, smooth',
},
{
value: 'fletcher',
name: 'Fletcher - british, adult, male, neutral, narrative, fast, gravelly',
},
{
value: 'florencio',
name: 'Madison - british, old, female, neutral, narrative, slow, round',
},
{
value: 'flynn',
name: 'Flynn - british, adult, male, neutral, narrative, fast, round',
},
{
value: 'gabriel',
name: 'Samantha - american, old, female, neutral, narrative, neutral, thick',
},
{
value: 'greg',
name: 'Greg - british, adult, male, high, narrative, slow, round',
},
{
value: 'harold',
name: 'Harold - american, adult, male, neutral, narrative, slow, smooth',
},
{
value: 'bill',
name: 'Harper - american, adult, female, high, videos, fast, smooth',
},
{
value: 'harris',
name: 'Harris - british, adult, male, low, narrative, fast, smooth',
},
{
value: 'harrison',
name: 'Harrison - american, adult, male, neutral, narrative, fast, round',
},
{
value: 'hayden',
name: 'Cooper - american, adult, male, neutral, narrative, neutral, round',
},
{
value: 'hipolito',
name: 'Audrey - american, adult, female, low, narrative, slow, round',
},
{
value:
's3://mockingbird-prod/hook_1_chico_a3e5e83f-08ae-4a9f-825c-7e48d32d2fd8/voices/speaker/manifest.json',
name: 'Hook - american, male, gaming',
},
{
value: 's3://peregrine-voices/hudson saad parrot/manifest.json',
name: 'Hudson - american, adult, male, neutral, videos, neutral, thick',
},
{
value: 'hunter',
name: 'Hunter - british, old, male, high, narrative, fast, round',
},
{
value: 'ignacio',
name: 'Delilah - american, adult, female, neutral, narrative, slow, smooth',
},
{
value: 's3://peregrine-voices/mel28/manifest.json',
name: 'Jack - american, adult, male,',
},
{
value: 'jarrett',
name: 'Jarrett - american, adult, male, low, advertising, slow, smooth',
},
{
value:
's3://voice-cloning-zero-shot/801a663f-efd0-4254-98d0-5c175514c3e8/jennifer/manifest.json',
name: 'Jennifer - american, adult, female,',
},
{
value: 'jerrell',
name: 'Jerrell - american, adult, male, low, narrative, neutral, round',
},
{
value: 'jordan',
name: 'Jordan - american, adult, male, neutral, training, slow, round',
},
{
value:
's3://voice-cloning-zero-shot/dc23bb38-f568-4323-b6fb-7d64f685b97a/joseph/manifest.json',
name: 'Joseph - american, adult, male,',
},
{
value: 'judson',
name: 'Judson - american, adult, male, low, narrative, slow, smooth',
},
{
value: 'lance',
name: 'Lance - british, adult, male, low, videos, neutral, smooth',
},
{
value: 'larry',
name: 'Larry - american, adult, male, neutral, narrative, neutral, smooth',
},
{
value: 's3://peregrine-voices/larry_ads3_parrot_saad/manifest.json',
name: 'Larry (Advertising) - american, adult, male, neutral, advertising, neutral, smooth',
},
{
value:
's3://mockingbird-prod/larry_vo_narrative_4bd5c1bd-f662-4a38-b5b9-76563f7b92ec/voices/speaker/manifest.json',
name: 'Larry (Narrative) - american, adult, male, neutral, narrative, neutral, smooth',
},
{
value: 'lillian',
name: 'Lillian - british, old, female, neutral, training, slow, round',
},
{
value: 'ahmed',
name: 'Logan - british, old, male, neutral, narrative, neutral, thick',
},
{
value: 'lottie',
name: 'Lottie - british, adult, female, low, narrative, slow, smooth',
},
{
value: 'lucius',
name: 'Lucius - british, adult, male, low, narrative, slow, smooth',
},
{
value: 'mickey',
name: 'Madelyn - british, adult, female, neutral, videos, fast, thick',
},
{
value:
's3://voice-cloning-zero-shot/09b5c0cc-a8f4-4450-aaab-3657b9965d0b/podcaster/manifest.json',
name: 'Matt - american, adult, male,',
},
{
value: 's3://peregrine-voices/mel21/manifest.json',
name: 'Melissa - american, adult, female,',
},
{
value: 'micah',
name: 'Micah - british, adult, female, neutral, narrative, neutral, smooth',
},
{
value:
's3://voice-cloning-zero-shot/7c339a9d-370f-4643-adf5-4134e3ec9886/mlae02/manifest.json',
name: 'Michael - american, adult, male,',
},
{
value: 'mickey',
name: 'Madelyn - british, adult, female, neutral, videos, fast, thick',
},
{
value:
's3://voice-cloning-zero-shot/7c38b588-14e8-42b9-bacd-e03d1d673c3c/nicole/manifest.json',
name: 'Nicole - american, adult, female,',
},
{
value: 's3://peregrine-voices/nolan saad parrot/manifest.json',
name: 'Nolan - british, adult, male, high, videos, neutral, round',
},
{
value: 'nova',
name: 'Nova - american, adult, female, whisper, narrative, slow, smooth',
},
{
value: 'oliver',
name: 'Oliver - british, adult, male, high, videos, neutral, round',
},
{
value: 'oscar',
name: 'Oscar - british, adult, male, neutral, narrative, slow, smooth',
},
{
value: 'owen',
name: 'Owen - american, youth, male, high, narrative, neutral, round',
},
{
value: 'pedro',
name: 'Pedro - american, adult, male, neutral, narrative, slow, round',
},
{
value: 'phoebe',
name: 'Phoebe - british, adult, female, high, videos, fast, smooth',
},
{
value: 'randall',
name: 'Randall - british, adult, male, high, narrative, fast, thick',
},
{
value: 'reynaldo',
name: 'Reynaldo - british, old, male, low, narrative, fast, smooth',
},
{
value: 'rodrick',
name: 'Rodrick - american, adult, male, neutral, narrative, neutral, smooth',
},
{
value: 'gabriel',
name: 'Samantha - american, old, female, neutral, narrative, neutral, thick',
},
{
value: 'samuel',
name: 'Samuel - american, old, male, high, narrative, slow, gravelly',
},
{
value:
// eslint-disable-next-line max-len
's3://mockingbird-prod/agent_47_carmelo_pampillonio_58e796e1-0b87-4f3e-8b36-7def6d65ce66/voices/speaker/manifest.json',
name: 'Sarge - american, male, gaming',
},
{
value:
's3://voice-cloning-zero-shot/1f44b3e7-22ea-4c2e-87d0-b4d9c8f1d47d/sophia/manifest.json',
name: 'Sophia - american, adult, female,',
},
{
value: 'spencer',
name: 'April - british, adult, female, neutral, narrative, slow, smooth',
},
{
value: 'stella',
name: 'Stella - british, old, female, neutral, training, slow, round',
},
{
value: 'susan',
name: 'Susan - american, adult, female, high, videos, neutral, round',
},
{
value:
// eslint-disable-next-line max-len
's3://mockingbird-prod/susan_vo_commercials_0f4fa663-6eba-4582-be1e-2d5bde798f1c/voices/speaker/manifest.json',
name: 'Susan (Advertising) - american, adult, female, high, advertising, neutral, round',
},
{
value:
's3://mockingbird-prod/susan_vo_narrative_73051c90-460b-4e54-adab-9235f45c5e5f/voices/speaker/manifest.json',
name: 'Susan (Narrative) - american, adult, female, high, narrative, neutral, round',
},
{
value:
's3://mockingbird-prod/susan_vo_training_46ffcc60-d630-42f6-acfe-4affd003ae7a/voices/speaker/manifest.json',
name: 'Susan (Training) - american, adult, female, high, training, neutral, round',
},
{
value: 'theodore',
name: 'Theodore - american, old, male, neutral, narrative, neutral, gravelly',
},
{
value: 'victor',
name: 'Ariana - american, youth, female, high, videos, fast, thick',
},
{
value: 'wilbert',
name: 'Wilbert - british, adult, male, neutral, narrative, neutral, round',
},
{
value: 'wilbur',
name: 'Wilbur - american, youth, male, neutral, narrative, neutral, smooth',
},
{
value: 'wilfred',
name: 'Wilfred - american, old, male, low, training, slow, smooth',
},
{
value: 's3://peregrine-voices/mel22/manifest.json',
name: 'Will - american, adult, male,',
},
{
value: 'william',
name: 'William - american, adult, male, neutral, videos, neutral, round',
},
{
value:
// eslint-disable-next-line max-len
's3://mockingbird-prod/william_vo_narrative_0eacdff5-6243-4e26-8b3b-66e03458c1d1/voices/speaker/manifest.json',
name: 'William (Narrative) - american, adult, male, neutral, narrative, neutral, round',
},
{
value:
's3://mockingbird-prod/william_vo_training_1b939b71-14fa-41f0-b1db-7d94f194ad0a/voices/speaker/manifest.json',
name: 'William (Training) - american, adult, male, neutral, training, neutral, round',
},
],
},
{
value: 'en-GB',
name: 'English (GB)',
voices: [
{
value: 's3://peregrine-voices/arthur ads parrot saad/manifest.json',
name: 'Arthur (Advertising) - british, adult, male, neutral, advertising, neutral, smooth',
},
{
value:
// eslint-disable-next-line max-len
's3://mockingbird-prod/arthur_vo_meditatoin_211f702d-b185-4115-b8b4-801f8130a38d/voices/speaker/manifest.json',
name: 'Arthur (Meditation) - british, adult, male, neutral, meditation, neutral, smooth',
},
{
value:
's3://mockingbird-prod/arthur_vo_narrative_a33fd610-73a9-4401-9a78-6b8219c68a9e/voices/speaker/manifest.json',
name: 'Arthur (Narrative) - british, adult, male, neutral, narrative, neutral, smooth',
},
{
value:
's3://mockingbird-prod/arthur_vo_training_9281c8fd-c7f0-4445-a148-466292d3d329/voices/speaker/manifest.json',
name: 'Arthur (Training) - british, adult, male, neutral, training, neutral, smooth',
},
{
value:
's3://mockingbird-prod/eileen_vo_5d7b2bcc-d635-4301-97e8-d97c13768514/voices/speaker/manifest.json',
name: 'Eileen - british, female, narrative',
},
{
value: 'frankie',
name: 'Frankie - british, old, male, neutral, training, neutral, thick',
},
{
value:
's3://voice-cloning-zero-shot/418a94fa-2395-4487-81d8-22daf107781f/george/manifest.json',
name: 'George - british, adult, male,',
},
{
value: 'julian',
name: 'Julian - british, adult, male, neutral, videos, neutral, round',
},
{
value:
's3://voice-cloning-zero-shot/0b5b2e4b-5103-425e-8aa0-510dd35226e2/mark/manifest.json',
name: 'Mark - british, adult, male,',
},
{
value: 's3://peregrine-voices/oliver_ads2_parrot_saad/manifest.json',
name: 'Oliver (Advertising) - british, adult, male, high, advertising, neutral, round',
},
{
value:
's3://peregrine-voices/oliver_narrative2_parrot_saad/manifest.json',
name: 'Oliver (Narrative) - british, adult, male, high, narrative, neutral, round',
},
{
value:
's3://mockingbird-prod/oliver_vo_training_6e3f604a-5605-4542-948d-347b0d7546fc/voices/speaker/manifest.json',
name: 'Oliver (Training) - british, adult, male, high, training, neutral, round',
},
{
value:
's3://voice-cloning-zero-shot/820da3d2-3a3b-42e7-844d-e68db835a206/sarah/manifest.json',
name: 'Sarah - british, adult, female,',
},
],
},
{
value: 'en-AU',
name: 'English (AU)',
voices: [
{
value: 's3://peregrine-voices/barry ads parrot saad/manifest.json',
name: 'Barry (Advertising) - australian, male, advertising',
},
{
value:
's3://peregrine-voices/barry narrative parrot saad/manifest.json',
name: 'Barry (Narrative) - australian, male, narrative',
},
{
value: 'frederick',
name: 'Frederick - australian, adult, male, low, narrative, slow, thick',
},
{
value: 's3://peregrine-voices/russell2_parrot_saad/manifest.json',
name: 'Russell - australian, male,',
},
],
},
{
value: 'en-CA',
name: 'English (CA)',
voices: [
{
value: 's3://peregrine-voices/charlotte ads parrot saad/manifest.json',
name: 'Charlotte (Advertising) - canadian, adult, female, low, advertising, neutral, smooth',
},
{
value:
's3://peregrine-voices/charlotte meditation 2 parrot saad/manifest.json',
name: 'Charlotte (Meditation) - canadian, adult, female, low, meditation, neutral, smooth',
},
{
value:
// eslint-disable-next-line max-len
's3://mockingbird-prod/charlotte_vo_narrative_9290be17-ccea-4700-a7fd-a8fe5c49fb20/voices/speaker/manifest.json',
name: 'Charlotte (Narrative) - canadian, adult, female, low, narrative, neutral, smooth',
},
{
value:
's3://peregrine-voices/charlotte_training_parrot_saad/manifest.json',
name: 'Charlotte (Training) - canadian, adult, female, low, training, neutral, smooth',
},
{
value:
// eslint-disable-next-line max-len
's3://mockingbird-prod/olivia_vo_commercials_6e3c384f-15d6-4fe7-b9a4-0cb1d69daeba/voices/speaker/manifest.json',
name: 'Olivia (Advertising) - canadian, female, advertising',
},
{
value: 's3://peregrine-voices/olivia_ads3_parrot_saad/manifest.json',
name: 'Olivia (Narrative) - canadian, female, narrative',
},
{
value:
's3://mockingbird-prod/olivia_vo_training_4376204f-a411-4e5d-a5c0-ce6cc3908052/voices/speaker/manifest.json',
name: 'Olivia (Training) - canadian, female, training',
},
],
},
{
value: 'en-IE',
name: 'English (IE)',
voices: [
{
value: 'florencio',
name: 'Madison - irish, old, female, neutral, narrative, slow, round',
},
],
},
{
value: 'en-NZ',
name: 'English (NZ)',
voices: [
{
value:
's3://voice-cloning-zero-shot/d9ff78ba-d016-47f6-b0ef-dd630f59414e/female-cs/manifest.json',
name: 'Ruby - australian, adult, female,',
},
],
},
];

View File

@@ -0,0 +1,39 @@
module.exports = [
{
value: 'en-US',
name: 'English (US)',
voices: [
{ value: '3', name: 'Alana B.' },
{ value: '4', name: 'Ramona J.' },
{ value: '5', name: 'Ramona J. (promo)' },
{ value: '7', name: 'Wade C.' },
{ value: '8', name: 'Sofia H.' },
{ value: '9', name: 'David D.' },
{ value: '11', name: 'Isabel V.' },
{ value: '12', name: 'Ava H.' },
{ value: '13', name: 'Jeremy G.' },
{ value: '14', name: 'Nicole L.' },
{ value: '15', name: 'Paige L.' },
{ value: '16', name: 'Tobin A.' },
{ value: '17', name: 'Kai M.' },
{ value: '18', name: 'Tristan F.' },
{ value: '19', name: 'Patrick K.' },
{ value: '20', name: 'Soifia H. (promo)' },
{ value: '21', name: 'Damian P. (promo)' },
{ value: '22', name: 'Jodi P. (promo)' },
{ value: '23', name: 'Lee M. (promo)' },
{ value: '24', name: 'Selene R. (promo)' },
{ value: '26', name: 'Wade C. (promo)' },
{ value: '27', name: 'Joe F.' },
{ value: '28', name: 'Joe F. (promo)' },
{ value: '29', name: 'Garry J. (character)' },
{ value: '33', name: 'Jude D.' },
{ value: '34', name: 'Eric S. (promo)' },
{ value: '35', name: 'Chase J.' },
{ value: '37', name: 'Steve B. (promo)' },
{ value: '38', name: 'Bella B. (promo)' },
{ value: '39', name: 'Tilda C. (promo)' },
{ value: '41', name: 'Paul B. (promo)' },
],
},
];

View File

@@ -0,0 +1,14 @@
module.exports = [
{
value: 'en-US',
name: 'English',
voices: [
{ value: 'alloy', name: 'Alloy' },
{ value: 'echo', name: 'Echo' },
{ value: 'fable', name: 'Fable' },
{ value: 'onyx', name: 'Onyx' },
{ value: 'nova', name: 'Nova' },
{ value: 'shimmer', name: 'Shimmer' },
],
},
];

View File

@@ -8,6 +8,33 @@ const fs = require('fs');
const { AssemblyAI } = require('assemblyai');
const {decrypt, obscureKey} = require('./encrypt-decrypt');
const TtsGoogleLanguagesVoices = require('./speech-data/tts-google');
const TtsAwsLanguagesVoices = require('./speech-data/tts-aws');
const TtsMicrosoftLanguagesVoices = require('./speech-data/tts-microsoft');
const TtsWellsaidLanguagesVoices = require('./speech-data/tts-wellsaid');
const TtsNuanceLanguagesVoices = require('./speech-data/tts-nuance');
const TtsIbmLanguagesVoices = require('./speech-data/tts-ibm');
const TtsNvidiaLanguagesVoices = require('./speech-data/tts-nvidia');
const TtsElevenlabsLanguagesVoices = require('./speech-data/tts-elevenlabs');
const TtsWhisperLanguagesVoices = require('./speech-data/tts-whisper');
const TtsPlayHtLanguagesVoices = require('./speech-data/tts-playht');
const TtsModelDeepgram = require('./speech-data/tts-model-deepgram');
const TtsModelElevenLabs = require('./speech-data/tts-model-elevenlabs');
const TtsModelWhisper = require('./speech-data/tts-model-whisper');
const TtsModelPlayHT = require('./speech-data/tts-model-playht');
const TtsModelRimelabs = require('./speech-data/tts-model-rimelabs');
const SttGoogleLanguagesVoices = require('./speech-data/stt-google');
const SttAwsLanguagesVoices = require('./speech-data/stt-aws');
const SttMicrosoftLanguagesVoices = require('./speech-data/stt-microsoft');
const SttNuanceLanguagesVoices = require('./speech-data/stt-nuance');
const SttDeepgramLanguagesVoices = require('./speech-data/stt-deepgram');
const SttIbmLanguagesVoices = require('./speech-data/stt-ibm');
const SttNvidiaLanguagesVoices = require('./speech-data/stt-nvidia');
const SttCobaltLanguagesVoices = require('./speech-data/stt-cobalt');
const SttSonioxLanguagesVoices = require('./speech-data/stt-soniox');
const SttAssemblyaiLanguagesVoices = require('./speech-data/stt-assemblyai');
const testSonioxStt = async(logger, credentials) => {
const api_key = credentials;
@@ -95,6 +122,13 @@ const testMicrosoftStt = async(logger, credentials) => {
const speechConfig = sdk.SpeechConfig.fromSubscription(api_key, region);
const audioConfig = sdk.AudioConfig.fromWavFileInput(fs.readFileSync(`${__dirname}/../../data/test_audio.wav`));
speechConfig.speechRecognitionLanguage = 'en-US';
if (process.env.JAMBONES_HTTP_PROXY_IP && process.env.JAMBONES_HTTP_PROXY_PORT) {
logger.debug(
`testMicrosoftStt: using proxy ${process.env.JAMBONES_HTTP_PROXY_IP}:${process.env.JAMBONES_HTTP_PROXY_PORT}`);
speechConfig.setProxy(process.env.JAMBONES_HTTP_PROXY_IP, process.env.JAMBONES_HTTP_PROXY_PORT);
}
const speechRecognizer = new sdk.SpeechRecognizer(speechConfig, audioConfig);
return new Promise((resolve, reject) => {
@@ -131,16 +165,26 @@ const testAwsTts = async(logger, getTtsVoices, credentials) => {
}
};
const testAwsStt = async(logger, credentials) => {
const testAwsStt = async(logger, getAwsAuthToken, credentials) => {
try {
const {region, accessKeyId, secretAccessKey} = credentials;
const client = new TranscribeClient({
region,
credentials: {
accessKeyId,
secretAccessKey
}
});
const {region, accessKeyId, secretAccessKey, roleArn} = credentials;
let client = null;
if (accessKeyId && secretAccessKey) {
client = new TranscribeClient({
region,
credentials: {
accessKeyId,
secretAccessKey
}
});
} else if (roleArn) {
client = new TranscribeClient({
region,
credentials: await getAwsAuthToken(null, null, region, roleArn),
});
} else {
client = new TranscribeClient({region});
}
const command = new ListVocabulariesCommand({});
const response = await client.send(command);
return response;
@@ -150,38 +194,19 @@ const testAwsStt = async(logger, credentials) => {
}
};
const testMicrosoftTts = async(logger, credentials) => {
const {
api_key,
region,
// eslint-disable-next-line no-unused-vars
use_custom_tts,
// eslint-disable-next-line no-unused-vars
custom_tts_endpoint,
// eslint-disable-next-line no-unused-vars
use_custom_stt,
// eslint-disable-next-line no-unused-vars
custom_stt_endpoint
} = credentials;
logger.info({
api_key,
region,
use_custom_tts,
custom_tts_endpoint,
use_custom_stt,
custom_stt_endpoint
}, 'testing microsoft tts');
if (!api_key) throw new Error('testMicrosoftTts: credentials are missing api_key');
if (!region) throw new Error('testMicrosoftTts: credentials are missing region');
const testMicrosoftTts = async(logger, synthAudio, credentials) => {
try {
const getJSON = bent('json', {
'Ocp-Apim-Subscription-Key': api_key
});
const response = await getJSON(`https://${region}.tts.speech.microsoft.com/cognitiveservices/voices/list`);
return response;
await synthAudio({increment: () => {}, histogram: () => {}},
{
vendor: 'microsoft',
credentials,
language: 'en-US',
voice: 'en-US-JennyMultilingualNeural',
text: 'Hi there and welcome to jambones!'
}
);
} catch (err) {
logger.info({err}, `testMicrosoftTts - failed to list voices for region ${region}`);
logger.info({err}, 'testMicrosoftTts returned error');
throw err;
}
};
@@ -228,6 +253,50 @@ const testElevenlabs = async(logger, credentials) => {
}
};
const testPlayHT = async(logger, synthAudio, credentials) => {
try {
await synthAudio(
{
increment: () => {},
histogram: () => {}
},
{
vendor: 'playht',
credentials,
language: 'en-US',
voice: 's3://voice-cloning-zero-shot/d9ff78ba-d016-47f6-b0ef-dd630f59414e/female-cs/manifest.json',
text: 'Hi there and welcome to jambones!'
}
);
// Test if playHT can fetch voices
await fetchLayHTVoices(credentials);
} catch (err) {
logger.info({err}, 'synth Playht returned error');
throw err;
}
};
const testRimelabs = async(logger, synthAudio, credentials) => {
try {
await synthAudio(
{
increment: () => {},
histogram: () => {}
},
{
vendor: 'rimelabs',
credentials,
language: 'en-US',
voice: 'amber',
text: 'Hi there and welcome to jambones!'
}
);
} catch (err) {
logger.info({err}, 'synth Playht returned error');
throw err;
}
};
const testWhisper = async(logger, synthAudio, credentials) => {
try {
await synthAudio({increment: () => {}, histogram: () => {}},
@@ -245,6 +314,22 @@ const testWhisper = async(logger, synthAudio, credentials) => {
}
};
const testDeepgramTTS = async(logger, synthAudio, credentials) => {
try {
await synthAudio({increment: () => {}, histogram: () => {}},
{
vendor: 'deepgram',
credentials,
model: 'aura-asteria-en',
text: 'Hi there and welcome to jambones!'
}
);
} catch (err) {
logger.info({err}, 'testDeepgramTTS returned error');
throw err;
}
};
const testIbmTts = async(logger, getTtsVoices, credentials) => {
const {tts_api_key, tts_region} = credentials;
const voices = await getTtsVoices({vendor: 'ibm', credentials: {tts_api_key, tts_region}});
@@ -325,6 +410,7 @@ const getSpeechCredential = (credential, logger) => {
...credential,
accessKeyId: credential.access_key_id,
secretAccessKey: credential.secret_access_key,
roleArn: credential.role_arn,
region: credential.aws_region || 'us-east-1'
};
}
@@ -346,6 +432,7 @@ function decryptCredential(obj, credential, logger, isObscureKey = true) {
else if ('aws' === obj.vendor) {
const o = JSON.parse(decrypt(credential));
obj.access_key_id = o.access_key_id;
obj.role_arn = o.role_arn;
obj.secret_access_key = isObscureKey ? obscureKey(o.secret_access_key) : o.secret_access_key;
obj.aws_region = o.aws_region;
logger.info({obj, o}, 'retrieving aws speech credential');
@@ -370,10 +457,14 @@ function decryptCredential(obj, credential, logger, isObscureKey = true) {
const o = JSON.parse(decrypt(credential));
obj.client_id = o.client_id;
obj.secret = o.secret ? (isObscureKey ? obscureKey(o.secret) : o.secret) : null;
obj.nuance_tts_uri = o.nuance_tts_uri;
obj.nuance_stt_uri = o.nuance_stt_uri;
}
else if ('deepgram' === obj.vendor) {
const o = JSON.parse(decrypt(credential));
obj.api_key = isObscureKey ? obscureKey(o.api_key) : o.api_key;
obj.deepgram_stt_uri = o.deepgram_stt_uri;
obj.deepgram_stt_use_tls = o.deepgram_stt_use_tls;
}
else if ('ibm' === obj.vendor) {
const o = JSON.parse(decrypt(credential));
@@ -395,6 +486,18 @@ function decryptCredential(obj, credential, logger, isObscureKey = true) {
const o = JSON.parse(decrypt(credential));
obj.api_key = isObscureKey ? obscureKey(o.api_key) : o.api_key;
obj.model_id = o.model_id;
obj.options = o.options;
} else if ('playht' === obj.vendor) {
const o = JSON.parse(decrypt(credential));
obj.api_key = isObscureKey ? obscureKey(o.api_key) : o.api_key;
obj.user_id = o.user_id;
obj.voice_engine = o.voice_engine;
obj.options = o.options;
} else if ('rimelabs' === obj.vendor) {
const o = JSON.parse(decrypt(credential));
obj.api_key = isObscureKey ? obscureKey(o.api_key) : o.api_key;
obj.model_id = o.model_id;
obj.options = o.options;
} else if (obj.vendor.startsWith('custom:')) {
const o = JSON.parse(decrypt(credential));
obj.auth_token = isObscureKey ? obscureKey(o.auth_token) : o.auth_token;
@@ -410,6 +513,434 @@ function decryptCredential(obj, credential, logger, isObscureKey = true) {
}
}
/**
*
* @param {*} logger logger
* @param {*} vendor vendor
* @param {*} credential STT/TTS vendor credential, can be null
* @returns List of language and coresponding voices for specific vendor follow below format
* {
"tts": [
{
code: "ar-XA",
name: "Arabic",
voices: [
{ value: "ar-XA-Standard-A", name: "Standard-A (Female)" },
]
}
],
"stt": [
{ name: "Afrikaans (South Africa)", code: "af-ZA" },
]
}
*/
async function getLanguagesAndVoicesForVendor(logger, vendor, credential, getTtsVoices) {
switch (vendor) {
case 'google':
return await getLanguagesVoicesForGoogle(credential, getTtsVoices, logger);
case 'aws':
return await getLanguagesVoicesForAws(credential, getTtsVoices, logger);
case 'microsoft':
return await getLanguagesVoicesForMicrosoft(credential, getTtsVoices, logger);
case 'wellsaid':
return await getLanguagesVoicesForWellsaid(credential, getTtsVoices, logger);
case 'nuance':
return await getLanguagesVoicesForNuane(credential, getTtsVoices, logger);
case 'deepgram':
return await getLanguagesVoicesForDeepgram(credential, getTtsVoices, logger);
case 'ibm':
return await getLanguagesVoicesForIbm(credential, getTtsVoices, logger);
case 'nvidia':
return await getLanguagesVoicesForNvida(credential, getTtsVoices, logger);
case 'cobalt':
return await getLanguagesVoicesForCobalt(credential, getTtsVoices, logger);
case 'soniox':
return await getLanguagesVoicesForSoniox(credential, getTtsVoices, logger);
case 'elevenlabs':
return await getLanguagesVoicesForElevenlabs(credential, getTtsVoices, logger);
case 'playht':
return await getLanguagesVoicesForPlayHT(credential, getTtsVoices, logger);
case 'rimelabs':
return await getLanguagesVoicesForRimelabs(credential, getTtsVoices, logger);
case 'assemblyai':
return await getLanguagesVoicesForAssemblyAI(credential, getTtsVoices, logger);
case 'whisper':
return await getLanguagesVoicesForWhisper(credential, getTtsVoices, logger);
default:
logger.info(`invalid vendor ${vendor}, return empty result`);
throw new Error(`Invalid vendor ${vendor}`);
}
}
async function getLanguagesVoicesForGoogle(credential, getTtsVoices, logger) {
if (credential) {
try {
const [result] = await getTtsVoices({
vendor: 'google',
credentials: credential
});
const tts = parseGooglelanguagesVoices(result.voices);
return tranform(tts, SttGoogleLanguagesVoices);
} catch (err) {
logger.info('Error while fetching google languages, voices, return predefined values', err);
}
}
return tranform(TtsGoogleLanguagesVoices, SttGoogleLanguagesVoices);
}
async function getLanguagesVoicesForAws(credential, getTtsVoices, logger) {
if (credential) {
try {
const result = await getTtsVoices({
vendor: 'aws',
credentials: {
accessKeyId: credential.access_key_id,
secretAccessKey: credential.secret_access_key,
roleArn: credential.role_arn,
region: credential.aws_region || process.env.AWS_REGION
}
});
const tts = parseAwsLanguagesVoices(result.Voices);
return tranform(tts, SttAwsLanguagesVoices);
} catch (err) {
logger.info('Error while fetching AWS languages, voices, return predefined values', err);
}
}
return tranform(TtsAwsLanguagesVoices, SttAwsLanguagesVoices);
}
async function getLanguagesVoicesForMicrosoft(credential, getTtsVoices, logger) {
if (credential) {
try {
const get = bent('https://westus.tts.speech.microsoft.com', 'GET', 'json', {
'Ocp-Apim-Subscription-Key' : credential.api_key
});
const voices = await get('/cognitiveservices/voices/list');
const tts = parseMicrosoftLanguagesVoices(voices);
return tranform(tts, SttMicrosoftLanguagesVoices);
} catch (err) {
logger.info('Error while fetching Microsoft languages, voices, return predefined values', err);
}
}
return tranform(TtsMicrosoftLanguagesVoices, SttMicrosoftLanguagesVoices);
}
async function getLanguagesVoicesForWellsaid(credential) {
return tranform(TtsWellsaidLanguagesVoices);
}
async function getLanguagesVoicesForNuane(credential, getTtsVoices, logger) {
if (credential) {
try {
const result = await getTtsVoices({
vendor: 'nuance',
credentials: credential
});
const tts = parseNuanceLanguagesVoices(result.result.voices);
return tranform(tts, SttNuanceLanguagesVoices);
} catch (err) {
logger.info('Error while fetching IBM languages, voices, return predefined values', err);
}
}
return tranform(TtsNuanceLanguagesVoices, SttNuanceLanguagesVoices);
}
async function getLanguagesVoicesForDeepgram(credential) {
return tranform(undefined, SttDeepgramLanguagesVoices, TtsModelDeepgram);
}
async function getLanguagesVoicesForIbm(credential, getTtsVoices, logger) {
if (credential) {
try {
const result = await getTtsVoices({
vendor: 'ibm',
credentials: credential
});
const tts = parseIBMLanguagesVoices(result.result.voices);
return tranform(tts, SttIbmLanguagesVoices);
} catch (err) {
logger.info('Error while fetching IBM languages, voices, return predefined values', err);
}
}
return tranform(TtsIbmLanguagesVoices, SttIbmLanguagesVoices);
}
async function getLanguagesVoicesForNvida(credential) {
return tranform(TtsNvidiaLanguagesVoices, SttNvidiaLanguagesVoices);
}
async function getLanguagesVoicesForCobalt(credential) {
return tranform(undefined, SttCobaltLanguagesVoices);
}
async function getLanguagesVoicesForSoniox(credential) {
return tranform(undefined, SttSonioxLanguagesVoices);
}
async function getLanguagesVoicesForElevenlabs(credential) {
if (credential) {
const get = bent('https://api.elevenlabs.io', 'GET', 'json', {
'xi-api-key' : credential.api_key
});
const [langResp, voiceResp] = await Promise.all([get('/v1/models'), get('/v1/voices')]);
const model = langResp.find((m) => m.model_id === credential.model_id);
const languages = model ? model.languages.map((l) => {
return {
value: l.language_id,
name: l.name
};
}).sort((a, b) => a.name.localeCompare(b.name)) : [];
if (languages && languages.length > 0) {
const voices = voiceResp ? voiceResp.voices.map((v) => {
let name = `${v.name}${v.category !== 'premade' ? ` (${v.category})` : ''} -
${v.labels.accent ? ` ${v.labels.accent},` : ''}
${v.labels.description ? ` ${v.labels.description},` : ''}
${v.labels.age ? ` ${v.labels.age},` : ''}
${v.labels.gender ? ` ${v.labels.gender},` : ''}
${v.labels['use case'] ? ` ${v.labels['use case']},` : ''}
`;
const lastIndex = name.lastIndexOf(',');
if (lastIndex !== -1) {
name = name.substring(0, lastIndex);
}
return {
value: v.voice_id,
name
};
}).sort((a, b) => a.name.localeCompare(b.name)) : [];
languages[0].voices = voices;
}
return tranform(languages, undefined, TtsModelElevenLabs);
} else {
return tranform(TtsElevenlabsLanguagesVoices, undefined, TtsModelElevenLabs);
}
}
const concat = (a) => {
return a ? ` ${a},` : '';
};
const fetchLayHTVoices = async(credential) => {
if (credential) {
const get = bent('https://api.play.ht', 'GET', 'json', {
'AUTHORIZATION' : credential.api_key,
'X-USER-ID': credential.user_id,
'Accept': 'application/json'
});
const voices = await get('/api/v2/voices');
let clone_voices = [];
try {
// try if the account has permission to cloned voice
//otherwise ignore this.
clone_voices = await get('/api/v2/cloned-voices');
} catch {}
return [clone_voices, voices];
}
};
async function getLanguagesVoicesForPlayHT(credential) {
if (credential) {
const [cloned_voice, voices] = await fetchLayHTVoices(credential);
const list_voices = [...cloned_voice, ...voices];
const buildVoice = (d) => {
let name = `${d.name} -${concat(d.accent)}${concat(d.age)}${concat(d.gender)}${concat(d.loudness)}` +
`${concat(d.style)}${concat(d.tempo)}${concat(d.texture)}` ;
name = name.endsWith(',') ? name.trim().slice(0, -1) : name;
return {
value: `${d.id}`,
name
};
};
const ttsVoices = list_voices.reduce((acc, voice) => {
if (!credential.voice_engine.includes(voice.voice_engine)) {
return acc;
}
const languageCode = voice.language_code;
// custom voice does not have language code
if (!languageCode) {
voice.language_code = 'en';
voice.language = 'Custom-English';
}
const existingLanguage = acc.find((lang) => lang.value === languageCode);
if (existingLanguage) {
existingLanguage.voices.push(buildVoice(voice));
} else {
acc.push({
value: voice.language_code,
name: voice.language,
voices: [buildVoice(voice)]
});
}
return acc;
}, []);
return tranform(ttsVoices, undefined, TtsModelPlayHT);
}
return tranform(TtsPlayHtLanguagesVoices, undefined, TtsModelPlayHT);
}
async function getLanguagesVoicesForRimelabs(credential) {
const model_id = credential ? credential.model_id : null;
const get = bent('https://users.rime.ai', 'GET', 'json', {
'Accept': 'application/json'
});
const voices = await get('/data/voices/all.json');
let selectedVoices = model_id ? voices[model_id] : Object.values(voices).reduce((acc, val) => [...acc, ...val], []);
selectedVoices = selectedVoices.map((v) => ({
name: v.charAt(0).toUpperCase() + v.slice(1),
value: v
}));
const ttsVoices = [
{
value: 'en-US',
name: 'English (US)',
voices: selectedVoices
}
];
return tranform(ttsVoices, undefined, TtsModelRimelabs);
}
async function getLanguagesVoicesForAssemblyAI(credential) {
return tranform(undefined, SttAssemblyaiLanguagesVoices);
}
async function getLanguagesVoicesForWhisper(credential) {
return tranform(TtsWhisperLanguagesVoices, undefined, TtsModelWhisper);
}
function tranform(tts, stt, models) {
return {
...(tts && {tts}),
...(stt && {stt}),
...(models && {models})
};
}
function parseGooglelanguagesVoices(data) {
return data.reduce((acc, voice) => {
const languageCode = voice.languageCodes[0];
const existingLanguage = acc.find((lang) => lang.value === languageCode);
if (existingLanguage) {
existingLanguage.voices.push({
value: voice.name,
name: `${voice.name.substring(languageCode.length + 1, voice.name.length)} (${voice.ssmlGender})`
});
} else {
acc.push({
value: languageCode,
name: SttGoogleLanguagesVoices.find((lang) => lang.value === languageCode)?.name || languageCode,
voices: [{
value: voice.name,
name: `${voice.name.substring(languageCode.length + 1, voice.name.length)} (${voice.ssmlGender})`
}]
});
}
return acc;
}, []);
}
function parseIBMLanguagesVoices(data) {
return data.reduce((acc, voice) => {
const languageCode = voice.language;
const existingLanguage = acc.find((lang) => lang.value === languageCode);
if (existingLanguage) {
existingLanguage.voices.push({
value: voice.name,
name: `(${voice.gender}) ${voice.description}`
});
} else {
acc.push({
value: languageCode,
name: SttGoogleLanguagesVoices.find((lang) => lang.value === languageCode)?.name || languageCode,
voices: [{
value: voice.name,
name: `(${voice.gender}) ${voice.description}`
}]
});
}
return acc;
}, []);
}
function parseAwsLanguagesVoices(data) {
return data.reduce((acc, voice) => {
const languageCode = voice.LanguageCode;
const existingLanguage = acc.find((lang) => lang.value === languageCode);
if (existingLanguage) {
existingLanguage.voices.push({
value: voice.Id,
name: `(${voice.Gender}) ${voice.Name}`
});
} else {
acc.push({
value: languageCode,
name: voice.LanguageName,
voices: [{
value: voice.Id,
name: `(${voice.Gender}) ${voice.Name}`
}]
});
}
return acc;
}, []);
}
function parseNuanceLanguagesVoices(data) {
return data.reduce((acc, voice) => {
const languageCode = voice.language;
const existingLanguage = acc.find((lang) => lang.value === languageCode);
if (existingLanguage) {
existingLanguage.voices.push({
value: voice.name,
name: voice.name,
model: voice.model
});
} else {
acc.push({
value: languageCode,
name: SttGoogleLanguagesVoices.find((lang) => lang.value === languageCode)?.name || languageCode,
voices: [{
value: voice.name,
name: voice.name,
model: voice.model
}]
});
}
return acc;
}, []);
}
function parseMicrosoftLanguagesVoices(data) {
return data.reduce((acc, voice) => {
const languageCode = voice.Locale;
const existingLanguage = acc.find((lang) => lang.value === languageCode);
if (existingLanguage) {
existingLanguage.voices.push({
value: voice.ShortName,
name: `${voice.DisplayName} (${voice.Gender})`,
});
} else {
acc.push({
value: voice.Locale,
name: voice.LocaleName,
voices: [{
value: voice.ShortName,
name: `${voice.DisplayName} (${voice.Gender})`,
}]
});
}
return acc;
}, []);
}
module.exports = {
testGoogleTts,
testGoogleStt,
@@ -426,8 +957,12 @@ module.exports = {
testIbmStt,
testSonioxStt,
testElevenlabs,
testPlayHT,
testRimelabs,
testAssemblyStt,
testDeepgramTTS,
getSpeechCredential,
decryptCredential,
testWhisper
testWhisper,
getLanguagesAndVoicesForVendor
};

13601
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,6 +1,6 @@
{
"name": "jambonz-api-server",
"version": "0.8.5",
"version": "0.9.0",
"description": "",
"main": "app.js",
"scripts": {
@@ -19,53 +19,54 @@
"url": "https://github.com/jambonz/jambonz-api-server.git"
},
"dependencies": {
"@aws-sdk/client-s3": "^3.363.0",
"@aws-sdk/client-transcribe": "^3.363.0",
"@azure/storage-blob": "^12.15.0",
"@aws-sdk/client-s3": "^3.550.0",
"@aws-sdk/client-transcribe": "^3.549.0",
"@azure/storage-blob": "^12.17.0",
"@deepgram/sdk": "^1.21.0",
"@google-cloud/speech": "^5.2.0",
"@google-cloud/storage": "^6.12.0",
"@jambonz/db-helpers": "^0.9.1",
"@google-cloud/speech": "^6.5.0",
"@google-cloud/storage": "^7.9.0",
"@jambonz/db-helpers": "^0.9.3",
"@jambonz/lamejs": "^1.2.2",
"@jambonz/mw-registrar": "^0.2.5",
"@jambonz/realtimedb-helpers": "^0.8.7",
"@jambonz/speech-utils": "^0.0.25",
"@jambonz/mw-registrar": "^0.2.7",
"@jambonz/realtimedb-helpers": "^0.8.8",
"@jambonz/speech-utils": "^0.1.0",
"@jambonz/time-series": "^0.2.8",
"@jambonz/verb-specifications": "^0.0.45",
"@soniox/soniox-node": "^1.1.1",
"argon2": "^0.30.3",
"assemblyai": "^3.0.1",
"@jambonz/verb-specifications": "^0.0.69",
"@soniox/soniox-node": "^1.2.2",
"argon2": "^0.40.1",
"assemblyai": "^4.3.4",
"bent": "^7.3.12",
"cors": "^2.8.5",
"debug": "^4.3.4",
"express": "^4.18.1",
"express-rate-limit": "^6.4.0",
"form-data": "^2.5.1",
"helmet": "^5.1.0",
"ibm-watson": "^7.1.2",
"jsonwebtoken": "^9.0.0",
"mailgun.js": "^9.1.2",
"microsoft-cognitiveservices-speech-sdk": "1.31.0",
"mysql2": "^2.3.3",
"nocache": "3.0.4",
"passport": "^0.6.0",
"express": "^4.19.2",
"express-rate-limit": "^7.2.0",
"form-data": "^4.0.0",
"helmet": "^7.1.0",
"ibm-watson": "^9.0.1",
"jsonwebtoken": "^9.0.2",
"mailgun.js": "^10.2.1",
"microsoft-cognitiveservices-speech-sdk": "1.36.0",
"mysql2": "^3.9.3",
"nocache": "4.0.0",
"passport": "^0.7.0",
"passport-http-bearer": "^1.0.1",
"pino": "^5.17.0",
"short-uuid": "^4.1.0",
"stripe": "^8.222.0",
"swagger-ui-express": "^4.4.0",
"uuid": "^8.3.2",
"pino": "^8.20.0",
"short-uuid": "^4.2.2",
"stream-buffers": "^3.0.2",
"stripe": "^14.24.0",
"swagger-ui-express": "^5.0.0",
"uuid": "^9.0.1",
"wav": "^1.0.2",
"ws": "^8.12.1",
"ws": "^8.16.0",
"yamljs": "^0.3.0"
},
"devDependencies": {
"eslint": "^8.39.0",
"eslint-plugin-promise": "^6.1.1",
"husky": "7.0.4",
"husky": "9.0.11",
"nyc": "^15.1.0",
"request": "^2.88.2",
"request-promise-native": "^1.0.9",
"tape": "^5.5.3"
"tape": "^5.7.5"
}
}

View File

@@ -85,6 +85,24 @@ test('client test', async(t) => {
t.ok(result.length === 1 && result[0] === 'dhorton@drachtio.org',
'successfully queried all registered clients');
result = await request.post(`/Accounts/${account_sid}/RegisteredSipUsers`, {
resolveWithFullResponse: true,
auth: authAdmin,
json: true,
body: ['dhorton']
});
t.ok(result.body.length === 1 && result.body[0].name === 'dhorton',
'successfully queried all registered clients');
result = await request.post(`/Accounts/${account_sid}/RegisteredSipUsers`, {
resolveWithFullResponse: true,
auth: authAdmin,
json: true,
body: []
});
t.ok(result.body.length === 1 && result.body[0].name === 'dhorton',
'successfully queried all registered clients');
/* query all entity */
result = await request.get('/Clients', {
auth: authAdmin,

View File

@@ -10,7 +10,7 @@ networks:
services:
mysql:
platform: linux/x86_64
# platform: linux/x86_64
image: mysql:5.7
ports:
- "3360:3306"
@@ -36,7 +36,7 @@ services:
ipv4_address: 172.58.0.3
influxdb:
platform: linux/x86_64
# platform: linux/x86_64
image: influxdb:1.8
ports:
- "8086:8086"

View File

@@ -170,6 +170,20 @@ test('speech credentials tests', async(t) => {
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200 && result.body.tts.status === 'ok', 'successfully tested speech credential for google tts');
t.ok(result.statusCode === 200 && result.body.stt.status === 'ok', 'successfully tested speech credential for google stt');
result = await request.post(`/Accounts/${account_sid}/TtsCache/Synthesize`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
speech_credential_sid: ms_sid,
text: "Hello How are you",
language: "en-US",
voice: "en-US-Standard-C"
}
});
t.ok(result.statusCode === 200, 'successfully google tested synthesize');
}
/* add / test a credential for microsoft */
@@ -198,6 +212,20 @@ test('speech credentials tests', async(t) => {
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200 && result.body.tts.status === 'ok', 'successfully tested speech credential for microsoft tts');
t.ok(result.statusCode === 200 && result.body.stt.status === 'ok', 'successfully tested speech credential for microsoft stt');
result = await request.post(`/Accounts/${account_sid}/TtsCache/Synthesize`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
speech_credential_sid: ms_sid,
text: "Hello How are you",
language: "en-US",
voice: "en-US-AvaMultilingualNeural"
}
});
t.ok(result.statusCode === 200, 'successfully microsoft tested synthesize');
}
/* add / test a credential for AWS */
@@ -227,6 +255,20 @@ test('speech credentials tests', async(t) => {
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200 && result.body.tts.status === 'ok', 'successfully tested speech credential for AWS tts');
t.ok(result.statusCode === 200 && result.body.stt.status === 'ok', 'successfully tested speech credential for AWS stt');
result = await request.post(`/Accounts/${account_sid}/TtsCache/Synthesize`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
speech_credential_sid: ms_sid,
text: "Hello How are you",
language: "en-US",
voice: "Joanna"
}
});
t.ok(result.statusCode === 200, 'successfully AWS tested synthesize');
}
/* add a credential for wellsaid */
@@ -253,6 +295,20 @@ test('speech credentials tests', async(t) => {
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200 && result.body.tts.status === 'ok', 'successfully tested speech credential for wellsaid');
result = await request.post(`/Accounts/${account_sid}/TtsCache/Synthesize`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
speech_credential_sid: ms_sid,
text: "Hello How are you",
language: "en-US",
voice: "3"
}
});
t.ok(result.statusCode === 200, 'successfully Wellsaid tested synthesize');
/* delete the credential */
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${ms_sid}`, {
auth: authUser,
@@ -285,6 +341,20 @@ test('speech credentials tests', async(t) => {
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200 && result.body.stt.status === 'ok', 'successfully tested speech credential for deepgram');
result = await request.post(`/Accounts/${account_sid}/TtsCache/Synthesize`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
speech_credential_sid: ms_sid,
text: "Hello How are you",
language: "en-US",
voice: "aura-asteria-en"
}
});
t.ok(result.statusCode === 200, 'successfully deepgram tested synthesize');
/* delete the credential */
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${ms_sid}`, {
auth: authUser,
@@ -292,6 +362,60 @@ test('speech credentials tests', async(t) => {
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential');
}
// test create deepgram onprem
result = await request.post(`/Accounts/${account_sid}/SpeechCredentials`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
vendor: 'deepgram',
use_for_stt: true,
deepgram_stt_uri: "127.0.0.1:50002",
deepgram_stt_use_tls: true
}
});
t.ok(result.statusCode === 201, 'successfully added speech credential for deepgram');
const dg_sid = result.body.sid;
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/${dg_sid}`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
});
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200, 'successfully get speech credential for deepgram');
t.ok(result.body.deepgram_stt_uri === '127.0.0.1:50002', "deepgram_stt_uri is correct for deepgram");
t.ok(result.body.deepgram_stt_use_tls === true, "deepgram_stt_use_tls is correct for deepgram");
result = await request.put(`/Accounts/${account_sid}/SpeechCredentials/${dg_sid}`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
vendor: 'deepgram',
use_for_stt: true,
deepgram_stt_uri: "127.0.0.2:50002",
deepgram_stt_use_tls: false
}
});
t.ok(result.statusCode === 204, 'successfully updated speech credential for deepgram onprem');
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/${dg_sid}`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
});
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200, 'successfully get speech credential for deepgram onprem');
t.ok(result.body.deepgram_stt_uri === '127.0.0.2:50002', "deepgram_stt_uri is correct for deepgram onprem");
t.ok(result.body.deepgram_stt_use_tls === false, "deepgram_stt_use_tls is correct for deepgram onprem");
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${dg_sid}`, {
auth: authUser,
resolveWithFullResponse: true,
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential for deepgram onprem');
/* add a credential for ibm tts */
if (process.env.IBM_TTS_API_KEY && process.env.IBM_TTS_REGION) {
result = await request.post(`/Accounts/${account_sid}/SpeechCredentials`, {
@@ -350,6 +474,20 @@ test('speech credentials tests', async(t) => {
//console.log(JSON.stringify(result));
t.ok(result.statusCode === 200 && result.body.stt.status === 'ok', 'successfully tested speech credential for ibm stt');
result = await request.post(`/Accounts/${account_sid}/TtsCache/Synthesize`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
speech_credential_sid: ms_sid,
text: "Hello How are you",
language: "en-US",
voice: "en-US_MichaelExpressive"
}
});
t.ok(result.statusCode === 200, 'successfully IBM tested synthesize');
/* delete the credential */
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${ms_sid}`, {
auth: authUser,
@@ -482,7 +620,7 @@ test('speech credentials tests', async(t) => {
model_id: 'eleven_multilingual_v2'
}
});
t.ok(result.statusCode === 201, 'successfully added speech credential for Cobalt');
t.ok(result.statusCode === 201, 'successfully added speech credential for elevenlabs');
const elevenlabs_sid = result.body.sid;
/* delete the credential */
@@ -490,7 +628,54 @@ test('speech credentials tests', async(t) => {
auth: authUser,
resolveWithFullResponse: true,
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential for Cobalt');
t.ok(result.statusCode === 204, 'successfully deleted speech credential for elevenlabs');
/* add a credential for playht */
result = await request.post(`/Accounts/${account_sid}/SpeechCredentials`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
vendor: 'playht',
use_for_stt: false,
use_for_tts: true,
api_key: 'asdasdasdasddsadasda',
user_id: 'user_id',
voice_engine: 'PlayHT2.0-turbo'
}
});
t.ok(result.statusCode === 201, 'successfully added speech credential for playht');
const playht_sid = result.body.sid;
/* delete the credential */
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${playht_sid}`, {
auth: authUser,
resolveWithFullResponse: true,
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential for playht');
/* add a credential for rimelabs */
result = await request.post(`/Accounts/${account_sid}/SpeechCredentials`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
vendor: 'rimelabs',
use_for_stt: false,
use_for_tts: true,
api_key: 'asdasdasdasddsadasda',
model_id: 'mist',
}
});
t.ok(result.statusCode === 201, 'successfully added speech credential for rimelabs');
const rimelabs_sid = result.body.sid;
/* delete the credential */
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${rimelabs_sid}`, {
auth: authUser,
resolveWithFullResponse: true,
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential for rimelabs');
/* add a credential for custom voices google */
@@ -558,6 +743,156 @@ test('speech credentials tests', async(t) => {
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential');
/* add a credential for aws polly by roleArn */
result = await request.post(`/Accounts/${account_sid}/SpeechCredentials`, {
resolveWithFullResponse: true,
auth: authUser,
json: true,
body: {
vendor: 'aws',
labe: 'aws_polly_with_arn',
use_for_tts: true,
use_for_stt: false,
role_arn: 'Arn::aws::role',
aws_region: 'us-east-1'
}
});
t.ok(result.statusCode === 201, 'successfully added speech credential for AWS Polly By RoleArn');
const awsPollySid = result.body.sid;
/* delete the credential */
result = await request.delete(`/Accounts/${account_sid}/SpeechCredentials/${awsPollySid}`, {
auth: authUser,
resolveWithFullResponse: true,
});
t.ok(result.statusCode === 204, 'successfully deleted speech credential');
/* Check google supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=google`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get google supported languages and voices');
t.ok(result.body.stt.length !== 0, 'successfully get google supported languages and voices');
/* Check aws supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=aws`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get aws supported languages and voices');
t.ok(result.body.stt.length !== 0, 'successfully get aws supported languages and voices');
/* Check microsoft supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=microsoft`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get microsoft supported languages and voices');
t.ok(result.body.stt.length !== 0, 'successfully get microsoft supported languages and voices');
/* Check wellsaid supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=wellsaid`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get wellsaid supported languages and voices');
/* Check nuance supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=nuance`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get nuance supported languages and voices');
t.ok(result.body.stt.length !== 0, 'successfully get nuance supported languages and voices');
/* Check deepgram supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=deepgram`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.stt.length !== 0, 'successfully get deepgram supported languages and voices');
t.ok(result.body.models.length !== 0, 'successfully get deepgram supported languages and voices');
/* Check ibm supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=ibm`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get ibm supported languages and voices');
t.ok(result.body.stt.length !== 0, 'successfully get ibm supported languages and voices');
/* Check ibm supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=nvidia`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get nvidia supported languages and voices');
t.ok(result.body.stt.length !== 0, 'successfully get nvidia supported languages and voices');
/* Check cobalt supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=cobalt`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.stt.length !== 0, 'successfully get cobalt supported languages and voices');
/* Check soniox supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=soniox`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.stt.length !== 0, 'successfully get soniox supported languages and voices');
/* Check elevenlabs supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=elevenlabs`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get elevenlabs supported languages and voices');
t.ok(result.body.models.length !== 0, 'successfully get elevenlabs supported languages and voices');
/* Check assemblyai supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=assemblyai`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.stt.length !== 0, 'successfully get assemblyai supported languages and voices');
/* Check whisper supportedLanguagesAndVoices */
result = await request.get(`/Accounts/${account_sid}/SpeechCredentials/speech/supportedLanguagesAndVoices?vendor=whisper`, {
resolveWithFullResponse: true,
simple: false,
auth: authAdmin,
json: true,
});
t.ok(result.body.tts.length !== 0, 'successfully get whisper supported languages and voices');
t.ok(result.body.models.length !== 0, 'successfully get whisper supported languages and voices');
await deleteObjectBySid(request, '/Accounts', account_sid);
await deleteObjectBySid(request, '/ServiceProviders', service_provider_sid);
t.end();