bot.py 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. import argparse
  2. import audioop
  3. import logging
  4. import sys
  5. import time
  6. from functools import partial
  7. import pymumble_py3 as pymumble
  8. from pymumble_py3.constants import PYMUMBLE_CLBK_TEXTMESSAGERECEIVED
  9. is_streaming = False
  10. silence_time = 0
  11. def message_received(mumble, message):
  12. global is_streaming
  13. global silence_time
  14. command = message.message
  15. if command == "/start":
  16. is_streaming = True
  17. silence_time = 0
  18. mumble.my_channel().send_text_message("Diretta iniziata")
  19. logging.info("Diretta iniziata")
  20. mumble.users.myself.recording()
  21. elif command == "/stop":
  22. is_streaming = False
  23. mumble.my_channel().send_text_message("Diretta terminata")
  24. logging.info("Diretta terminata")
  25. mumble.users.myself.unrecording()
  26. def get_parser():
  27. parser = argparse.ArgumentParser(description="Regia pienamente automatizzata")
  28. parser.add_argument("--channel", help="Set channel", default="")
  29. parser.add_argument("--name", help="Set bot nickname", default="RadioRobbot")
  30. parser.add_argument("--server", help="Set server", default="mumble.esiliati.org")
  31. parser.add_argument("--port", help="Set port", type=int, default=64738)
  32. parser.add_argument(
  33. "--stream",
  34. action="store_true",
  35. help="Ignore commands in chat and stream everything",
  36. )
  37. parser.add_argument(
  38. "--auto-suspend-stream",
  39. action="store_true",
  40. help="Ignore commands in chat and stream everything",
  41. )
  42. parser.add_argument(
  43. "--max-silence-time", type=int, help="max silence time in seconds", default=30
  44. )
  45. return parser
  46. def main():
  47. global is_streaming
  48. global silence_time
  49. args = get_parser().parse_args()
  50. logging.basicConfig(level=logging.DEBUG)
  51. pwd = "" # password
  52. server = args.server
  53. nick = args.name
  54. channel = args.channel
  55. port = args.port
  56. is_streaming = False
  57. stream_always = args.stream
  58. # Spin up a client and connect to mumble server
  59. mumble = pymumble.Mumble(server, nick, password=pwd, port=port)
  60. mumble.callbacks.set_callback(
  61. PYMUMBLE_CLBK_TEXTMESSAGERECEIVED, partial(message_received, mumble)
  62. )
  63. mumble.set_receive_sound(1) # Enable receiving sound from mumble server
  64. mumble.start()
  65. mumble.is_ready() # Wait for client is ready
  66. mumble.channels.find_by_name(channel).move_in()
  67. mumble.users.myself.mute()
  68. if is_streaming:
  69. mumble.users.myself.recording()
  70. BUFFER = 0.1
  71. BITRATE = 48000
  72. RESOLUTION = 10 # in ms
  73. FLOAT_RESOLUTION = float(RESOLUTION) / 1000
  74. MONO_CHUNK_SIZE = BITRATE * 2 * RESOLUTION / 1000
  75. STEREO_CHUNK_SIZE = MONO_CHUNK_SIZE * 2
  76. silent = b"\x00" * int(STEREO_CHUNK_SIZE)
  77. cursor_time = time.time() - BUFFER
  78. while mumble.is_alive():
  79. if cursor_time < time.time() - BUFFER:
  80. base_sound = None
  81. try:
  82. for user in mumble.users.values(): # check the audio queue of each user
  83. if user.sound.is_sound():
  84. # available sound is to be treated now and not later
  85. sound = user.sound.get_sound(FLOAT_RESOLUTION)
  86. stereo_pcm = audioop.tostereo(sound.pcm, 2, 1, 1)
  87. if base_sound is None:
  88. base_sound = stereo_pcm
  89. else:
  90. base_sound = audioop.add(base_sound, stereo_pcm, 2)
  91. except RuntimeError:
  92. print("ignored exception in stderr...", file=sys.stderr)
  93. if is_streaming or stream_always:
  94. if base_sound:
  95. silence_time = 0
  96. sys.stdout.buffer.write(base_sound)
  97. else:
  98. silence_time += RESOLUTION
  99. sys.stdout.buffer.write(silent)
  100. if (
  101. args.auto_suspend_stream
  102. and (silence_time >= args.max_silence_time * 1000)
  103. and is_streaming
  104. ):
  105. is_streaming = False
  106. logging.info("max-silence-time reached")
  107. mumble.my_channel().send_text_message(
  108. "Diretta terminata in automatico dopo %d secondi circa di silenzio"
  109. % args.max_silence_time
  110. )
  111. mumble.users.myself.unrecording()
  112. cursor_time += FLOAT_RESOLUTION
  113. else:
  114. time.sleep(FLOAT_RESOLUTION)
  115. if __name__ == "__main__":
  116. main()