« get me outta code hell

http-music - Command-line music player + utils (not a server!)
about summary refs log tree commit diff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/loop-play.js88
1 files changed, 65 insertions, 23 deletions
diff --git a/src/loop-play.js b/src/loop-play.js
index d22d55d..e43cb31 100644
--- a/src/loop-play.js
+++ b/src/loop-play.js
@@ -4,6 +4,7 @@ const { spawn } = require('child_process')
 const promisifyProcess = require('./promisify-process')
 const sanitize = require('sanitize-filename')
 const tempy = require('tempy')
+const path = require('path')
 
 const EventEmitter = require('events')
 
@@ -46,7 +47,7 @@ class DownloadController extends EventEmitter {
     // If the picker returns null, nothing was picked; that means that we
     // should stop now. No point in trying to play nothing!
     if (picked == null) {
-      this.wavFile = null
+      this.playFile = null
       return false
     }
 
@@ -67,26 +68,18 @@ class DownloadController extends EventEmitter {
     // got.
     const fromFile = await this.downloader(downloaderArg)
 
-    // Before we convert the file, we'll check if it's already an audio file.
-    // This goes on the assumption that if avprobe understands a file, play
-    // also does; which is probably true almost all the time.
-    let probeCode
+    // Ignore the '.' at the start.
+    const format = path.extname(fromFile).slice(1)
 
-    try {
-      // Well, lovely. avprobe ALWAYS outputs "# avprobe output" - even if
-      // its loglevel is set to silent! Blasphemy, but whatever. We're forced
-      // to not pipe to stdout (which we do by passing false to
-      // promisifyProcess).
-      await promisifyProcess(spawn('avprobe', [fromFile]), false)
-    } catch(errorCode) {
-      probeCode = errorCode
-    }
+    // We'll only want to convert the "from" file if it's not already supported
+    // by SoX; so we check the supported format list.
+
+    const supportedFormats = await this.getSupportedFormats()
 
-    // We'll use this wav file later, to actually play the track.
-    if (probeCode > 0) {
-      this.wavFile = await this.convert(picked, fromFile)
+    if (supportedFormats.includes(format)) {
+      this.playFile = fromFile
     } else {
-      this.wavFile = fromFile
+      this.playFile = await this.convert(picked, fromFile)
     }
 
     // If this download was destroyed, we quit now; we don't want to emit that
@@ -99,13 +92,62 @@ class DownloadController extends EventEmitter {
     this.emit('downloadFinished')
   }
 
+  async getSupportedFormats() {
+    // Gets the formats supported by SoX (i.e., the `play` command) by
+    // searching the help output for the line that starts with
+    // 'AUDIO FILE FORMATS:'. This seems to be the only way to list the formats
+    // that any installation of SoX accepts; in the documentation, this is also
+    // the recommended way (though it's not particularly suggested to be parsed
+    // automatically): "To see if SoX has support for an optional format or
+    // device, enter sox −h and look for its name under the list: 'AUDIO FILE
+    // FORMATS' or 'AUDIO DEVICE DRIVERS'."
+
+    if (this._supportedSoXFormats) {
+      return this._supportedSoXFormats
+    }
+
+    const sox = spawn('sox', ['-h'])
+
+    const buffers = []
+
+    sox.stdout.on('data', buf => {
+      buffers.push(buf)
+    })
+
+    await promisifyProcess(sox, false)
+
+    const str = Buffer.concat(buffers).toString()
+
+    const lines = str.split('\n')
+
+    const prefix = 'AUDIO FILE FORMATS: '
+
+    const formatsLine = lines.find(line => line.startsWith(prefix))
+
+    const formats = formatsLine.slice(prefix.length).split(' ')
+
+    this._supportedSoXFormats = formats
+
+    return formats
+  }
+
   async convert(picked, fromFile) {
-    // The "to" file is simply a WAV file. We give this WAV file a specific
+    // The "to" file is simply an MP3 file. We give this MP3 file a specific
     // name - the title of the track we got earlier, sanitized to be file-safe
     // - so that when `play` outputs the name of the song, it's obvious to the
     // user what's being played.
+    //
+    // Previously a WAV file was used here. Converting to a WAV file is
+    // considerably faster than converting to an MP3; however, the file sizes
+    // of WAVs tend to be drastically larger than MP3s. When saving disk space
+    // is one of the greatest concerns (it's essentially the point of
+    // http-music!), it's better to opt for an MP3. Additionally, most audio
+    // convertion is done in the background, while another track is already
+    // playing, so an extra few seconds of background time can hardly be
+    // noticed.
+    const title = picked[1]
     const tempDir = tempy.directory()
-    const toFile = tempDir + `/.${sanitize(title)}.wav`
+    const toFile = tempDir + `/.${sanitize(title)}.mp3`
 
     // Now that we've got the `to` and `from` file names, we can actually do
     // the convertion. We don't want any output from `avconv` at all, since the
@@ -145,7 +187,7 @@ class DownloadController extends EventEmitter {
       }
     }
 
-    return to
+    return toFile
   }
 
   skipUpNext() {
@@ -183,11 +225,11 @@ class PlayController {
 
     await this.downloadController.downloadNext()
 
-    while (this.downloadController.wavFile) {
+    while (this.downloadController.playFile) {
       this.currentTrack = this.downloadController.pickedTrack
 
 
-      const file = this.downloadController.wavFile
+      const file = this.downloadController.playFile
       const playProcess = spawn('play', [...this.playArgs, file])
       const playPromise = promisifyProcess(playProcess)
       this.process = playProcess