file / app.py
Kanhshsh's picture
Update app.py
3b4e567 verified
import gradio as gr
import subprocess
import asyncio
import threading
from queue import Queue
import os
import shlex
import signal
import glob
from telegram import Update
from telegram.ext import ApplicationBuilder, CommandHandler, ContextTypes
BOT_TOKEN = ""
log_queue = Queue()
MAX_LOGS = 20000
terminal_logs = []
bot_logs = []
bot_app = None
bot_running = False
bot_thread = None
current_dir = os.getcwd()
current_proc = None # Track running process
LOG_FILE = "terminal_logs.txt"
# Load previous logs if available
if os.path.exists(LOG_FILE):
with open(LOG_FILE, "r") as f:
terminal_logs = f.read().splitlines()[-MAX_LOGS:]
# Logging
def save_log_to_file(entry):
with open(LOG_FILE, "a") as f:
f.write(entry + "\n")
def add_terminal_log(entry):
terminal_logs.append(entry)
save_log_to_file(entry)
if len(terminal_logs) > MAX_LOGS:
terminal_logs.pop(0)
def add_bot_log(entry):
bot_logs.append(entry)
if len(bot_logs) > MAX_LOGS:
bot_logs.pop(0)
# Command Execution
def execute_command(cmd):
global current_dir, current_proc
output = []
add_terminal_log(f"$ {cmd}")
if cmd.strip() == "ctrl+c":
if current_proc:
current_proc.send_signal(signal.SIGINT)
msg = "[Info] Sent Ctrl+C to running process."
add_terminal_log(msg)
return msg
else:
msg = "[Info] No running process to interrupt."
add_terminal_log(msg)
return msg
if cmd.strip().startswith("cd"):
parts = shlex.split(cmd)
if len(parts) > 1:
pattern = os.path.join(current_dir, parts[1])
matches = glob.glob(pattern)
if matches:
new_path = os.path.abspath(matches[0])
if os.path.isdir(new_path):
current_dir = new_path
msg = f"Changed directory to {current_dir}"
output.append(msg)
add_terminal_log(msg)
else:
msg = f"Not a directory: {new_path}"
output.append(msg)
add_terminal_log(msg)
else:
msg = f"No matching path for: {parts[1]}"
output.append(msg)
add_terminal_log(msg)
else:
msg = "Usage: cd <path>"
output.append(msg)
add_terminal_log(msg)
else:
try:
current_proc = subprocess.Popen(
cmd,
cwd=current_dir,
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT,
text=True,
bufsize=1
)
for line in iter(current_proc.stdout.readline, ''):
line = line.strip()
if line:
output.append(line)
add_terminal_log(line)
current_proc.stdout.close()
current_proc.wait()
except Exception as e:
error_msg = f"[Error] {e}"
output.append(error_msg)
add_terminal_log(error_msg)
finally:
current_proc = None
return "\n".join(output)
# Telegram Bot
async def start_command(update: Update, context: ContextTypes.DEFAULT_TYPE):
await update.message.reply_text("Hello! I'm your terminal bot.\nUse /bash <command> to execute shell commands.")
add_bot_log("[Bot] Received /start command.")
async def bash_command(update: Update, context: ContextTypes.DEFAULT_TYPE):
cmd = ' '.join(context.args)
if not cmd:
await update.message.reply_text("Usage: /bash <command>")
return
add_bot_log(f"[Bot] $ {cmd}")
try:
result = execute_command(cmd)
except Exception as e:
result = f"Error: {e}"
add_bot_log(result)
await update.message.reply_text(f"$ {cmd}\n{result}")
def start_bot():
global bot_app, bot_running, bot_thread, BOT_TOKEN
if bot_running or not BOT_TOKEN:
add_bot_log("[Bot] Already running or token missing.")
return
async def run_bot():
global bot_app, bot_running
add_bot_log("[Bot] Starting bot...")
bot_app = ApplicationBuilder().token(BOT_TOKEN).build()
bot_app.add_handler(CommandHandler("start", start_command))
bot_app.add_handler(CommandHandler("bash", bash_command))
bot_running = True
await bot_app.run_polling()
add_bot_log("[Bot] Bot stopped.")
bot_running = False
def runner():
try:
asyncio.run(run_bot())
except Exception as e:
add_bot_log(f"[Bot Error] {e}")
global bot_running
bot_running = False
bot_thread = threading.Thread(target=runner, daemon=True)
bot_thread.start()
add_bot_log("[Bot] Bot thread started.")
def stop_bot():
global bot_app, bot_running
if not bot_running or not bot_app:
add_bot_log("[Bot] Bot not running.")
return
async def shutdown():
await bot_app.shutdown()
await bot_app.stop()
add_bot_log("[Bot] Bot stopped.")
global bot_running
bot_running = False
threading.Thread(target=lambda: asyncio.run(shutdown()), daemon=True).start()
# UI Hooks
def update_bot_logs():
return "\n".join(bot_logs[-100:])
def update_terminal_logs():
return "\n".join(terminal_logs[-100:])
def live_terminal(cmd):
result = [f"$ {cmd}"]
try:
result.append(execute_command(cmd))
except Exception as e:
result.append(f"[Error] {e}")
result.append(f"$ {current_dir} >")
return "\n".join(result)
# Gradio UI
with gr.Blocks() as demo:
with gr.Tab("πŸ’» Terminal"):
gr.Markdown("## πŸ–₯️ Interactive Terminal")
with gr.Row():
terminal_output = gr.Textbox(label="πŸ“Ÿ Terminal Output", lines=25, interactive=False)
with gr.Row():
cmd_input = gr.Textbox(placeholder="Enter shell command (use 'ctrl+c' to cancel)", label="Command Input")
run_btn = gr.Button("▢️ Run Command")
run_btn.click(fn=live_terminal, inputs=cmd_input, outputs=terminal_output)
demo.load(fn=update_terminal_logs, outputs=terminal_output)
with gr.Tab("πŸ€– Telegram Bot"):
gr.Markdown("## πŸ€– Telegram Bot Controls")
with gr.Row():
bot_output = gr.Textbox(label="πŸ€– Telegram Bot Logs", lines=25, interactive=False)
with gr.Row():
token_box = gr.Textbox(label="Bot Token", placeholder="Enter Telegram Bot Token")
with gr.Row():
start_btn = gr.Button("πŸš€ Start Telegram Bot")
stop_btn = gr.Button("πŸ›‘ Stop Telegram Bot")
def set_token_and_start(token):
global BOT_TOKEN
BOT_TOKEN = token
start_bot()
return update_bot_logs()
start_btn.click(fn=set_token_and_start, inputs=token_box, outputs=bot_output)
def handle_stop():
stop_bot()
return update_bot_logs()
stop_btn.click(fn=handle_stop, outputs=bot_output)
demo.load(fn=update_bot_logs, outputs=bot_output)
# Launch App
demo.launch()