bot.py 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. import pymumble_py3 as pymumble
  2. from pymumble_py3.constants import *
  3. import subprocess as sp
  4. import time
  5. import sys
  6. import os
  7. import fcntl
  8. import audioop
  9. import argparse
  10. def message_received(message):
  11. global is_streaming
  12. global silence_time
  13. command=message.message
  14. if command == "/start":
  15. is_streaming = True
  16. silence_time = 0
  17. mumble.users.myself.recording()
  18. elif command == "/stop":
  19. is_streaming = False
  20. mumble.users.myself.unrecording()
  21. parser = argparse.ArgumentParser(description='Regia pienamente automatizzata')
  22. parser.add_argument('--channel', help='Set channel', default="")
  23. parser.add_argument('--name', help='Set bot nickname', default="RadioRobbot")
  24. parser.add_argument('--server', help='Set server', default="mumble.esiliati.org")
  25. parser.add_argument('--port', help='Set port', type=int, default=64738)
  26. parser.add_argument('--stream', action='store_true', help='Ignore commands in chat and stream everything')
  27. parser.add_argument('--auto-suspend-stream', action='store_true', help='Ignore commands in chat and stream everything')
  28. parser.add_argument('--max-silence-time', type=int, help='max silence time in seconds', default=30)
  29. sys.argv.pop(0)
  30. args = parser.parse_args(sys.argv)
  31. pwd = "" # password
  32. server = args.server
  33. nick = args.name
  34. channel = args.channel
  35. port = args.port
  36. is_streaming=False
  37. stream_always= args.stream
  38. auto_suspend_stream = args.auto_suspend_stream
  39. silence_limit = 30
  40. # Spin up a client and connect to mumble server
  41. mumble = pymumble.Mumble(server, nick, password=pwd, port=port)
  42. mumble.callbacks.set_callback(PYMUMBLE_CLBK_TEXTMESSAGERECEIVED, message_received)
  43. mumble.set_receive_sound(1) # Enable receiving sound from mumble server
  44. mumble.start()
  45. mumble.is_ready() # Wait for client is ready
  46. mumble.channels.find_by_name(channel).move_in()
  47. mumble.users.myself.mute()
  48. if is_streaming:
  49. mumble.users.myself.recording()
  50. BUFFER = 0.1
  51. BITRATE = 48000
  52. RESOLUTION = 10 # in ms
  53. FLOAT_RESOLUTION = float(RESOLUTION) / 1000
  54. MONO_CHUNK_SIZE = BITRATE * 2 * RESOLUTION / 1000
  55. STEREO_CHUNK_SIZE = MONO_CHUNK_SIZE * 2
  56. silent = b"\x00" * int(STEREO_CHUNK_SIZE)
  57. cursor_time = None
  58. cursor_time = time.time() - BUFFER
  59. silence_time = 0
  60. silence_limit_ms = silence_limit * 1000
  61. while mumble.is_alive():
  62. if cursor_time < time.time() - BUFFER:
  63. base_sound = None
  64. try:
  65. for user in mumble.users.values(): # check the audio queue of each user
  66. if user.sound.is_sound():
  67. # available sound is to be treated now and not later
  68. sound = user.sound.get_sound(FLOAT_RESOLUTION)
  69. stereo_pcm = audioop.tostereo(sound.pcm, 2, 1, 1)
  70. if base_sound == None:
  71. base_sound = stereo_pcm
  72. else:
  73. base_sound = audioop.add(base_sound, stereo_pcm, 2)
  74. except RuntimeError:
  75. print("ignored exception in stderr...", file=sys.stderr)
  76. if is_streaming or stream_always:
  77. if base_sound:
  78. silence_time = 0
  79. sys.stdout.buffer.write(base_sound)
  80. else:
  81. silence_time += RESOLUTION
  82. sys.stdout.buffer.write(silent)
  83. if auto_suspend_stream and (silence_time >= silence_limit_ms):
  84. is_streaming = False
  85. mumble.users.myself.unrecording()
  86. cursor_time += FLOAT_RESOLUTION
  87. else:
  88. time.sleep(FLOAT_RESOLUTION)