index.js 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. 'use strict';
  2. var child_process = require('child_process');
  3. var path = require('path');
  4. var say = exports;
  5. var childD;
  6. // use the correct library per platform
  7. if (process.platform === 'darwin') {
  8. say.speaker = 'say';
  9. say.base_speed = 175;
  10. } else if (process.platform === 'linux') {
  11. say.speaker = 'festival';
  12. say.base_speed = 100;
  13. } else if (process.platform === 'win32') {
  14. say.speaker = 'cmd';
  15. }
  16. /**
  17. * Uses system libraries to speak text via the speakers.
  18. *
  19. * @param {string} text Text to be spoken
  20. * @param {string|null} voice Name of voice to be spoken with
  21. * @param {number|null} speed Speed of text (e.g. 1.0 for normal, 0.5 half, 2.0 double)
  22. * @param {Function|null} callback A callback of type function(err) to return.
  23. */
  24. say.speak = function(text, voice, speed, callback) {
  25. var commands, pipedData;
  26. if (typeof callback !== 'function') {
  27. callback = function() {};
  28. }
  29. if (!text) {
  30. // throw TypeError because API was used incorrectly
  31. throw new TypeError('Must provide text parameter');
  32. }
  33. // tailor command arguments to specific platforms
  34. if (process.platform === 'darwin') {
  35. if (!voice) {
  36. commands = [ text ];
  37. } else {
  38. commands = [ '-v', voice, text];
  39. }
  40. if (speed) {
  41. commands.push('-r', convertSpeed(speed));
  42. }
  43. } else if (process.platform === 'linux') {
  44. commands = ['--pipe'];
  45. if (speed) {
  46. pipedData = '(Parameter.set \'Audio_Command "aplay -q -c 1 -t raw -f s16 -r $(($SR*' + convertSpeed(speed) + '/100)) $FILE") ';
  47. }
  48. if (voice) {
  49. pipedData += '(' + voice + ') ';
  50. }
  51. pipedData += '(SayText \"' + text + '\")';
  52. } else if (process.platform === 'win32') {
  53. commands = [ '/s /c "' + path.join(__dirname, 'say.vbs') + ' ' + JSON.stringify(text) + '"' ];
  54. } else {
  55. // if we don't support the platform, callback with an error (next tick) - don't continue
  56. return process.nextTick(function() {
  57. callback(new Error('say.js speak does not support platform ' + process.platform));
  58. });
  59. }
  60. var options = (process.platform === 'win32') ? { windowsVerbatimArguments: true } : undefined;
  61. childD = child_process.spawn(say.speaker, commands, options);
  62. childD.stdin.setEncoding('ascii');
  63. childD.stderr.setEncoding('ascii');
  64. if (pipedData) {
  65. childD.stdin.end(pipedData);
  66. }
  67. childD.stderr.once('data', function(data) {
  68. // we can't stop execution from this function
  69. callback(new Error(data));
  70. });
  71. childD.addListener('exit', function (code, signal) {
  72. if (code === null || signal !== null) {
  73. return callback(new Error('say.js: could not talk, had an error [code: ' + code + '] [signal: ' + signal + ']'));
  74. }
  75. childD = null;
  76. callback(null);
  77. });
  78. };
  79. say.export = function(text, voice, speed, filename, callback) {
  80. var commands, pipedData;
  81. if (!text) {
  82. // throw TypeError because API was used incorrectly
  83. throw new TypeError('Must provide text parameter');
  84. }
  85. if (!filename) {
  86. // throw TypeError because API was used incorrectly
  87. throw new TypeError('Must provide a filename');
  88. }
  89. if (typeof callback !== 'function') {
  90. callback = function() {};
  91. }
  92. // tailor command arguments to specific platforms
  93. if (process.platform === 'darwin') {
  94. if (!voice) {
  95. commands = [ text ];
  96. } else {
  97. commands = [ '-v', voice, text];
  98. }
  99. if (speed) {
  100. commands.push('-r', convertSpeed(speed));
  101. }
  102. if (filename){
  103. commands.push('-o', filename, '--data-format=LEF32@32000');
  104. }
  105. } else {
  106. // if we don't support the platform, callback with an error (next tick) - don't continue
  107. return process.nextTick(function() {
  108. callback(new Error('say.js export does not support platform ' + process.platform));
  109. });
  110. }
  111. childD = child_process.spawn(say.speaker, commands);
  112. childD.stdin.setEncoding('ascii');
  113. childD.stderr.setEncoding('ascii');
  114. if (pipedData) {
  115. childD.stdin.end(pipedData);
  116. }
  117. childD.stderr.once('data', function(data) {
  118. // we can't stop execution from this function
  119. callback(new Error(data));
  120. });
  121. childD.addListener('exit', function (code, signal) {
  122. if (code === null || signal !== null) {
  123. return callback(new Error('say.js: could not talk, had an error [code: ' + code + '] [signal: ' + signal + ']'));
  124. }
  125. childD = null;
  126. callback(null);
  127. });
  128. };
  129. /**
  130. * Stops currently playing audio. There will be unexpected results if multiple audios are being played at once
  131. *
  132. * TODO: If two messages are being spoken simultaneously, childD points to new instance, no way to kill previous
  133. */
  134. exports.stop = function(callback) {
  135. if (typeof callback !== 'function') {
  136. callback = function() {};
  137. }
  138. if (!childD) {
  139. return callback(new Error('No speech to kill'));
  140. }
  141. if (process.platform === 'linux') {
  142. // TODO: Need to ensure the following is true for all users, not just me. Danger Zone!
  143. // On my machine, original childD.pid process is completely gone. Instead there is now a
  144. // childD.pid + 1 sh process. Kill it and nothing happens. There's also a childD.pid + 2
  145. // aplay process. Kill that and the audio actually stops.
  146. process.kill(childD.pid + 2);
  147. } else if (process.platform === 'win32') {
  148. childD.stdin.pause();
  149. child_process.exec('taskkill /pid ' + childD.pid + ' /T /F')
  150. } else {
  151. childD.stdin.pause();
  152. childD.kill();
  153. }
  154. childD = null;
  155. callback(null);
  156. };
  157. function convertSpeed(speed) {
  158. return Math.ceil(say.base_speed * speed);
  159. }