From 25279976b58daa24f20d0db299690cfb03cde52c Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Wed, 14 Aug 2024 14:33:26 -0700 Subject: [PATCH 01/12] add livekit command --- software/start.py | 80 ++++++++++++++++++++++++++++++++++++++++++++-- software/worker.py | 76 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 154 insertions(+), 2 deletions(-) create mode 100644 software/worker.py diff --git a/software/start.py b/software/start.py index 7fd2258..1a3c105 100644 --- a/software/start.py +++ b/software/start.py @@ -7,6 +7,11 @@ import importlib from source.server.tunnel import create_tunnel from source.server.async_server import start_server import subprocess +from livekit import api +import socket +import json +import segno +import time import signal @@ -60,7 +65,9 @@ def run( "--debug", help="Print latency measurements and save microphone recordings locally for manual playback.", ), - + livekit: bool = typer.Option( + False, "--livekit", help="Creates QR code for livekit server and token." + ), ): _run( server=server, @@ -76,6 +83,7 @@ def run( domain=domain, profiles=profiles, profile=profile, + livekit=livekit, ) @@ -93,6 +101,7 @@ def _run( domain = None, profiles = None, profile = None, + livekit: bool = False, ): profiles_dir = os.path.join(os.path.dirname(os.path.realpath(__file__)), "source", "server", "profiles") @@ -124,7 +133,7 @@ def _run( if not server_url: server_url = f"{server_host}:{server_port}" - if not server and not client: + if not server and not client and not livekit: server = True client = True @@ -191,6 +200,73 @@ def _run( client_thread = threading.Thread(target=module.main, args=[server_url, debug, play_audio]) client_thread.start() + if livekit: + def run_command(command): + subprocess.run(command, shell=True, check=True) + + def getToken(): + token = ( + api.AccessToken("devkey", "secret") + .with_identity("identity") + .with_name("my name") + .with_grants( + api.VideoGrants( + room_join=True, + room="my-room", + ) + ) + ) + return token.to_jwt() + + # Get local IP address + s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) + s.connect(("8.8.8.8", 80)) + ip_address = s.getsockname()[0] + s.close() + + # Create threads for each command and store handles + interpreter_thread = threading.Thread( + target=run_command, args=("poetry run interpreter --server",) + ) + livekit_thread = threading.Thread( + target=run_command, args=('livekit-server --dev --bind "0.0.0.0"',) + ) + worker_thread = threading.Thread( + target=run_command, args=("python worker.py dev",) + ) + + threads = [interpreter_thread, livekit_thread, worker_thread] + + # Start all threads and set up logging for thread completion + for thread in threads: + thread.start() + time.sleep(5) + + # Create QR code + url = f"ws://{ip_address}:7880" + token = getToken() + content = json.dumps({"livekit_server": url, "token": token}) + qr_code = segno.make(content) + qr_code.terminal(compact=True) + + print("Mobile setup complete. Scan the QR code to connect.") + + def signal_handler(sig, frame): + print("Termination signal received. Shutting down...") + for thread in threads: + if thread.is_alive(): + # This will only work if the subprocess uses shell=True and the OS is Unix-like + subprocess.run(f"pkill -P {os.getpid()}", shell=True) + os._exit(0) + + # Register the signal handler + signal.signal(signal.SIGINT, signal_handler) + signal.signal(signal.SIGTERM, signal_handler) + + # Wait for all threads to complete + for thread in threads: + thread.join() + try: if server: server_thread.join() diff --git a/software/worker.py b/software/worker.py new file mode 100644 index 0000000..12b19ce --- /dev/null +++ b/software/worker.py @@ -0,0 +1,76 @@ +import asyncio +import copy +import os + +from livekit.agents import AutoSubscribe, JobContext, WorkerOptions, cli +from livekit.agents.llm import ChatContext, ChatMessage +from livekit import rtc +from livekit.agents.voice_assistant import VoiceAssistant +from livekit.plugins import deepgram, openai, silero, elevenlabs +from dotenv import load_dotenv + +load_dotenv() + + +# This function is the entrypoint for the agent. +async def entrypoint(ctx: JobContext): + # Create an initial chat context with a system prompt + initial_ctx = ChatContext().append( + role="system", + text=( + "You are a voice assistant created by LiveKit. Your interface with users will be voice. " + "You should use short and concise responses, and avoiding usage of unpronounceable punctuation." + ), + ) + + # Connect to the LiveKit room + await ctx.connect(auto_subscribe=AutoSubscribe.AUDIO_ONLY) + + # VoiceAssistant is a class that creates a full conversational AI agent. + # See https://github.com/livekit/agents/blob/main/livekit-agents/livekit/agents/voice_assistant/assistant.py + # for details on how it works. + open_interpreter = openai.LLM( + model="open-interpreter", base_url="http://0.0.0.0:8000/openai" + ) + assistant = VoiceAssistant( + vad=silero.VAD.load(), # Voice Activity Detection + stt=deepgram.STT(), # Speech-to-Text + llm=open_interpreter, # Language Model + tts=elevenlabs.TTS(), # Text-to-Speech + chat_ctx=initial_ctx, # Chat history context + ) + + chat = rtc.ChatManager(ctx.room) + + async def _answer_from_text(text: str): + chat_ctx = copy.deepcopy(assistant._chat_ctx) + chat_ctx.messages.append(ChatMessage(role="user", content=text)) + + stream = open_interpreter.chat(chat_ctx=chat_ctx) + await assistant.say(stream) + + @chat.on("message_received") + def on_chat_received(msg: rtc.ChatMessage): + print("RECEIVED MESSAGE OMG!!!!!!!!!!") + print("RECEIVED MESSAGE OMG!!!!!!!!!!") + print("RECEIVED MESSAGE OMG!!!!!!!!!!") + print("RECEIVED MESSAGE OMG!!!!!!!!!!") + if not msg.message: + return + + asyncio.create_task(_answer_from_text(msg.message)) + + # Start the voice assistant with the LiveKit room + assistant.start(ctx.room) + + await asyncio.sleep(1) + + # Greets the user with an initial message + await assistant.say("Hey, how can I help you today?", allow_interruptions=True) + + +if __name__ == "__main__": + # Initialize the worker with the entrypoint + cli.run_app( + WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret") + ) From 0572d6e9f166f71289f026b910819a3a41c20eb2 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Wed, 14 Aug 2024 18:10:43 -0700 Subject: [PATCH 02/12] increase time between processes --- software/start.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/software/start.py b/software/start.py index 1a3c105..e4b9277 100644 --- a/software/start.py +++ b/software/start.py @@ -240,7 +240,7 @@ def _run( # Start all threads and set up logging for thread completion for thread in threads: thread.start() - time.sleep(5) + time.sleep(7) # Create QR code url = f"ws://{ip_address}:7880" From 8f08dc2f8a14e0b597651b502c21fc2c7758b566 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Wed, 14 Aug 2024 18:10:54 -0700 Subject: [PATCH 03/12] specify unused port --- software/worker.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/software/worker.py b/software/worker.py index 12b19ce..82b96c9 100644 --- a/software/worker.py +++ b/software/worker.py @@ -72,5 +72,5 @@ async def entrypoint(ctx: JobContext): if __name__ == "__main__": # Initialize the worker with the entrypoint cli.run_app( - WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret") + WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret", port=8082) ) From 7182f10c38547e4be49a3195d36c89cff762e45b Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 12:30:58 -0700 Subject: [PATCH 04/12] refactor domain into expose --- software/start.py | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/software/start.py b/software/start.py index e4b9277..27536cf 100644 --- a/software/start.py +++ b/software/start.py @@ -34,7 +34,11 @@ def run( tunnel_service: str = typer.Option( "ngrok", "--tunnel-service", help="Specify the tunnel service" ), - expose: bool = typer.Option(False, "--expose", help="Expose server to internet"), + expose: str = typer.Option( + '', + "--expose", + help="Expose localhost to internet with auth and optional custom domain if specified", + ), client: bool = typer.Option(False, "--client", help="Run client"), server_url: str = typer.Option( None, @@ -47,9 +51,6 @@ def run( qr: bool = typer.Option( False, "--qr", help="Display QR code to scan to connect to the server" ), - domain: str = typer.Option( - None, "--domain", help="Connect ngrok to a custom domain" - ), profiles: bool = typer.Option( False, "--profiles", @@ -80,7 +81,6 @@ def run( client_type=client_type, qr=qr, debug=debug, - domain=domain, profiles=profiles, profile=profile, livekit=livekit, @@ -92,13 +92,12 @@ def _run( server_host: str = "0.0.0.0", server_port: int = 10001, tunnel_service: str = "bore", - expose: bool = False, + expose: str = '', client: bool = False, server_url: str = None, client_type: str = "auto", qr: bool = False, debug: bool = False, - domain = None, profiles = None, profile = None, livekit: bool = False, @@ -165,7 +164,7 @@ def _run( if expose: tunnel_thread = threading.Thread( - target=create_tunnel, args=[tunnel_service, server_host, server_port, qr, domain] + target=create_tunnel, args=[tunnel_service, server_host, server_port, qr] ) tunnel_thread.start() From 278eee820b03b7011bd6ae8846a9cf27ead7076d Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 12:48:24 -0700 Subject: [PATCH 05/12] amend logic error in expose --- software/start.py | 17 ++++++++++++----- 1 file changed, 12 insertions(+), 5 deletions(-) diff --git a/software/start.py b/software/start.py index 27536cf..88f56cb 100644 --- a/software/start.py +++ b/software/start.py @@ -35,9 +35,9 @@ def run( "ngrok", "--tunnel-service", help="Specify the tunnel service" ), expose: str = typer.Option( - '', + None, "--expose", - help="Expose localhost to internet with auth and optional custom domain if specified", + help="Expose localhost to internet. Optionally specify a custom domain.", ), client: bool = typer.Option(False, "--client", help="Run client"), server_url: str = typer.Option( @@ -92,7 +92,7 @@ def _run( server_host: str = "0.0.0.0", server_port: int = 10001, tunnel_service: str = "bore", - expose: str = '', + expose: str = None, client: bool = False, server_url: str = None, client_type: str = "auto", @@ -162,9 +162,16 @@ def _run( ) server_thread.start() - if expose: + if expose is not None: + # `--expose` was passed + if expose == '': + # `--expose` was called without a custom domain + # Autogenerate domain with ngrok + expose = "unspecified" + # else, `--expose was passed a custom domain` + tunnel_thread = threading.Thread( - target=create_tunnel, args=[tunnel_service, server_host, server_port, qr] + target=create_tunnel, args=[tunnel_service, server_host, server_port, qr, expose] ) tunnel_thread.start() From 6cfa5b98aed23abc08aa056a8e220de8c7a65bea Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 14:52:08 -0700 Subject: [PATCH 06/12] Revert "specify unused port" This reverts commit 8f08dc2f8a14e0b597651b502c21fc2c7758b566. --- software/worker.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/software/worker.py b/software/worker.py index 82b96c9..12b19ce 100644 --- a/software/worker.py +++ b/software/worker.py @@ -72,5 +72,5 @@ async def entrypoint(ctx: JobContext): if __name__ == "__main__": # Initialize the worker with the entrypoint cli.run_app( - WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret", port=8082) + WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret") ) From 1a26f3ec7bb260acceb781f28c31219d32ce1e87 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 14:52:54 -0700 Subject: [PATCH 07/12] Reapply "specify unused port" This reverts commit 6cfa5b98aed23abc08aa056a8e220de8c7a65bea. --- software/worker.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/software/worker.py b/software/worker.py index 12b19ce..82b96c9 100644 --- a/software/worker.py +++ b/software/worker.py @@ -72,5 +72,5 @@ async def entrypoint(ctx: JobContext): if __name__ == "__main__": # Initialize the worker with the entrypoint cli.run_app( - WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret") + WorkerOptions(entrypoint_fnc=entrypoint, api_key="devkey", api_secret="secret", port=8082) ) From 1398afc4a11b583c5802883ac9d1d035f5aca992 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 14:53:25 -0700 Subject: [PATCH 08/12] Revert "amend logic error in expose" This reverts commit 278eee820b03b7011bd6ae8846a9cf27ead7076d. --- software/start.py | 17 +++++------------ 1 file changed, 5 insertions(+), 12 deletions(-) diff --git a/software/start.py b/software/start.py index 88f56cb..27536cf 100644 --- a/software/start.py +++ b/software/start.py @@ -35,9 +35,9 @@ def run( "ngrok", "--tunnel-service", help="Specify the tunnel service" ), expose: str = typer.Option( - None, + '', "--expose", - help="Expose localhost to internet. Optionally specify a custom domain.", + help="Expose localhost to internet with auth and optional custom domain if specified", ), client: bool = typer.Option(False, "--client", help="Run client"), server_url: str = typer.Option( @@ -92,7 +92,7 @@ def _run( server_host: str = "0.0.0.0", server_port: int = 10001, tunnel_service: str = "bore", - expose: str = None, + expose: str = '', client: bool = False, server_url: str = None, client_type: str = "auto", @@ -162,16 +162,9 @@ def _run( ) server_thread.start() - if expose is not None: - # `--expose` was passed - if expose == '': - # `--expose` was called without a custom domain - # Autogenerate domain with ngrok - expose = "unspecified" - # else, `--expose was passed a custom domain` - + if expose: tunnel_thread = threading.Thread( - target=create_tunnel, args=[tunnel_service, server_host, server_port, qr, expose] + target=create_tunnel, args=[tunnel_service, server_host, server_port, qr] ) tunnel_thread.start() From 109a028ee36ca812ced0ac237515c726b259b8a4 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 14:53:35 -0700 Subject: [PATCH 09/12] Revert "refactor domain into expose" This reverts commit 7182f10c38547e4be49a3195d36c89cff762e45b. --- software/start.py | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/software/start.py b/software/start.py index 27536cf..e4b9277 100644 --- a/software/start.py +++ b/software/start.py @@ -34,11 +34,7 @@ def run( tunnel_service: str = typer.Option( "ngrok", "--tunnel-service", help="Specify the tunnel service" ), - expose: str = typer.Option( - '', - "--expose", - help="Expose localhost to internet with auth and optional custom domain if specified", - ), + expose: bool = typer.Option(False, "--expose", help="Expose server to internet"), client: bool = typer.Option(False, "--client", help="Run client"), server_url: str = typer.Option( None, @@ -51,6 +47,9 @@ def run( qr: bool = typer.Option( False, "--qr", help="Display QR code to scan to connect to the server" ), + domain: str = typer.Option( + None, "--domain", help="Connect ngrok to a custom domain" + ), profiles: bool = typer.Option( False, "--profiles", @@ -81,6 +80,7 @@ def run( client_type=client_type, qr=qr, debug=debug, + domain=domain, profiles=profiles, profile=profile, livekit=livekit, @@ -92,12 +92,13 @@ def _run( server_host: str = "0.0.0.0", server_port: int = 10001, tunnel_service: str = "bore", - expose: str = '', + expose: bool = False, client: bool = False, server_url: str = None, client_type: str = "auto", qr: bool = False, debug: bool = False, + domain = None, profiles = None, profile = None, livekit: bool = False, @@ -164,7 +165,7 @@ def _run( if expose: tunnel_thread = threading.Thread( - target=create_tunnel, args=[tunnel_service, server_host, server_port, qr] + target=create_tunnel, args=[tunnel_service, server_host, server_port, qr, domain] ) tunnel_thread.start() From a595bdf570679a9d6e93a6fa1b0d65758b37a6f6 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 15:18:56 -0700 Subject: [PATCH 10/12] consolidate tunneling to ngrok service --- software/source/server/tunnel.py | 150 ++++--------------------------- 1 file changed, 19 insertions(+), 131 deletions(-) diff --git a/software/source/server/tunnel.py b/software/source/server/tunnel.py index a40c0f3..a749e07 100644 --- a/software/source/server/tunnel.py +++ b/software/source/server/tunnel.py @@ -1,142 +1,30 @@ -import subprocess -import re +import ngrok import pyqrcode -import time from ..utils.print_markdown import print_markdown - def create_tunnel( - tunnel_method="ngrok", server_host="localhost", server_port=10001, qr=False, domain=None -): - print_markdown("Exposing server to the internet...") - - server_url = "" - if tunnel_method == "bore": - try: - output = subprocess.check_output("command -v bore", shell=True) - except subprocess.CalledProcessError: - print( - "The bore-cli command is not available. Please run 'cargo install bore-cli'." - ) - print("For more information, see https://github.com/ekzhang/bore") - exit(1) - - time.sleep(6) - # output = subprocess.check_output(f'bore local {server_port} --to bore.pub', shell=True) - process = subprocess.Popen( - f"bore local {server_port} --to bore.pub", - shell=True, - stdout=subprocess.PIPE, - stderr=subprocess.STDOUT, - universal_newlines=True, - ) - - while True: - line = process.stdout.readline() - print(line) - if not line: - break - if "listening at bore.pub:" in line: - remote_port = re.search("bore.pub:([0-9]*)", line).group(1) - server_url = f"bore.pub:{remote_port}" - print_markdown( - f"Your server is being hosted at the following URL: bore.pub:{remote_port}" - ) - break - - elif tunnel_method == "localtunnel": - if subprocess.call("command -v lt", shell=True): - print("The 'lt' command is not available.") - print( - "Please ensure you have Node.js installed, then run 'npm install -g localtunnel'." - ) - print( - "For more information, see https://github.com/localtunnel/localtunnel" - ) - exit(1) - else: - process = subprocess.Popen( - f"npx localtunnel --port {server_port}", - shell=True, - stdout=subprocess.PIPE, - stderr=subprocess.STDOUT, - universal_newlines=True, - ) + server_host="localhost", server_port=10001, qr=False, domain=None +): + """ + To use most of ngrok’s features, you’ll need an authtoken. To obtain one, sign up for free at ngrok.com and + retrieve it from the authtoken page in your ngrok dashboard. - found_url = False - url_pattern = re.compile(r"your url is: https://[a-zA-Z0-9.-]+") + https://dashboard.ngrok.com/get-started/your-authtoken - while True: - line = process.stdout.readline() - if not line: - break # Break out of the loop if no more output - match = url_pattern.search(line) - if match: - found_url = True - remote_url = match.group(0).replace("your url is: ", "") - server_url = remote_url - print( - f"\nYour server is being hosted at the following URL: {remote_url}" - ) - break # Exit the loop once the URL is found - - if not found_url: - print( - "Failed to extract the localtunnel URL. Please check localtunnel's output for details." - ) - - elif tunnel_method == "ngrok": - # Check if ngrok is installed - is_installed = ( - subprocess.check_output("command -v ngrok", shell=True).decode().strip() - ) - if not is_installed: - print("The ngrok command is not available.") - print( - "Please install ngrok using the instructions at https://ngrok.com/docs/getting-started/" - ) - exit(1) - - # If ngrok is installed, start it on the specified port - # process = subprocess.Popen(f'ngrok http {server_port} --log=stdout', shell=True, stdout=subprocess.PIPE) - - if domain: - domain = f"--domain={domain}" - else: - domain = "" - process = subprocess.Popen( - f"ngrok http {server_port} --scheme http,https {domain} --log=stdout", - shell=True, - stdout=subprocess.PIPE, - ) - - # Initially, no URL is found - found_url = False - # Regular expression to match the ngrok URL - url_pattern = re.compile(r"https://[a-zA-Z0-9-]+\.ngrok(-free)?\.app") + You can set it as `NGROK_AUTHTOKEN` in your environment variables + """ + print_markdown("Exposing server to the internet...") - # Read the output line by line - while True: - line = process.stdout.readline().decode("utf-8") - if not line: - break # Break out of the loop if no more output - match = url_pattern.search(line) - if match: - found_url = True - remote_url = match.group(0) - server_url = remote_url - print( - f"\nYour server is being hosted at the following URL: {remote_url}" - ) - break # Exit the loop once the URL is found + if domain: + listener = ngrok.forward(f"{server_host}:{server_port}", authtoken_from_env=True, domain=domain) + else: + listener = ngrok.forward(f"{server_host}:{server_port}", authtoken_from_env=True) - if not found_url: - print( - "Failed to extract the ngrok tunnel URL. Please check ngrok's output for details." - ) + listener_url = listener.url() - if server_url and qr: - text = pyqrcode.create(remote_url) + print(f"Ingress established at: {listener_url}"); + if listener_url and qr: + text = pyqrcode.create(listener_url) print(text.terminal(quiet_zone=1)) - return server_url + return listener_url From d9ee20efaada91cc076a2d12eac7e6a2ba288a68 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 15:19:16 -0700 Subject: [PATCH 11/12] support ngrok tunnels on livekit --- software/start.py | 45 ++++++++++++++++++++++++++++----------------- 1 file changed, 28 insertions(+), 17 deletions(-) diff --git a/software/start.py b/software/start.py index e4b9277..9247e4c 100644 --- a/software/start.py +++ b/software/start.py @@ -1,5 +1,5 @@ import typer -import asyncio +import ngrok import platform import threading import os @@ -12,11 +12,13 @@ import socket import json import segno import time +from dotenv import load_dotenv import signal app = typer.Typer() +load_dotenv() @app.command() def run( @@ -31,9 +33,6 @@ def run( "--server-port", help="Specify the server port where the server will deploy", ), - tunnel_service: str = typer.Option( - "ngrok", "--tunnel-service", help="Specify the tunnel service" - ), expose: bool = typer.Option(False, "--expose", help="Expose server to internet"), client: bool = typer.Option(False, "--client", help="Run client"), server_url: str = typer.Option( @@ -73,7 +72,6 @@ def run( server=server, server_host=server_host, server_port=server_port, - tunnel_service=tunnel_service, expose=expose, client=client, server_url=server_url, @@ -91,7 +89,6 @@ def _run( server: bool = False, server_host: str = "0.0.0.0", server_port: int = 10001, - tunnel_service: str = "bore", expose: bool = False, client: bool = False, server_url: str = None, @@ -163,9 +160,9 @@ def _run( ) server_thread.start() - if expose: + if expose and not livekit: tunnel_thread = threading.Thread( - target=create_tunnel, args=[tunnel_service, server_host, server_port, qr, domain] + target=create_tunnel, args=[server_host, server_port, qr, domain] ) tunnel_thread.start() @@ -218,12 +215,6 @@ def _run( ) return token.to_jwt() - # Get local IP address - s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) - s.connect(("8.8.8.8", 80)) - ip_address = s.getsockname()[0] - s.close() - # Create threads for each command and store handles interpreter_thread = threading.Thread( target=run_command, args=("poetry run interpreter --server",) @@ -242,10 +233,30 @@ def _run( thread.start() time.sleep(7) - # Create QR code - url = f"ws://{ip_address}:7880" token = getToken() - content = json.dumps({"livekit_server": url, "token": token}) + + # Create QR code + if expose and domain: + listener = ngrok.forward("localhost:7880", authtoken_from_env=True, domain=domain) + url= listener.url() + print(url) + content = json.dumps({"livekit_server": url, "token": token}) + elif expose and not domain: + listener = ngrok.forward("localhost:7880", authtoken_from_env=True) + url= listener.url() + print(url) + content = json.dumps({"livekit_server": url, "token": token}) + else: + # Get local IP address + s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) + s.connect(("8.8.8.8", 80)) + ip_address = s.getsockname()[0] + s.close() + + url = f"ws://{ip_address}:7880" + print(url) + content = json.dumps({"livekit_server": url, "token": token}) + qr_code = segno.make(content) qr_code.terminal(compact=True) From bb98b27040ac4e5fa77ae554b4a201ce18fd1ea9 Mon Sep 17 00:00:00 2001 From: Ben Xu Date: Thu, 15 Aug 2024 16:19:29 -0700 Subject: [PATCH 12/12] remove token from QR code --- software/start.py | 22 +++------------------- 1 file changed, 3 insertions(+), 19 deletions(-) diff --git a/software/start.py b/software/start.py index 9247e4c..3b54766 100644 --- a/software/start.py +++ b/software/start.py @@ -201,20 +201,6 @@ def _run( def run_command(command): subprocess.run(command, shell=True, check=True) - def getToken(): - token = ( - api.AccessToken("devkey", "secret") - .with_identity("identity") - .with_name("my name") - .with_grants( - api.VideoGrants( - room_join=True, - room="my-room", - ) - ) - ) - return token.to_jwt() - # Create threads for each command and store handles interpreter_thread = threading.Thread( target=run_command, args=("poetry run interpreter --server",) @@ -233,19 +219,17 @@ def _run( thread.start() time.sleep(7) - token = getToken() - # Create QR code if expose and domain: listener = ngrok.forward("localhost:7880", authtoken_from_env=True, domain=domain) url= listener.url() print(url) - content = json.dumps({"livekit_server": url, "token": token}) + content = json.dumps({"livekit_server": url}) elif expose and not domain: listener = ngrok.forward("localhost:7880", authtoken_from_env=True) url= listener.url() print(url) - content = json.dumps({"livekit_server": url, "token": token}) + content = json.dumps({"livekit_server": url}) else: # Get local IP address s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) @@ -255,7 +239,7 @@ def _run( url = f"ws://{ip_address}:7880" print(url) - content = json.dumps({"livekit_server": url, "token": token}) + content = json.dumps({"livekit_server": url}) qr_code = segno.make(content) qr_code.terminal(compact=True)