Explorar o código

fix permissions+user to use host beet db and user files, can play a song!
did not get pause/play to be consistently working
did not get skipping to work

tarfeef101 %!s(int64=4) %!d(string=hai) anos
pai
achega
6d6ee6ae01
Modificáronse 3 ficheiros con 150 adicións e 6 borrados
  1. 11 3
      Dockerfile
  2. 3 2
      src/package.json
  3. 136 1
      src/viki.js

+ 11 - 3
Dockerfile

@@ -1,6 +1,14 @@
 FROM node:15-alpine
-RUN apk add git make g++ gcc musl-dev python beets python3
-COPY src/ /home/node/src/
-WORKDIR /home/node/src
+RUN apk add git make g++ gcc musl-dev python beets python3 ffmpeg shadow && \
+    groupmod --new-name tdedhar node && \
+    usermod -l tdedhar node && \
+    mv /home/node /home/tdedhar && \
+    chown -R tdedhar /home/tdedhar
+USER tdedhar
+ENV HOME /home/tdedhar
+RUN mkdir -p /home/tdedhar/src && \
+    mkdir -p /home/tdedhar/.config/beets
+COPY src/ /home/tdedhar/src/
+WORKDIR /home/tdedhar/src
 RUN yarn install
 ENTRYPOINT node viki.js

+ 3 - 2
src/package.json

@@ -1,7 +1,7 @@
 {
   "name": "viki",
-  "version": "0.0.1",
-  "description": "go fuck yourself",
+  "version": "0.1.1",
+  "description": "A music bot for local file streaming using the beets music DB",
   "main": "viki.js",
   "license": "MIT",
   "dependencies": {
@@ -9,6 +9,7 @@
     "bufferutil": "^4.0.2",
     "discord.js": "^12.5.1",
     "erlpack": "discord/erlpack",
+    "ffmpeg-static": "^4.2.7",
     "libsodium-wrappers": "^0.7.8",
     "utf-8-validate": "^5.0.3",
     "zlib-sync": "^0.1.7"

+ 136 - 1
src/viki.js

@@ -44,7 +44,7 @@ function play()
   dispatcher.setVolume(0.2);
   dispatcher.setBitrate(96);
 
-  dispatcher.on("end", reason => 
+  dispatcher.on("finish", reason => 
   {
     if (!(playlist.isEmpty()))
     {
@@ -136,6 +136,141 @@ client.on('message', async msg => {
       msg.reply(`Sorry ${msg.author.username}, that channel doesn't appear to exist.`);
     }
   }
+
+  if (command === "addmusic") // adds songs to queue, starts playback if none already
+  {
+    var type = args[0];
+    if (!(musicTypes.includes(type)))
+    {
+      return msg.reply("Sorry, that is not a valid command. Please enter something from: " + musicTypesString);
+    }
+    if (type == 'song' || type == 'track') type = 'title'; // account for poor beets API
+    args.splice(0, 1);
+    const query = args.join(' '); // creates a single string of all args (the query)
+    var path; // this will hold the filepaths from our query
+    exec(`beet ls -p ${type}:${query} | wc -l`, function (error, stdout, stderr)
+    {
+      if (error)
+      {
+        return msg.reply(`Sorry, I encountered an issue looking for that: ${error}`);
+      }
+      else if (stdout === '\n' || stdout === '' || stdout === undefined)
+      {
+        return msg.reply(`There were no results that matched your search. Please give the type and name of your query (e.g. song songname, album albumname...)`);
+      }
+      else
+      {  
+        exec(`beet ls -p ${type}:${query}`, function (error, stdout, stderr)
+        {
+          if (error)
+          {
+            return msg.reply(`Sorry, I encountered an issue looking for that: ${error}`);
+          }
+          else
+          {
+            path = stdout.trim();
+            path = path.split("\n"); // now an array of paths (with spaces)
+            
+            // for each song, get the path and readable info, send to queue
+            for (var i = 0; i < path.length; i++)
+            {
+              let filepathRaw = path[i];
+              path[i] = path[i].replaceAll(" ", "\\ ");
+              path[i] = path[i].replaceAll("'", "\\'");
+              path[i] = path[i].replaceAll("&", "\\\46");
+              path[i] = path[i].replaceAll("(", "\\(");
+              path[i] = path[i].replaceAll(")", "\\)");
+              let filepath = path[i]; // path[i] descoped in callback
+
+              exec(`beet ls ${path[i]}`, function (error, stdouts, stderr)
+              {
+                if (error)
+                {
+                  return msg.reply(`Sorry, I encountered an issue looking for song ${i}: ${error}`);
+                }
+                else
+                {
+                  stdouts = stdouts.trim();
+                  playlist.enqueue([filepathRaw, msg, stdouts]);
+                  
+                  // check if music is playing, if not start it
+                  if ((dispatcher === undefined || dispatcher.destroyed == true) && !(playlist.isEmpty()))
+                  {
+                    play();
+                  }
+                }
+              });
+            }
+          }
+        });
+      }
+
+      let amt = stdout.trim();
+      msg.reply(`${amt} songs added!`);
+    });
+  }
+  
+  if (command === 'stop') // clears playlist, stops music
+  {
+    playlist.reset();
+    dispatcher.destroy();
+    console.log("Playback stopped, playlist cleared.")
+  }
+  
+  if (command === 'next') // returns next song in playlist, or informs that there is none
+  {
+    if (playlist.isEmpty())
+    {
+      msg.reply("The playlist is empty.");
+    }
+    else
+    {
+      const next = playlist.peek();
+      msg.reply(`Next song is: ${next[2]}.`);
+    }
+  }
+  
+  if (command === 'pause') // pauses the dispatcher if playing, or does nothing
+  {
+    dispatcher.pause();
+    msg.reply("Playback paused.");
+  }
+  
+  if (command === 'resume') // resumes the dispatcher, or does nothing
+  {
+    dispatcher.resume();
+    msg.reply("Playback resumed.");
+  }
+  
+  if (command === 'skip') // starts playing the next song in the queue if it exists
+  {
+    if (playlist.isEmpty())
+    {
+      msg.reply("Sorry, the playlist is empty.");
+    }
+    else
+    {
+      function resolveEnd()
+      {
+        return new Promise((success, fail) =>
+        {
+          dispatcher.destroy();
+
+          dispatcher.on("finish", () =>
+          {
+            success('Track skipped!');
+          });
+
+          dispatcher.on("error", () =>
+          {
+            fail('Couldn\'t skip :(');
+          });
+        });
+      }
+
+      resolveEnd();
+    }
+  }
 });
 
 client.login(config.token);