First commit with working stuff
First push, no css for now
This commit is contained in:
parent
4716cbb46f
commit
1d7fa3e473
38
README.md
38
README.md
@ -6,28 +6,36 @@
|
|||||||
Botamusique is a mumble bot which goal is to allow users to listen music together with its audio output.
|
Botamusique is a mumble bot which goal is to allow users to listen music together with its audio output.
|
||||||
Predicted functionnalities will be the one you could expect from any classic music player.
|
Predicted functionnalities will be the one you could expect from any classic music player.
|
||||||
|
|
||||||
|
1. Who to start
|
||||||
|
You need to create a folder for all your music. Organize your music by subfolder.
|
||||||
|
The main folder need to be declare into the config (with a '/' at the end)
|
||||||
|
|
||||||
1.TODO list
|
|
||||||
|
2.TODO list
|
||||||
|
|
||||||
### TODOLIST
|
### TODOLIST
|
||||||
|
|
||||||
#### Features
|
#### Features
|
||||||
- [ ] Next song
|
- [ ] Next song
|
||||||
- [ ] Previous song
|
- [ ] Previous song
|
||||||
- [ ] Randomizer
|
- [x] Randomizer
|
||||||
- [ ] Looking for songs previously downloaded in a folder by users.
|
- [ ] Looking for songs previously downloaded in a folder by users.
|
||||||
|
|
||||||
#### Commands
|
#### Commands with the interface
|
||||||
- [ ] list
|
- [x] list
|
||||||
- [ ] play
|
- [x] play
|
||||||
- [ ] playfolder
|
- [x] playfolder
|
||||||
- [ ] list
|
- [x] random
|
||||||
- [ ] random
|
|
||||||
- [ ] volume
|
|
||||||
- [ ] skip
|
|
||||||
- [ ] stop
|
|
||||||
- [ ] joinme
|
|
||||||
- [ ] away
|
|
||||||
|
|
||||||
#### Future
|
#### Commands by message to the bot
|
||||||
A web interface could eventually be added to make it easier to use.
|
- [x] volume
|
||||||
|
- [ ] skip
|
||||||
|
- [x] stop
|
||||||
|
- [x] joinme
|
||||||
|
- [x] away
|
||||||
|
|
||||||
|
#### Web Interface
|
||||||
|
- [x] Primary functions
|
||||||
|
- [ ] CSS
|
||||||
|
|
||||||
|
The web interface listen the 8181 port !
|
||||||
|
28
configuration.ini
Normal file
28
configuration.ini
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
[bot]
|
||||||
|
comment = Coucou, Je suis née du savoir du Azlux
|
||||||
|
volume = 0.1
|
||||||
|
admin = Azlux;AzMobile
|
||||||
|
music_folder = /home/dmichel/botamusique/music/
|
||||||
|
|
||||||
|
[debug]
|
||||||
|
ffmpeg = False
|
||||||
|
mumbleConnection = False
|
||||||
|
|
||||||
|
[command]
|
||||||
|
play_file = play
|
||||||
|
stop = stop
|
||||||
|
current_music = np
|
||||||
|
volume = v
|
||||||
|
kill = kill
|
||||||
|
stop_and_getout = oust
|
||||||
|
joinme = joinme
|
||||||
|
list = list
|
||||||
|
|
||||||
|
[strings]
|
||||||
|
current_volume = volume : %d%%
|
||||||
|
change_volume = volume : %d%% par %s
|
||||||
|
bad_command = Commande incorrecte
|
||||||
|
not_admin = T'es pas admin, patate !
|
||||||
|
not_playing = Aucun stream en lecture
|
||||||
|
bad_file = Bad file asked
|
||||||
|
no_file = Not file here
|
45
interface.py
Normal file
45
interface.py
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
#!/usr/bin/python3
|
||||||
|
|
||||||
|
from flask import Flask, render_template, request
|
||||||
|
import variables as var
|
||||||
|
import os.path
|
||||||
|
from os import listdir
|
||||||
|
import random
|
||||||
|
|
||||||
|
web = Flask(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
@web.route("/", methods=['GET', 'POST'])
|
||||||
|
def index():
|
||||||
|
folder_path = var.music_folder
|
||||||
|
files = {}
|
||||||
|
dirs = [f for f in listdir(folder_path) if os.path.isdir(os.path.join(folder_path, f))]
|
||||||
|
for director in dirs:
|
||||||
|
files[director] = [f for f in listdir(folder_path + director) if os.path.isfile(os.path.join(folder_path + director, f))]
|
||||||
|
|
||||||
|
if request.method == 'POST':
|
||||||
|
if 'add_music' in request.form and ".." not in request.form['add_music']:
|
||||||
|
var.playlist.append(request.form['add_music'])
|
||||||
|
if 'add_folder' in request.form and ".." not in request.form['add_folder']:
|
||||||
|
var.playlist.extend(files[request.form['add_folder']])
|
||||||
|
elif 'delete_music' in request.form:
|
||||||
|
var.playlist.remove(request.form['delete_music'])
|
||||||
|
elif 'action' in request.form:
|
||||||
|
action = request.form['action']
|
||||||
|
if action == "randomize":
|
||||||
|
random.shuffle(var.playlist)
|
||||||
|
if var.current_music:
|
||||||
|
current_music = var.current_music[len(var.music_folder):]
|
||||||
|
else:
|
||||||
|
current_music = None
|
||||||
|
|
||||||
|
return render_template('index.html',
|
||||||
|
current_music=current_music,
|
||||||
|
user=var.user,
|
||||||
|
playlist=var.playlist,
|
||||||
|
all_files=files
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
web.run(port=8181, host="0.0.0.0")
|
201
mumbleBot.py
Normal file
201
mumbleBot.py
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
#!/usr/bin/python3
|
||||||
|
import threading
|
||||||
|
|
||||||
|
import time
|
||||||
|
import sys
|
||||||
|
import signal
|
||||||
|
import configparser
|
||||||
|
import audioop
|
||||||
|
import subprocess as sp
|
||||||
|
import argparse
|
||||||
|
import os.path
|
||||||
|
from os import listdir
|
||||||
|
import pymumble.pymumble_py3 as pymumble
|
||||||
|
import interface
|
||||||
|
import variables as var
|
||||||
|
|
||||||
|
|
||||||
|
class MumbleBot:
|
||||||
|
def __init__(self):
|
||||||
|
signal.signal(signal.SIGINT, self.ctrl_caught)
|
||||||
|
|
||||||
|
self.config = configparser.ConfigParser(interpolation=None)
|
||||||
|
self.config.read("configuration.ini", encoding='latin-1')
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser(description='Bot for playing radio stream on Mumble')
|
||||||
|
parser.add_argument("-s", "--server", dest="host", type=str, required=True, help="The server's hostame of a mumble server")
|
||||||
|
parser.add_argument("-u", "--user", dest="user", type=str, required=True, help="Username you wish, Default=abot")
|
||||||
|
parser.add_argument("-P", "--password", dest="password", type=str, default="", help="Password if server requires one")
|
||||||
|
parser.add_argument("-p", "--port", dest="port", type=int, default=64738, help="Port for the mumble server")
|
||||||
|
parser.add_argument("-c", "--channel", dest="channel", type=str, default="", help="Default chanel for the bot")
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
self.volume = self.config.getfloat('bot', 'volume')
|
||||||
|
self.channel = args.channel
|
||||||
|
var.current_music = None
|
||||||
|
var.playlist = []
|
||||||
|
var.user = args.user
|
||||||
|
var.music_folder = self.config.get('bot', 'music_folder')
|
||||||
|
self.exit = False
|
||||||
|
self.nb_exit = 0
|
||||||
|
self.thread = None
|
||||||
|
|
||||||
|
t = threading.Thread(target=start_web_interface)
|
||||||
|
t.daemon = True
|
||||||
|
t.start()
|
||||||
|
|
||||||
|
self.mumble = pymumble.Mumble(args.host, user=args.user, port=args.port, password=args.password,
|
||||||
|
debug=self.config.getboolean('debug', 'mumbleConnection'))
|
||||||
|
self.mumble.callbacks.set_callback("text_received", self.message_received)
|
||||||
|
|
||||||
|
self.mumble.set_codec_profile("audio")
|
||||||
|
self.mumble.start() # start the mumble thread
|
||||||
|
self.mumble.is_ready() # wait for the connection
|
||||||
|
self.set_comment()
|
||||||
|
self.mumble.users.myself.unmute() # by sure the user is not muted
|
||||||
|
if self.channel:
|
||||||
|
self.mumble.channels.find_by_name(self.channel).move_in()
|
||||||
|
self.mumble.set_bandwidth(200000)
|
||||||
|
|
||||||
|
self.loop()
|
||||||
|
|
||||||
|
def ctrl_caught(self, signal, frame):
|
||||||
|
print("\ndeconnection asked")
|
||||||
|
self.exit = True
|
||||||
|
self.stop()
|
||||||
|
if self.nb_exit > 1:
|
||||||
|
print("Forced Quit")
|
||||||
|
sys.exit(0)
|
||||||
|
self.nb_exit += 1
|
||||||
|
|
||||||
|
def message_received(self, text):
|
||||||
|
message = text.message
|
||||||
|
if message[0] == '!':
|
||||||
|
message = message[1:].split(' ', 1)
|
||||||
|
if len(message) > 0:
|
||||||
|
command = message[0]
|
||||||
|
parameter = ''
|
||||||
|
if len(message) > 1:
|
||||||
|
parameter = message[1]
|
||||||
|
else:
|
||||||
|
return
|
||||||
|
|
||||||
|
print(command + ' - ' + parameter + ' by ' + self.mumble.users[text.actor]['name'])
|
||||||
|
|
||||||
|
if command == self.config.get('command', 'play_file') and parameter:
|
||||||
|
path = self.config.get('bot', 'music_folder') + parameter
|
||||||
|
if "/" in parameter:
|
||||||
|
self.mumble.users[text.actor].send_message(self.config.get('strings', 'bad_file'))
|
||||||
|
elif os.path.isfile(path):
|
||||||
|
self.launch_play_file(path)
|
||||||
|
else:
|
||||||
|
self.mumble.users[text.actor].send_message(self.config.get('strings', 'bad_file'))
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'stop'):
|
||||||
|
self.stop()
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'kill'):
|
||||||
|
if self.is_admin(text.actor):
|
||||||
|
self.stop()
|
||||||
|
self.exit = True
|
||||||
|
else:
|
||||||
|
self.mumble.users[text.actor].send_message(self.config.get('strings', 'not_admin'))
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'stop_and_getout'):
|
||||||
|
self.stop()
|
||||||
|
if self.channel:
|
||||||
|
self.mumble.channels.find_by_name(self.channel).move_in()
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'joinme'):
|
||||||
|
self.mumble.users.myself.move_in(self.mumble.users[text.actor]['channel_id'])
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'volume'):
|
||||||
|
if parameter is not None and parameter.isdigit() and 0 <= int(parameter) <= 100:
|
||||||
|
self.volume = float(float(parameter) / 100)
|
||||||
|
self.send_msg_channel(self.config.get('strings', 'change_volume') % (
|
||||||
|
int(self.volume * 100), self.mumble.users[text.actor]['name']))
|
||||||
|
else:
|
||||||
|
self.send_msg_channel(self.config.get('strings', 'current_volume') % int(self.volume * 100))
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'current_music'):
|
||||||
|
if var.current_music is not None:
|
||||||
|
self.send_msg_channel(var.current_music)
|
||||||
|
else:
|
||||||
|
self.mumble.users[text.actor].send_message(self.config.get('strings', 'not_playing'))
|
||||||
|
|
||||||
|
elif command == self.config.get('command', 'list'):
|
||||||
|
folder_path = self.config.get('bot', 'music_folder')
|
||||||
|
files = [f for f in listdir(folder_path) if os.path.isfile(os.path.join(folder_path, f))]
|
||||||
|
if files:
|
||||||
|
self.mumble.users[text.actor].send_message('<br>'.join(files))
|
||||||
|
else:
|
||||||
|
self.mumble.users[text.actor].send_message(self.config.get('strings', 'no_file'))
|
||||||
|
|
||||||
|
else:
|
||||||
|
self.mumble.users[text.actor].send_message(self.config.get('strings', 'bad_command'))
|
||||||
|
|
||||||
|
def is_admin(self, user):
|
||||||
|
username = self.mumble.users[user]['name']
|
||||||
|
list_admin = self.config.get('bot', 'admin').split(';')
|
||||||
|
if username in list_admin:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def launch_play_file(self, path=None):
|
||||||
|
if not path:
|
||||||
|
path = self.config.get('bot', 'music_folder') + var.current_music
|
||||||
|
if self.config.getboolean('debug', 'ffmpeg'):
|
||||||
|
ffmpeg_debug = "debug"
|
||||||
|
else:
|
||||||
|
ffmpeg_debug = "warning"
|
||||||
|
command = ["ffmpeg", '-v', ffmpeg_debug, '-nostdin', '-i', path, '-ac', '1', '-f', 's16le', '-ar', '48000', '-']
|
||||||
|
self.thread = sp.Popen(command, stdout=sp.PIPE, bufsize=480)
|
||||||
|
var.current_music = path
|
||||||
|
|
||||||
|
def loop(self):
|
||||||
|
while not self.exit:
|
||||||
|
|
||||||
|
while self.mumble.sound_output.get_buffer_size() > 0.5:
|
||||||
|
time.sleep(0.01)
|
||||||
|
if self.thread:
|
||||||
|
raw_music = self.thread.stdout.read(480)
|
||||||
|
if raw_music:
|
||||||
|
self.mumble.sound_output.add_sound(audioop.mul(raw_music, 2, self.volume))
|
||||||
|
else:
|
||||||
|
time.sleep(0.1)
|
||||||
|
else:
|
||||||
|
time.sleep(0.1)
|
||||||
|
|
||||||
|
if (self.thread is None or not raw_music) and len(var.playlist) != 0:
|
||||||
|
print("plop")
|
||||||
|
var.current_music = var.playlist[0]
|
||||||
|
var.playlist.pop(0)
|
||||||
|
self.launch_play_file()
|
||||||
|
|
||||||
|
while self.mumble.sound_output.get_buffer_size() > 0:
|
||||||
|
time.sleep(0.01)
|
||||||
|
time.sleep(0.5)
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
if self.thread:
|
||||||
|
var.current_music = None
|
||||||
|
self.thread.kill()
|
||||||
|
self.thread = None
|
||||||
|
var.playlist = []
|
||||||
|
|
||||||
|
def set_comment(self):
|
||||||
|
self.mumble.users.myself.comment(self.config.get('bot', 'comment'))
|
||||||
|
|
||||||
|
def send_msg_channel(self, msg, channel=None):
|
||||||
|
if not channel:
|
||||||
|
channel = self.mumble.channels[self.mumble.users.myself['channel_id']]
|
||||||
|
channel.send_text_message(msg)
|
||||||
|
|
||||||
|
|
||||||
|
def start_web_interface():
|
||||||
|
interface.web.run(port=8181, host="0.0.0.0")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
botamusique = MumbleBot()
|
0
static/index.css
Normal file
0
static/index.css
Normal file
53
templates/index.html
Normal file
53
templates/index.html
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title></title>
|
||||||
|
<link rel="stylesheet" href="{{ url_for('static', filename='index.css') }}">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<a href="."><h5>Refresh</h5></a>
|
||||||
|
<br>
|
||||||
|
|
||||||
|
<div id="playlist">
|
||||||
|
Current Playing :
|
||||||
|
{% if current_music %}
|
||||||
|
{{ current_music }}
|
||||||
|
{% else %}
|
||||||
|
No music
|
||||||
|
{% endif %}
|
||||||
|
<br>
|
||||||
|
Playlist :
|
||||||
|
<form method="post"><input type=text value="randomize" name="action" hidden><input type="submit" value="Randomize playlist"></form>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
{% for m in playlist %}
|
||||||
|
<li>{{ m }}
|
||||||
|
<form method="post"><input type=text value={{ m }} name="delete_music" hidden><input type="submit" value="X"></form>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
<br>
|
||||||
|
{% for dir in all_files %}
|
||||||
|
{{ dir }}
|
||||||
|
<form method="post"><input type=text value={{ dir }} name="add_folder" hidden><input type="submit" value="add all folder"></form>
|
||||||
|
<br>
|
||||||
|
<ul>
|
||||||
|
|
||||||
|
{% for m in all_files[dir] %}
|
||||||
|
<li>
|
||||||
|
<form method="post"><input type=text value="{{ dir }}/{{ m }}" name="add_music" hidden><input type="submit" value={{ m }}></form>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
{% endfor %}
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div id="browser">
|
||||||
|
|
||||||
|
</div>
|
||||||
|
<div id="upload">
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
4
variables.py
Normal file
4
variables.py
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
current_music = ""
|
||||||
|
playlist = []
|
||||||
|
user = ""
|
||||||
|
music_folder = ""
|
Loading…
x
Reference in New Issue
Block a user