import sys from gradio_base import UIHelper, WebUI import os from gradio_base import WebUI, UIHelper, PORT, HOST, Client from gradio_config import GradioConfig as gc from typing import List, Tuple, Any import gradio as gr import time class DebateUI(WebUI): FORMAT = "{}\n\n{}\nAffirmative viewpoint:{}\nNegative viewpoint:{}\n{}" AUDIENCE = "Audience" cache = {} all_agents_name = [] receive_server = None @classmethod def extract(cls, content): topic = content.split("")[1].split("Affirmative viewpoint:")[0] positive = content.split("")[1].split("Affirmative viewpoint:")[1].split("negative viewpoint:")[0] negative = content.split("")[1].split("Affirmative viewpoint:")[1].split("negative viewpoint:")[1] return topic.strip(), positive.strip(), negative.strip() @classmethod def merge(cls, theme, positive, negative, origin_content) -> str: return cls.FORMAT.format( origin_content.split("")[0], theme, positive, negative, origin_content.split("")[-1] ) @classmethod def convert2list4agentname(cls, sop): only_name = [] agent_name = [] roles_to_names = sop.roles_to_names for state_name,roles_names in roles_to_names.items(): for role,name in roles_names.items(): agent_name.append(f"{name}({role})") only_name.append(name) agent_name.append(cls.AUDIENCE) agent_name = list(set(agent_name)) agent_name.sort() return agent_name, only_name def render_and_register_ui(self): gc.add_agent(self.cache["only_name"]) def __init__( self, client_cmd: list, socket_host: str = HOST, socket_port: int = PORT, bufsize: int = 1024, ui_name: str = "DebateUI" ): super(DebateUI, self).__init__(client_cmd, socket_host, socket_port, bufsize, ui_name) self.first_recieve_from_client() self.data_history = list() self.caller = 0 def handle_message(self, history:list, state, agent_name, token, node_name): if state % 10 == 0: self.data_history.append({agent_name: token}) elif state % 10 == 1: # Same state. Need to add new bubble in same bubble. self.data_history[-1][agent_name] += token elif state % 10 == 2: # New state. Need to add new bubble. history.append([None, ""]) self.data_history.clear() self.data_history.append({agent_name: token}) else: assert False, "Invalid state." render_data = self.render_bubble(history, self.data_history, node_name, render_node_name= True or state % 10 == 2) return render_data def start_button_when_click(self, theme, positive, negative, choose, mode): """ inputs=[self.text_theme, self.text_positive, self.text_negative, self.radio_choose], outputs=[self.chatbot, self.btn_send] """ cosplay = None if choose == self.AUDIENCE else choose.split("(")[0] message = dict(theme=theme, positive=positive, negative=negative, cosplay=cosplay, mode=mode) self.send_start_cmd(message=message) return gr.Chatbot.update( visible=True ), gr.Button.update(visible=False) def start_button_after_click(self, history): """ inputs=[self.chatbot], outputs=[self.chatbot, self.text_user, self.btn_send, self.btn_reset, self.btn_next] """ if self.caller == 0: # not single mode self.data_history = list() self.caller = 0 receive_server = self.receive_server while True: data_list: List = receive_server.send(None) for item in data_list: data = eval(item) assert isinstance(data, list) state, agent_name, token, node_name = data assert isinstance(state, int) if state == 30: # user input yield history,\ gr.Textbox.update(visible=True, interactive=True), \ gr.Button.update(visible=True, interactive=True),\ gr.Button.update(visible=True, interactive=True),\ gr.Button.update(visible=False) return elif state == 99: # finish yield history, gr.Textbox.update(visible=True, interactive=False, value="finish!"), \ gr.Button.update(visible=True, interactive=False, value="finish!"), gr.Button.update(visible=True, interactive=True),\ gr.Button.update(visible=False) elif state == 98: yield history, \ gr.Textbox.update(visible=False, interactive=False), \ gr.Button.update(visible=False, interactive=False),\ gr.Button.update(visible=False, interactive=False),\ gr.Button.update(visible=True, value=f"Next Agent: 🤖{agent_name} | Next Node: ⭕{node_name}") return else: history = self.handle_message(history, state, agent_name, token, node_name) yield history, \ gr.Textbox.update(visible=False, interactive=False), \ gr.Button.update(visible=False, interactive=False),\ gr.Button.update(visible=False, interactive=False),\ gr.Button.update(visible=False) def send_button_when_click(self, text_user, history:list): """ inputs=[self.text_user, self.chatbot], outputs=[self.text_user, self.btn_send, self.chatbot] """ history.append( [UIHelper.wrap_css(text_user, "User"), None] ) # print(f"server: send {text_user} to client") self.send_message(""+text_user+self.SIGN["SPLIT"]) return gr.Textbox.update(value="", visible=False),\ gr.Button.update(visible=False), \ history,\ gr.Button.update(visible=False) def reset_button_when_click(self, history, text_positive, text_negative, text_theme, text_user, btn_send, btn_start, btn_reset): """ self.chatbot, self.text_positive, self.text_negative, self.text_theme, self.text_user, self.btn_send, self.btn_start, self.btn_reset self.btn_next """ self.caller = 0 return None, \ "", \ "", \ "", \ "", \ gr.Button.update(value="Restarting...", interactive=False, visible=True),\ gr.Button.update(value="Restarting...", interactive=False, visible=True),\ gr.Button.update(value="Restarting...", interactive=False, visible=True),\ gr.Button.update(value="Restarting...", interactive=False, visible=False) def reset_button_after_click(self, history, text_positive, text_negative, text_theme, text_user, btn_send, btn_start, btn_reset): self.reset() self.first_recieve_from_client(reset_mode=True) return gr.Chatbot.update(value=None, visible=False),\ gr.Textbox.update(value=f"{self.cache['positive']}", interactive=True, visible=True),\ gr.Textbox.update(value=f"{self.cache['negative']}", interactive=True, visible=True),\ gr.Textbox.update(value=f"{self.cache['theme']}", interactive=True, visible=True),\ gr.Textbox.update(value=f"", interactive=True, visible=False),\ gr.Button.update(interactive=True, visible=False, value="Send"),\ gr.Button.update(interactive=True, visible=True, value="Start"),\ gr.Button.update(interactive=False, visible=False, value="Restart"),\ gr.Button.update(interactive=True, visible=False, value="Next Agent") def btn_next_when_click(self): yield gr.Button.update(visible=False) self.send_message("nothing") self.caller = 1 # will note clear the self.data_history time.sleep(0.5) return def construct_ui( self, theme:str=None, positive:str=None, negative:str=None, agents_name:List=None, default_cos_play_id:int=None ): theme = self.cache["theme"] if theme is None else theme positive = self.cache["positive"] if positive is None else positive negative = self.cache["negative"] if negative is None else negative agents_name = self.cache["agents_name"] if agents_name is None else agents_name default_cos_play_id = self.cache["default_cos_play_id"] if default_cos_play_id is None else default_cos_play_id with gr.Blocks(css=gc.CSS) as demo: with gr.Row(): with gr.Column(): self.radio_mode = gr.Radio( [Client.AUTO_MODE, Client.SINGLE_MODE], value=Client.AUTO_MODE, interactive=True, label = Client.MODE_LABEL, info = Client.MODE_INFO ) self.text_theme = gr.Textbox( label="Debate Topic:", value=theme, placeholder="Please input the Debate Topic" ) self.text_positive = gr.Textbox( label="Affirmative viewpoint:", value=positive, placeholder="Please input the Affirmative viewpoint" ) self.text_negative = gr.Textbox( label="Negative viewpoint:", value=negative, placeholder="Please input the Negative viewpoint" ) self.radio_choose = gr.Radio( agents_name, value=agents_name[default_cos_play_id], label="User'agent", interactive=True ) self.btn_start = gr.Button( value="run" ) VISIBLE = False with gr.Column(): self.chatbot = gr.Chatbot( height= 650, elem_id="chatbot1", label="Dialog", visible=VISIBLE ) self.btn_next = gr.Button( value="Next Agent Start", visible=False ) self.text_user = gr.Textbox( label="Input", placeholder="Input here", visible=VISIBLE ) self.btn_send = gr.Button( value="Send", visible=VISIBLE ) self.btn_reset = gr.Button( value="Restart", visible=VISIBLE ) self.btn_start.click( fn=self.start_button_when_click, inputs=[self.text_theme, self.text_positive, self.text_negative, self.radio_choose, self.radio_mode], outputs=[self.chatbot, self.btn_start] ).then( fn=self.start_button_after_click, inputs=[self.chatbot], outputs=[self.chatbot, self.text_user, self.btn_send, self.btn_reset, self.btn_next] ) self.btn_send.click( fn=self.send_button_when_click, inputs=[self.text_user, self.chatbot], outputs=[self.text_user, self.btn_send, self.chatbot, self.btn_reset] ).then( fn=self.start_button_after_click, inputs=[self.chatbot], outputs=[self.chatbot, self.text_user, self.btn_send, self.btn_reset, self.btn_next] ) self.btn_reset.click( fn=self.reset_button_when_click, inputs=[ self.chatbot, self.text_positive, self.text_negative, self.text_theme, self.text_user, self.btn_send, self.btn_start, self.btn_reset ], outputs=[ self.chatbot, self.text_positive, self.text_negative, self.text_theme, self.text_user, self.btn_send, self.btn_start, self.btn_reset, self.btn_next ] ).then( fn=self.reset_button_after_click, inputs=[ self.chatbot, self.text_positive, self.text_negative, self.text_theme, self.text_user, self.btn_send, self.btn_start, self.btn_reset ], outputs=[ self.chatbot, self.text_positive, self.text_negative, self.text_theme, self.text_user, self.btn_send, self.btn_start, self.btn_reset, self.btn_next ] ) self.btn_next.click( fn=self.btn_next_when_click, inputs=[], outputs=[self.btn_next] ).then( fn=self.start_button_after_click, inputs=[self.chatbot], outputs=[self.chatbot, self.text_user, self.btn_send, self.btn_reset, self.btn_next] ) self.demo = demo if __name__ == '__main__': ui = DebateUI(client_cmd=["python","gradio_backend.py"]) ui.construct_ui() ui.run()