bot.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  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("--password", help="Set password", default="")
  32. parser.add_argument("--port", help="Set port", type=int, default=64738)
  33. parser.add_argument(
  34. "--stream",
  35. action="store_true",
  36. help="Ignore commands in chat and stream everything",
  37. )
  38. parser.add_argument(
  39. "--auto-suspend-stream",
  40. action="store_true",
  41. help="Ignore commands in chat and stream everything",
  42. )
  43. parser.add_argument(
  44. "--max-silence-time", type=int, help="max silence time in seconds", default=30
  45. )
  46. parser.add_argument(
  47. "--tokens",
  48. help="Set tokens list",
  49. nargs="*"
  50. )
  51. return parser
  52. def main():
  53. global is_streaming
  54. global silence_time
  55. args = get_parser().parse_args()
  56. logging.basicConfig(level=logging.DEBUG)
  57. pwd = args.password
  58. server = args.server
  59. nick = args.name
  60. channel = args.channel
  61. port = args.port
  62. tokens = args.tokens
  63. is_streaming = False
  64. stream_always = args.stream
  65. # Spin up a client and connect to mumble server
  66. mumble = pymumble.Mumble(server, nick, password=pwd, port=port, tokens=tokens)
  67. mumble.callbacks.set_callback(
  68. PYMUMBLE_CLBK_TEXTMESSAGERECEIVED, partial(message_received, mumble)
  69. )
  70. mumble.set_receive_sound(1) # Enable receiving sound from mumble server
  71. mumble.start()
  72. mumble.is_ready() # Wait for client is ready
  73. mumble.channels.find_by_tree(channel.split('/')).move_in()
  74. mumble.users.myself.mute()
  75. if is_streaming:
  76. mumble.users.myself.recording()
  77. BUFFER = 0.1
  78. BITRATE = 48000
  79. RESOLUTION = 10 # in ms
  80. FLOAT_RESOLUTION = float(RESOLUTION) / 1000
  81. MONO_CHUNK_SIZE = BITRATE * 2 * RESOLUTION / 1000
  82. STEREO_CHUNK_SIZE = MONO_CHUNK_SIZE * 2
  83. silent = b"\x00" * int(STEREO_CHUNK_SIZE)
  84. cursor_time = time.time() - BUFFER
  85. while mumble.is_alive():
  86. if cursor_time < time.time() - BUFFER:
  87. base_sound = None
  88. try:
  89. for user in mumble.users.values(): # check the audio queue of each user
  90. if user.sound.is_sound():
  91. # available sound is to be treated now and not later
  92. sound = user.sound.get_sound(FLOAT_RESOLUTION)
  93. stereo_pcm = audioop.tostereo(sound.pcm, 2, 1, 1)
  94. if base_sound is None:
  95. base_sound = stereo_pcm
  96. else:
  97. base_sound = audioop.add(base_sound, stereo_pcm, 2)
  98. except RuntimeError:
  99. print("ignored exception in stderr...", file=sys.stderr)
  100. if is_streaming or stream_always:
  101. if base_sound:
  102. silence_time = 0
  103. sys.stdout.buffer.write(base_sound)
  104. else:
  105. silence_time += RESOLUTION
  106. sys.stdout.buffer.write(silent)
  107. if (
  108. args.auto_suspend_stream
  109. and (silence_time >= args.max_silence_time * 1000)
  110. and is_streaming
  111. ):
  112. is_streaming = False
  113. logging.info("max-silence-time reached")
  114. mumble.my_channel().send_text_message(
  115. "Diretta terminata in automatico dopo %d secondi circa di silenzio"
  116. % args.max_silence_time
  117. )
  118. mumble.users.myself.unrecording()
  119. cursor_time += FLOAT_RESOLUTION
  120. else:
  121. time.sleep(FLOAT_RESOLUTION)
  122. if __name__ == "__main__":
  123. main()