Kudasai / webgui.py
Bikatr7's picture
added everything
4bc65dc verified
raw
history blame
132 kB
## built-in libraries
import typing
import base64
## third-party libraries
import gradio as gr
from kairyou import Indexer
from kairyou import Kairyou
from kairyou import InvalidReplacementJsonKeys
from kairyou.util import _validate_replacement_json
from easytl import EasyTL, ALLOWED_GEMINI_MODELS, ALLOWED_OPENAI_MODELS
## custom modules
from modules.common.toolkit import Toolkit
from modules.common.file_ensurer import FileEnsurer
from modules.gui.gui_file_util import gui_get_text_from_file, gui_get_json_from_file
from modules.gui.gui_json_util import GuiJsonUtil
from handlers.json_handler import JsonHandler
from modules.common.translator import Translator
from kudasai import Kudasai
##-------------------start-of-KudasaiGUI---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
class KudasaiGUI:
"""
KudasaiGUI is a class that contains the GUI for Kudasai.
"""
## scary javascript code that allows us to save textbox contents to a file
with open(FileEnsurer.js_save_to_file_path, 'r', encoding='utf-8') as f:
js_save_to_file = f.read()
## used for whether the debug log tab for Translator should be actively refreshing based of Logger.current_batch
is_translation_ongoing = False
with open(FileEnsurer.translation_settings_description_path, 'r', encoding='utf-8') as f:
lines = f.readlines()
description_dict = {
"prompt_assembly_mode": lines[4-1].strip(),
"number_of_lines_per_batch": lines[6-1].strip(),
"sentence_fragmenter_mode": lines[8-1].strip(),
"je_check_mode": lines[10-1].strip(),
"number_of_malformed_batch_retries": lines[12-1].strip(),
"batch_retry_timeout": lines[14-1].strip(),
"number_of_concurrent_batches": lines[16-1].strip(),
"openai_help_link": lines[19-1].strip(),
"openai_model": lines[21-1].strip(),
"openai_system_message": lines[23-1].strip(),
"openai_temperature": lines[25-1].strip(),
"openai_top_p": lines[27-1].strip(),
"openai_n": lines[29-1].strip(),
"openai_stream": lines[31-1].strip(),
"openai_stop": lines[33-1].strip(),
"openai_logit_bias": lines[35-1].strip(),
"openai_max_tokens": lines[37-1].strip(),
"openai_presence_penalty": lines[39-1].strip(),
"openai_frequency_penalty": lines[41-1].strip(),
"openai_disclaimer": lines[43-1].strip(),
"gemini_help_link": lines[46-1].strip(),
"gemini_model": lines[48-1].strip(),
"gemini_prompt": lines[50-1].strip(),
"gemini_temperature": lines[52-1].strip(),
"gemini_top_p": lines[54-1].strip(),
"gemini_top_k": lines[56-1].strip(),
"gemini_candidate_count": lines[58-1].strip(),
"gemini_stream": lines[60-1].strip(),
"gemini_stop_sequences": lines[62-1].strip(),
"gemini_max_output_tokens": lines[64-1].strip(),
"gemini_disclaimer": lines[66-1].strip(),
"deepl_help_link": lines[69-1].strip(),
"deepl_context": lines[71-1].strip(),
"deepl_split_sentences": lines[73-1].strip(),
"deepl_preserve_formatting": lines[75-1].strip(),
"deepl_formality": lines[77-1].strip(),
}
##-------------------start-of-build_gui()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def build_gui(self) -> None:
"""
Builds the GUI.
"""
with gr.Blocks(title="Kudasai") as self.gui:
##-------------------start-of-Utility-Functions---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
##-------------------start-of-fetch_log_content()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def fetch_log_content() -> str:
"""
Fetches the log content from the current log batch.
Returns:
log_text (str) : The log text.
"""
if(self.is_translation_ongoing == False):
return "No translation ongoing"
with open(FileEnsurer.debug_log_path, 'r', encoding='utf-8') as f:
log_text = f.read()
return log_text
##-------------------start-of-get_saved_api_key()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
@staticmethod
def get_saved_api_key(service_name:typing.Literal["openai","gemini","deepl"]) -> str:
"""
Gets the saved api key from the config folder, if it exists.
Parameters:
service_name (str): The name of the service (e.g., "deepl", "openai", "gemini").
Returns:
api_key (str) : The api key.
"""
service_to_path = {
"openai": FileEnsurer.openai_api_key_path,
"gemini": FileEnsurer.gemini_api_key_path,
"deepl": FileEnsurer.deepl_api_key_path
}
api_key_path = service_to_path.get(service_name, "")
try:
## Api key is encoded in base 64 so we need to decode it before returning
return base64.b64decode(FileEnsurer.standard_read_file(api_key_path).encode('utf-8')).decode('utf-8')
except:
return ""
##-------------------start-of-set_translator_api_key()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
async def set_translator_api_key(api_key) -> None:
"""
Sets the translator api key.
Parameters:
api_key (str) : The api key.
"""
try:
EasyTL.set_credentials(Translator.TRANSLATION_METHOD, str(api_key))
is_valid, e = EasyTL.test_credentials(Translator.TRANSLATION_METHOD)
if(is_valid == False and e is not None):
raise e
except:
raise gr.Error("Invalid API key")
##-------------------start-of-update_translator_api_key()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def update_translator_api_key(api_key) -> None:
"""
Updates the translator api key.
Parameters:
api_key (str) : The api key.
"""
method_to_path = {
"openai": FileEnsurer.openai_api_key_path,
"gemini": FileEnsurer.gemini_api_key_path,
"deepl": FileEnsurer.deepl_api_key_path
}
path_to_api_key = method_to_path.get(Translator.TRANSLATION_METHOD, None)
assert path_to_api_key is not None, "Invalid translation method"
FileEnsurer.standard_overwrite_file(path_to_api_key, base64.b64encode(str(api_key).encode('utf-8')).decode('utf-8'), omit=True)
##-------------------start-of-create_new_key_value_tuple_pairs()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def create_new_key_value_tuple_pairs(translation_settings:typing.List[str]) -> typing.List[typing.Tuple[str, str]]:
"""
Applies the new translation to the uploaded translation_settings.json file.
Parameters:
translation_settings (typing.List[typing.Union[gr.Textbox,gr.Slider,gr.Dropdown]]) : The translation settings.
Returns:
key_value_tuple_pairs (typing.List[typing.Tuple[str, str]]) : The new key value tuple pairs.
"""
key_value_tuple_pairs = []
translation_settings_key_names = {
0: "prompt_assembly_mode",
1: "number_of_lines_per_batch",
2: "sentence_fragmenter_mode",
3: "je_check_mode",
4: "number_of_malformed_batch_retries",
5: "batch_retry_timeout",
6: "number_of_concurrent_batches",
7: "openai_model",
8: "openai_system_message",
9: "openai_temperature",
10: "openai_top_p",
11: "openai_n",
12: "openai_stream",
13: "openai_stop",
14: "openai_logit_bias",
15: "openai_max_tokens",
16: "openai_presence_penalty",
17: "openai_frequency_penalty",
18: "gemini_model",
19: "gemini_prompt",
20: "gemini_temperature",
21: "gemini_top_p",
22: "gemini_top_k",
23: "gemini_candidate_count",
24: "gemini_stream",
25: "gemini_stop_sequences",
26: "gemini_max_output_tokens",
27: "deepl_context",
28: "deepl_split_sentences",
29: "deepl_preserve_formatting",
30: "deepl_formality",
}
for index, setting in enumerate(translation_settings):
key = translation_settings_key_names.get(index)
key_value_tuple_pairs.append((key, setting))
return key_value_tuple_pairs
##-------------------start-of-GUI-Structure---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
## tab 1 | Main
with gr.Tab("Kudasai") as self.kudasai_tab:
## tab 2 | indexing
with gr.Tab("Name Indexing | Kairyou") as self.indexing_tab:
with gr.Row():
## input fields, text input for indexing, and replacement json file input.
with gr.Column():
self.input_txt_file_indexing = gr.File(label='TXT file with Japanese Text', file_count='single', file_types=['.txt'], type='filepath')
self.input_json_file_indexing = gr.File(label='Replacements JSON file', file_count='single', file_types=['.json'], type='filepath')
self.input_knowledge_base_file = gr.File(label='Knowledge Base Single File', file_count='single', file_types=['.txt'], type='filepath')
self.input_knowledge_base_directory = gr.File(label='Knowledge Base Directory', file_count='directory', type='filepath')
## run and clear buttons
with gr.Row():
self.indexing_run_button = gr.Button('Run', variant='primary')
self.indexing_clear_button = gr.Button('Clear', variant='stop')
with gr.Row():
self.send_to_kairyou_button = gr.Button('Send to Preprocessing (Kairyou)')
## output fields
with gr.Column():
self.indexing_output_field = gr.Textbox(label='Indexed text', lines=56, max_lines=56, show_label=True, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_indexed_text = gr.Button('Save As')
with gr.Column():
self.indexing_results_output_field = gr.Textbox(label='Indexing Results', lines=56, max_lines=56, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_indexing_results = gr.Button('Save As')
with gr.Column():
self.debug_log_output_field_indexing_tab = gr.Textbox(label='Debug Log', lines=56, max_lines=56, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_debug_log_indexing_tab = gr.Button('Save As')
## tab 3 | preprocessing
with gr.Tab("Text Preprocessing | Kairyou") as self.preprocessing_tab:
with gr.Row():
## input fields, text input for preprocessing, and replacement json file input.
with gr.Column():
self.input_txt_file_preprocessing = gr.File(label='TXT file with Japanese Text', file_count='single', file_types=['.txt'], type='filepath')
self.input_json_file_preprocessing = gr.File(label='Replacements JSON file', file_count='single', file_types=['.json'], type='filepath')
self.input_text_kairyou = gr.Textbox(label='Japanese Text', placeholder='Use this or the text file input, if you provide both, Kudasai will use the file input.', lines=10, show_label=True, interactive=True)
## run and clear buttons
with gr.Row():
self.preprocessing_run_button = gr.Button('Run', variant='primary')
self.preprocessing_clear_button = gr.Button('Clear', variant='stop')
with gr.Row():
self.send_to_translator_button = gr.Button('Send to Translator')
## output fields
with gr.Column():
self.preprocessing_output_field = gr.Textbox(label='Preprocessed text', lines=42, max_lines=42, show_label=True, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_preprocessed_text = gr.Button('Save As')
with gr.Column():
self.preprocessing_results_output_field = gr.Textbox(label='Preprocessing Results', lines=42, max_lines=42, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_preprocessing_results = gr.Button('Save As')
with gr.Column():
self.debug_log_output_field_preprocess_tab = gr.Textbox(label='Debug Log', lines=42, max_lines=42, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_debug_log_preprocessing_tab = gr.Button('Save As')
## tab 4 | Translation
with gr.Tab("Text Translation | Translator") as self.translator_tab:
with gr.Row():
## input file or text input, gui allows for both but will prioritize file input
with gr.Column():
self.input_txt_file_translator = gr.File(label='TXT file with Japanese Text', file_count='single', file_types=['.txt'], type='filepath', interactive=True)
self.input_text_translator = gr.Textbox(label='Japanese Text', placeholder='Use this or the text file input, if you provide both, Kudasai will use the file input.', lines=10, show_label=True, interactive=True, type='text')
self.input_translation_rules_file = gr.File(value = FileEnsurer.config_translation_settings_path, label='Translation Settings File', file_count='single', file_types=['.json'], type='filepath')
with gr.Row():
self.llm_option_dropdown = gr.Dropdown(label='Translation Method', choices=["OpenAI", "Gemini", "DeepL"], value="OpenAI", show_label=True, interactive=True)
with gr.Row():
self.translator_api_key_input = gr.Textbox(label='API Key', value=get_saved_api_key("openai"), lines=1, max_lines=2, show_label=True, interactive=True, type='password')
with gr.Row():
self.translator_translate_button = gr.Button('Translate', variant="primary")
self.translator_calculate_cost_button = gr.Button('Calculate Cost', variant='secondary')
with gr.Row():
self.translator_clear_button = gr.Button('Clear', variant='stop')
## output fields
with gr.Column():
self.translator_translated_text_output_field = gr.Textbox(label='Translated Text', lines=43,max_lines=43, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_translator_translated_text = gr.Button('Save As')
with gr.Column():
self.translator_je_check_text_output_field = gr.Textbox(label='JE Check Text', lines=43,max_lines=43, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_translator_je_check_text = gr.Button('Save As')
with gr.Column():
self.translator_debug_log_output_field = gr.Textbox(label='Debug Log', lines=43, max_lines=43, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_debug_log_translator_tab = gr.Button('Save As')
## tab 5 | Settings
with gr.Tab("Translation Settings") as self.translation_settings_tab:
with gr.Row():
with gr.Column():
gr.Markdown("Base Translation Settings")
gr.Markdown("These settings are used for OpenAI, Gemini, and DeepL.")
gr.Markdown("Please ensure to thoroughly read and understand these settings before making any modifications. Each setting has a specific impact on the translation methods. Some settings may affect one or two translation methods, but not the others. Incorrect adjustments could lead to unexpected results or errors in the translation process.")
self.prompt_assembly_mode_input_field = gr.Dropdown(label='Prompt Assembly Mode',
value=int(GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","prompt_assembly_mode")),
choices=[1,2],
info=KudasaiGUI.description_dict.get("prompt_assembly_mode", "An error occurred while fetching the description for this setting."),
show_label=True,
interactive=True,
elem_id="prompt_assembly_mode")
self.number_of_lines_per_batch_input_field = gr.Textbox(label='Number of Lines Per Batch',
value=(GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","number_of_lines_per_batch")),
info=KudasaiGUI.description_dict.get("number_of_lines_per_batch"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="number_of_lines_per_batch",
show_copy_button=True)
self.sentence_fragmenter_mode_input_field = gr.Dropdown(label='Sentence Fragmenter Mode',
value=int(GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","sentence_fragmenter_mode")),
choices=[1,2],
info=KudasaiGUI.description_dict.get("sentence_fragmenter_mode"),
show_label=True,
interactive=True,
elem_id="sentence_fragmenter_mode")
self.je_check_mode_input_field = gr.Dropdown(label='JE Check Mode',
value=int(GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","je_check_mode")),
choices=[1,2],
info=KudasaiGUI.description_dict.get("je_check_mode"),
show_label=True,
interactive=True,
elem_id="je_check_mode")
self.number_of_malformed_batch_retries_input_field = gr.Textbox(label="Number Of Malformed Batch Retries",
value=GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","number_of_malformed_batch_retries"),
info=KudasaiGUI.description_dict.get("number_of_malformed_batch_retries"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="number_of_malformed_batch_retries",
show_copy_button=True)
self.batch_retry_timeout_input_field = gr.Textbox(label="Batch Retry Timeout",
value=GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","batch_retry_timeout"),
info=KudasaiGUI.description_dict.get("batch_retry_timeout"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="batch_retry_timeout",
show_copy_button=True)
self.number_of_concurrent_batches_input_field = gr.Textbox(label="Number Of Concurrent Batches",
value=GuiJsonUtil.fetch_translation_settings_key_values("base translation settings","number_of_concurrent_batches"),
info=KudasaiGUI.description_dict.get("number_of_concurrent_batches"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="number_of_concurrent_batches",
show_copy_button=True)
with gr.Column():
## all these need to be changed later as well
gr.Markdown("OpenAI API Settings")
gr.Markdown(str(KudasaiGUI.description_dict.get("openai_help_link")))
gr.Markdown(str(KudasaiGUI.description_dict.get("openai_disclaimer")))
self.openai_model_input_field = gr.Dropdown(label="OpenAI Model",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_model")),
choices=[model for model in ALLOWED_OPENAI_MODELS],
info=KudasaiGUI.description_dict.get("openai_model"),
show_label=True,
interactive=True,
elem_id="openai_model")
self.openai_system_message_input_field = gr.Textbox(label="OpenAI System Message",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_system_message")),
info=KudasaiGUI.description_dict.get("openai_system_message"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="openai_system_message",
show_copy_button=True)
self.openai_temperature_input_field = gr.Slider(label="OpenAI Temperature",
value=float(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_temperature")),
minimum=0.0,
maximum=2.0,
info=KudasaiGUI.description_dict.get("openai_temperature"),
interactive=True,
elem_id="openai_temperature")
self.openai_top_p_input_field = gr.Slider(label="OpenAI Top P",
value=float(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_top_p")),
minimum=0.0,
maximum=1.0,
info=KudasaiGUI.description_dict.get("openai_top_p"),
show_label=True,
interactive=True,
elem_id="openai_top_p")
self.openai_n_input_field = gr.Textbox(label="OpenAI N",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_n")),
info=KudasaiGUI.description_dict.get("openai_n"),
show_label=True,
interactive=False,
elem_id="openai_n",
show_copy_button=True)
self.openai_stream_input_field = gr.Textbox(label="OpenAI Stream",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_stream")),
info=KudasaiGUI.description_dict.get("openai_stream"),
show_label=True,
interactive=False,
elem_id="openai_stream",
show_copy_button=True)
self.openai_stop_input_field = gr.Textbox(label="OpenAI Stop",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_stop")),
info=KudasaiGUI.description_dict.get("openai_stop"),
show_label=True,
interactive=False,
elem_id="openai_stop",
show_copy_button=True)
self.openai_logit_bias_input_field = gr.Textbox(label="OpenAI Logit Bias",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_logit_bias")),
info=KudasaiGUI.description_dict.get("openai_logit_bias"),
show_label=True,
interactive=False,
elem_id="openai_logit_bias",
show_copy_button=True)
self.openai_max_tokens_input_field = gr.Textbox(label="OpenAI Max Tokens",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_max_tokens")),
info=KudasaiGUI.description_dict.get("openai_max_tokens"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="openai_max_tokens",
show_copy_button=True)
self.openai_presence_penalty_input_field = gr.Slider(label="OpenAI Presence Penalty",
value=float(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_presence_penalty")),
info=KudasaiGUI.description_dict.get("openai_presence_penalty"),
minimum=-2.0,
maximum=2.0,
show_label=True,
interactive=True,
elem_id="openai_presence_penalty")
self.openai_frequency_penalty_input_field = gr.Slider(label="OpenAI Frequency Penalty",
value=float(GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_frequency_penalty")),
info=KudasaiGUI.description_dict.get("openai_frequency_penalty"),
minimum=-2.0,
maximum=2.0,
show_label=True,
interactive=True,
elem_id="openai_frequency_penalty")
with gr.Row():
with gr.Column():
gr.Markdown("Gemini API Settings")
gr.Markdown(str(KudasaiGUI.description_dict.get("gemini_help_link")))
gr.Markdown(str(KudasaiGUI.description_dict.get("gemini_disclaimer")))
self.gemini_model_input_field = gr.Dropdown(label="Gemini Model",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_model")),
choices=[model for model in ALLOWED_GEMINI_MODELS],
info=KudasaiGUI.description_dict.get("gemini_model"),
show_label=True,
interactive=True,
elem_id="gemini_model")
self.gemini_prompt_input_field = gr.Textbox(label="Gemini Prompt",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_prompt")),
info=KudasaiGUI.description_dict.get("gemini_prompt"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="gemini_prompt",
show_copy_button=True)
self.gemini_temperature_input_field = gr.Slider(label="Gemini Temperature",
value=float(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_temperature")),
minimum=0.0,
maximum=2.0,
info=KudasaiGUI.description_dict.get("gemini_temperature"),
show_label=True,
interactive=True,
elem_id="gemini_temperature")
self.gemini_top_p_input_field = gr.Textbox(label="Gemini Top P",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_top_p")),
info=KudasaiGUI.description_dict.get("gemini_top_p"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="gemini_top_p",
show_copy_button=True)
self.gemini_top_k_input_field = gr.Textbox(label="Gemini Top K",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_top_k")),
info=KudasaiGUI.description_dict.get("gemini_top_k"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="gemini_top_k",
show_copy_button=True)
self.gemini_candidate_count_input_field = gr.Textbox(label="Gemini Candidate Count",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_candidate_count")),
info=KudasaiGUI.description_dict.get("gemini_candidate_count"),
lines=1,
max_lines=1,
show_label=True,
interactive=False,
elem_id="gemini_candidate_count",
show_copy_button=True)
self.gemini_stream_input_field = gr.Textbox(label="Gemini Stream",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_stream")),
info=KudasaiGUI.description_dict.get("gemini_stream"),
lines=1,
max_lines=1,
show_label=True,
interactive=False,
elem_id="gemini_stream",
show_copy_button=True)
self.gemini_stop_sequences_input_field = gr.Textbox(label="Gemini Stop Sequences",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_stop_sequences")),
info=KudasaiGUI.description_dict.get("gemini_stop_sequences"),
lines=1,
max_lines=1,
show_label=True,
interactive=False,
elem_id="gemini_stop_sequences",
show_copy_button=True)
self.gemini_max_output_tokens_input_field = gr.Textbox(label="Gemini Max Output Tokens",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_max_output_tokens")),
info=KudasaiGUI.description_dict.get("gemini_max_output_tokens"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="gemini_max_output_tokens",
show_copy_button=True)
with gr.Column():
gr.Markdown("DeepL API Settings")
gr.Markdown(str(KudasaiGUI.description_dict.get("deepl_help_link")))
gr.Markdown("DeepL API settings are not as extensive as OpenAI and Gemini, a lot of the settings are simply not included for Kudasai as they do not have a decent use case to warrant their inclusion. Settings may be added in the future if a use case is found or is suggested.")
self.deepl_context_input_field = gr.Textbox(label="DeepL Context",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("deepl settings","deepl_context")),
info=KudasaiGUI.description_dict.get("deepl_context"),
lines=1,
max_lines=1,
show_label=True,
interactive=True,
elem_id="deepl_context",
show_copy_button=True)
self.deepl_split_sentences_input_field = gr.Dropdown(label="DeepL Split Sentences",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("deepl settings","deepl_split_sentences")),
choices=['OFF', 'ALL', 'NO_NEWLINES'],
info=KudasaiGUI.description_dict.get("deepl_split_sentences"),
show_label=True,
interactive=True,
elem_id="deepl_split_sentences")
self.deepl_preserve_formatting_input_field = gr.Checkbox(label="DeepL Preserve Formatting",
value=bool(GuiJsonUtil.fetch_translation_settings_key_values("deepl settings","deepl_preserve_formatting")),
info=KudasaiGUI.description_dict.get("deepl_preserve_formatting"),
show_label=True,
interactive=True,
elem_id="deepl_preserve_formatting")
self.deepl_formality_input_field = gr.Dropdown(label="DeepL Formality",
value=str(GuiJsonUtil.fetch_translation_settings_key_values("deepl settings","deepl_formality")),
choices=['default', 'more', 'less', 'prefer_more', 'prefer_less'],
info=KudasaiGUI.description_dict.get("deepl_formality"),
show_label=True,
interactive=True,
elem_id="deepl_formality")
with gr.Row():
self.translation_settings_reset_to_default_button = gr.Button('Reset to Default', variant='secondary')
self.translation_settings_discard_changes_button = gr.Button('Discard Changes', variant='stop')
with gr.Row():
self.translation_settings_apply_changes_button = gr.Button('Apply Changes', variant='primary')
## tab 6 | Logging
with gr.Tab("Logging") as self.logging_tab:
with gr.Row():
self.logging_tab_debug_log_output_field = gr.Textbox(label='Debug Log', lines=10, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_debug_log_logging_tab = gr.Button('Save As')
with gr.Row():
self.logging_tab_error_log_output_field = gr.Textbox(label='Error Log', lines=10, interactive=False, show_copy_button=True)
with gr.Row():
self.save_to_file_error_log_logging_tab = gr.Button('Save As')
with gr.Row():
self.logging_clear_logs_button = gr.Button('Clear Logs', variant='stop')
##-------------------start-of-Listener-Functions---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def webgui_update_check() -> None:
"""
Checks for if a Kudasai update is available.
"""
Kudasai.connection, update_prompt = Toolkit.check_update(do_pause=False)
if(update_prompt != ""):
gr.Info("Update available, see https://github.com/Bikatr7/Kudasai/releases/latest/ for more information.")
if(Kudasai.connection == False):
gr.Warning("No internet connection, Auto-MTL features disabled (Indexing and Preprocessing still functional). Please reload the page when you have an internet connection.")
##-------------------start-of-index()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def index(input_txt_file:gr.File, input_json_file_preprocessing:gr.File, input_knowledge_base_file:gr.File, input_knowledge_base_directory:typing.List[str]) -> typing.Tuple[str, str, str, str, str]:
"""
Runs the indexing and displays the results in the indexing output field. If no txt file is selected, an error is raised. If no json file is selected, an error is raised. If no knowledge base file or directory is selected, an error is raised.
Knowledge base file or directory must be selected, but not both.
Also displays the indexing results, the debug log, and the error log.
Parameters:
input_txt_file (gr.File) : The input txt file.
input_json_file_preprocessing (gr.File) : The input json file.
input_knowledge_base_file (gr.File) : The knowledge base file.
input_knowledge_base_directory (typing.List[str]) : List of knowledge base file paths.
Returns:
indexed_text (str) : The indexed text.
indexing_log (str) : The indexing log.
log_text (str) : The log text for the Indexing tab.
log_text (str) : The log text for the log tab.
error_log (str) : The error log for the log tab.
"""
if(input_txt_file is not None):
if(input_json_file_preprocessing is not None):
## must be one, but not both
if(input_knowledge_base_file is not None or input_knowledge_base_directory is not None) and not (input_knowledge_base_file is not None and input_knowledge_base_directory is not None):
## looks like file will just be the file path
## but directory will be a list of file paths, I can't think of a workaround right now, so will have to update kairyou to accept a list of file paths.
## wait nvm im a genius, let's just read all the files and concatenate them into one string lmfao
## index does not produce an error log
error_log = ""
knowledge_base_paths = []
knowledge_base_string = ""
text_to_index = gui_get_text_from_file(input_txt_file)
replacements = gui_get_json_from_file(input_json_file_preprocessing)
## DON"T DO THIS, THIS IS BAD PRACTICE, I'M JUST DOING BECAUSE I'M LAZY AND I MADE THE LIBRARY
try:
_validate_replacement_json(replacements)
except InvalidReplacementJsonKeys:
raise gr.Error("Invalid JSON file, please ensure that the JSON file contains the correct keys See https://github.com/Bikatr7/Kairyou for more information.")
if(input_knowledge_base_file is not None):
knowledge_base_paths.append(input_knowledge_base_file)
else:
knowledge_base_paths = [file for file in input_knowledge_base_directory]
for file in knowledge_base_paths:
knowledge_base_string += gui_get_text_from_file(file)
gr.Info("Indexing may take a while, please be patient.")
unique_names, indexing_log = Indexer.index(text_to_index, knowledge_base_string, replacements)
## Indexer does not directly log anything, in case of anything else touching it, we will grab the log from the log file
log_text = FileEnsurer.standard_read_file(FileEnsurer.debug_log_path)
indexed_text = Kudasai.mark_indexed_names(text_to_index, unique_names)
return indexed_text, indexing_log, log_text, log_text, error_log
else:
raise gr.Error("No knowledge base file or directory selected (or both selected, select one or the other)")
else:
raise gr.Error("No JSON file selected")
else:
raise gr.Error("No TXT file selected")
##-------------------start-of-preprocess()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def preprocess(input_txt_file:gr.File, input_json_file_preprocessing:gr.File, input_text_field_contents:str) -> typing.Tuple[str, str, str, str, str]:
"""
Runs the preprocessing and displays the results in the preprocessing output field. If no txt file is selected, an error is raised. If no json file is selected, an error is raised.
Also displays the preprocessing results, the debug log, and the error log.
Prioritizes the txt file input over the text input field.
Parameters:
input_txt_file (gr.File) : The input txt file.
input_json_file_preprocessing (gr.File) : The input json file.
input_text (str) : The input text.
Returns:
text_to_preprocess (str) : The preprocessed text.
preprocessing_log (str) : The preprocessing log.
log_text (str) : The log text for the Kairyou tab.
log_text (str) : The log text for the log tab.
error_log (str) : The error log for the log tab.
"""
if(input_txt_file == None and input_text_field_contents == ""):
raise gr.Error("No TXT file selected and no text input")
if(input_json_file_preprocessing is not None):
if(input_txt_file is not None):
text_to_preprocess = gui_get_text_from_file(input_txt_file)
else:
text_to_preprocess = input_text_field_contents
replacements = gui_get_json_from_file(input_json_file_preprocessing)
try:
preprocessed_text, preprocessing_log, error_log = Kairyou.preprocess(text_to_preprocess, replacements)
except InvalidReplacementJsonKeys:
## link needs to be updated later
raise gr.Error("Invalid JSON file, please ensure that the JSON file contains the correct keys See: https://github.com/Bikatr7/Kairyou?tab=readme-ov-file#usage")
timestamp = Toolkit.get_timestamp(is_archival=True)
FileEnsurer.write_kairyou_results(preprocessed_text, preprocessing_log, error_log, timestamp)
log_text = FileEnsurer.standard_read_file(FileEnsurer.debug_log_path)
return preprocessed_text, preprocessing_log, log_text, log_text, error_log
else:
raise gr.Error("No JSON file selected")
##-------------------start-of-translate_with_translator()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
async def translate_with_translator(input_txt_file:gr.File, input_text:str, api_key:str, translation_method:str, translation_settings_file:gr.File) -> typing.Tuple[str, str, str, str]:
"""
Translates the text in the input_txt_file or input_text using either OpenAI, Gemini, or DeepL. If no txt file or text is selected, an error is raised. If no API key is provided or the API key is invalid, an error is raised.
Displays the translated text, the debug log, and the error log.
Parameters:
input_txt_file (gr.File) : The input txt file.
input_text (gr.Textbox) : The input text.
api_key_input (gr.Textbox) : The API key input.
Returns:
translated_text (str) : The translated text.
je_check_text (str) : The je check text.
log_text (str) : The log text for the Log tab.
error_text (str) : The error text for the Log tab.
"""
## check if we have stuff to translate
if(input_txt_file is None and input_text == ""):
raise gr.Error("No TXT file or text selected")
if(api_key == ""):
raise gr.Error("No API key provided")
if(translation_settings_file is None):
raise gr.Error("No Translation Settings File selected")
if(Kudasai.connection == False):
raise gr.Error("No internet connection detected, please connect to the internet and reload the page to use translation features of Kudasai.")
## in case of subsequent runs, we need to reset the static variables
Translator.reset_static_variables()
## start of translation, so we can assume that that we don't want to interrupt it
FileEnsurer.do_interrupt = False
## if translate button is clicked, we can assume that the translation is ongoing
self.is_translation_ongoing = True
## first, set the json in the json handler to the json currently set as in gui_json_util
JsonHandler.current_translation_settings = GuiJsonUtil.current_translation_settings
## next, set the llm type
if(translation_method == "OpenAI"):
Translator.TRANSLATION_METHOD = "openai"
elif(translation_method == "Gemini"):
Translator.TRANSLATION_METHOD = "gemini"
else:
Translator.TRANSLATION_METHOD = "deepl"
## api key as well
await set_translator_api_key(api_key)
if(input_txt_file is not None):
text_to_translate = gui_get_text_from_file(input_txt_file)
else:
text_to_translate = input_text
## need to convert the text to translate to list of strings
Translator.text_to_translate = [line for line in str(text_to_translate).splitlines()]
## commence translation
await Translator.commence_translation(is_webgui=True)
Translator.write_translator_results()
## je check text and translated text are lists of strings, so we need to convert them to strings
translated_text = "\n".join(Translator.translated_text)
je_check_text = "\n".join(Translator.je_check_text)
## Log text is cleared from the client, so we need to get it from the log file
log_text = FileEnsurer.standard_read_file(FileEnsurer.debug_log_path)
error_text = FileEnsurer.standard_read_file(FileEnsurer.error_log_path)
## then overwrite the api key file with the new api key
update_translator_api_key(api_key)
return translated_text, je_check_text, log_text, error_text
##-------------------start-of translator_calculate_costs_button_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
async def translator_calculate_costs_button_click(input_txt_file:str, input_text:str, translation_method:str, api_key:str, translation_settings_file:gr.File) -> str:
"""
Calculates the cost of the text in the input_txt_file or input_text using the OpenAI, Gemini, or DeepL APIs. If no txt file or text is selected, an error is raised.
Displays the cost, and the debug log.
Parameters:
input_txt_file (gr.File) : The input txt file.
input_text (str) : The input text.
translation_method (str) : The language model type.
api_key (str) : The
Returns:
cost_estimation (str) : The cost estimation formatted as a string.
"""
if(input_txt_file is None and input_text == ""):
raise gr.Error("No TXT file or text selected")
if(api_key == "" and translation_method not in ["OpenAI","DeepL"]):
raise gr.Error("No API key provided. Does not charge for cost estimation, but is required for Gemini Cost Calculation")
if(Kudasai.connection == False and translation_method != "OpenAI"):
raise gr.Error("No internet connection detected, please connect to the internet and reload the page to calculate costs for Gemini")
if(translation_settings_file is None):
raise gr.Error("No Translation Settings File selected")
## in case of subsequent runs, we need to reset the static variables
Translator.reset_static_variables()
cost_estimation = ""
Translator.TRANSLATION_METHOD = str(translation_method.lower()) # type: ignore
await set_translator_api_key(api_key)
translation_methods = {
"openai": GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_model"),
"gemini": GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_model"),
"deepl": "deep"
}
model = translation_methods.get(Translator.TRANSLATION_METHOD)
if(input_txt_file is not None):
text_to_translate = gui_get_text_from_file(input_txt_file)
else:
text_to_translate = input_text
translation_instructions_dict = {
"openai": GuiJsonUtil.fetch_translation_settings_key_values("openai settings","openai_system_message"),
"gemini": GuiJsonUtil.fetch_translation_settings_key_values("gemini settings","gemini_prompt"),
"deepl": None
}
translation_instructions = translation_instructions_dict.get(Translator.TRANSLATION_METHOD)
num_tokens, estimated_cost, model = EasyTL.calculate_cost(text=text_to_translate, service=Translator.TRANSLATION_METHOD, model=model, translation_instructions=translation_instructions)
if(Translator.TRANSLATION_METHOD == "gemini"):
cost_estimation = f"As of Kudasai {Toolkit.CURRENT_VERSION}, Gemini Pro 1.0 is free to use under 15 requests per minute, Gemini Pro 1.5 is free to use under 2 requests per minute.\nIt is up to you to set these in the settings json.\n"
token_type = "characters" if Translator.TRANSLATION_METHOD == "deepl" else "tokens"
cost_estimation += f"Estimated number of {token_type} : {num_tokens}\nEstimated minimum cost : {estimated_cost} USD\nThis is a rough estimate, please remember to check actual cost on the appropriate platform when needed"
gr.Info(cost_estimation)
update_translator_api_key(api_key)
return cost_estimation
##-------------------start-of-clear_index_tab()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def clear_index_tab() -> typing.Tuple[None, None, None, None, str, str, str]:
"""
Clears all fields on the indexing tab. As well as the input fields.
Returns:
input_txt_file_indexing (gr.File) : An empty file.
input_json_file_indexing (gr.File) : An empty file.
input_knowledge_base_file (gr.File) : An empty file.
indexing_output_field (str) : An empty string.
indexing_results_output_field (str) : An empty string.
debug_log_output_field_indexing_tab (str) : An empty string.
"""
input_txt_file_indexing = None
input_json_file_indexing = None
input_knowledge_base_file = None
input_knowledge_base_directory = None
indexing_output_field = ""
indexing_results_output_field = ""
debug_log_output_field_indexing_tab = ""
return input_txt_file_indexing, input_json_file_indexing, input_knowledge_base_file, input_knowledge_base_directory, indexing_output_field, indexing_results_output_field, debug_log_output_field_indexing_tab
##-------------------start-of-clear_preprocessing_tab()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def clear_preprocessing_tab() -> typing.Tuple[None, None, str, str, str, str]:
"""
Clears all fields on the preprocessing tab. As well as the input fields.
Returns:
input_txt_file (gr.File) : An empty file.
input_json_file_preprocessing (gr.File) : An empty file.
preprocessing_output_field (str) : An empty string.
preprocessing_results_output_field (str) : An empty string.
debug_log_output_field_preprocess_tab (str) : An empty string.
"""
input_txt_file = None
input_json_file_preprocessing = None
input_text = ""
preprocessing_output_field = ""
preprocessing_results_output_field = ""
debug_log_output_field_preprocess_tab = ""
return input_txt_file, input_json_file_preprocessing, input_text, preprocessing_output_field, preprocessing_results_output_field, debug_log_output_field_preprocess_tab
##-------------------start-of-clear_translator_tab()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def clear_translator_tab() -> typing.Tuple[None, str, gr.File, str, str, str]:
"""
Clears all fields on the Translator tab. As well as the input fields.
Returns:
input_txt_file_translator (gr.File) : An empty file.
input_text_translator (str) : An empty string.
translator_translated_text_output_field (str) : An empty string.
je_check_text_field_translator (str) : An empty string.
translator_debug_log_output_field (str) : An empty string.
"""
## if clear button is clicked, we can assume that the translation is over, or that the user wants to cancel the translation
self.is_translation_ongoing = False
## Same as above, we can assume that the user wants to cancel the translation if it's ongoing
FileEnsurer.do_interrupt = True
input_file_translator = None
input_text_translator = ""
## Also gonna want to reset the json input field to the default json file
input_translation_rules_file = gr.File(value = FileEnsurer.config_translation_settings_path, label='Translation Settings File', file_count='single', file_types=['.json'], type='filepath')
translator_translated_text_output_field = ""
je_check_text_field_translator = ""
translator_debug_log_output_field = ""
return input_file_translator, input_text_translator, input_translation_rules_file, translator_translated_text_output_field, je_check_text_field_translator, translator_debug_log_output_field
##-------------------start-of-clear_log_button_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def clear_log_button_click() -> typing.Tuple[str, str]:
"""
Clears the logs on the log tab.
Returns:
logging_tab_debug_log_output_field (str) : An empty string.
logging_tab_error_log_output_field (str) : An empty string.
"""
## We don't actually clear the logs as they need to be kept. We just clear the output fields
logging_tab_debug_log_output_field = ""
logging_tab_error_log_output_field = ""
return logging_tab_debug_log_output_field, logging_tab_error_log_output_field
##-------------------start-of-apply_new_translator_settings()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def apply_new_translator_settings(
input_translation_rules_file:gr.File,
prompt_assembly_mode:int,
number_of_lines_per_batch:int,
sentence_fragmenter_mode:int,
je_check_mode:int,
num_malformed_batch_retries:int,
batch_retry_timeout:int,
num_concurrent_batches:int,
openai_model:str,
openai_system_message:str,
openai_temperature:float,
openai_top_p:float,
openai_n:str,
openai_stream:str,
openai_stop:str,
openai_logit_bias:str,
openai_max_tokens:str,
openai_presence_penalty:float,
openai_frequency_penalty:float,
gemini_model:str,
gemini_prompt:str,
gemini_temperature:float,
gemini_top_p:str,
gemini_top_k:str,
gemini_candidate_count:str,
gemini_stream:str,
gemini_stop_sequences:str,
gemini_max_output_tokens:str,
deepl_context:str,
deepl_split_sentences:str,
deepl_preserve_formatting:bool,
deepl_formality:str) -> None:
"""
Applies the new translation settings to the Translation Settings File.
"""
if(input_translation_rules_file is None):
raise gr.Error("No Translation Settings File Selected. Cannot apply settings.")
## build the new translation settings list so we can create a key-value pair list
settings_list = [
prompt_assembly_mode,
number_of_lines_per_batch,
sentence_fragmenter_mode,
je_check_mode,
num_malformed_batch_retries,
batch_retry_timeout,
num_concurrent_batches,
openai_model,
openai_system_message,
openai_temperature,
openai_top_p,
openai_n,
openai_stream,
openai_stop,
openai_logit_bias,
openai_max_tokens,
openai_presence_penalty,
openai_frequency_penalty,
gemini_model,
gemini_prompt,
gemini_temperature,
gemini_top_p,
gemini_top_k,
gemini_candidate_count,
gemini_stream,
gemini_stop_sequences,
gemini_max_output_tokens,
deepl_context,
deepl_split_sentences,
deepl_preserve_formatting,
deepl_formality
]
## create the new key-value pair list
new_key_value_tuple_pairs = create_new_key_value_tuple_pairs(settings_list)
try:
## and then have the GuiJsonUtil apply the new translator settings
GuiJsonUtil.update_translation_settings_with_new_values(input_translation_rules_file, new_key_value_tuple_pairs)
except:
raise gr.Error("Invalid Translator Settings")
gr.Info("Translator Settings Applied")
##-------------------start-of-reset_to_default_translation_settings()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def reset_to_default_translation_settings(input_translation_rules_file:gr.File):
"""
Resets the translation settings to the default values.
"""
if(input_translation_rules_file is None):
raise gr.Error("No Translation Settings File Selected. Cannot reset settings.")
GuiJsonUtil.current_translation_settings = FileEnsurer.DEFAULT_TRANSLATION_SETTING
settings = [
("base translation settings", "prompt_assembly_mode", int),
("base translation settings", "number_of_lines_per_batch", str),
("base translation settings", "sentence_fragmenter_mode", int),
("base translation settings", "je_check_mode", int),
("base translation settings", "number_of_malformed_batch_retries", str),
("base translation settings", "batch_retry_timeout", str),
("base translation settings", "number_of_concurrent_batches", str),
("openai settings", "openai_model", str),
("openai settings", "openai_system_message", str),
("openai settings", "openai_temperature", float),
("openai settings", "openai_top_p", float),
("openai settings", "openai_n", str),
("openai settings", "openai_stream", str),
("openai settings", "openai_stop", str),
("openai settings", "openai_logit_bias", str),
("openai settings", "openai_max_tokens", str),
("openai settings", "openai_presence_penalty", float),
("openai settings", "openai_frequency_penalty", float),
("gemini settings", "gemini_model", str),
("gemini settings", "gemini_prompt", str),
("gemini settings", "gemini_temperature", float),
("gemini settings", "gemini_top_p", str),
("gemini settings", "gemini_top_k", str),
("gemini settings", "gemini_candidate_count", str),
("gemini settings", "gemini_stream", str),
("gemini settings", "gemini_stop_sequences", str),
("gemini settings", "gemini_max_output_tokens", str),
("deepl settings", "deepl_context", str),
("deepl settings", "deepl_split_sentences", str),
("deepl settings", "deepl_preserve_formatting", bool),
("deepl settings", "deepl_formality", str),
]
return_batch = [cast(GuiJsonUtil.fetch_translation_settings_key_values(setting, key)) for setting, key, cast in settings]
gr.Info("Translator Settings Reset to Default. Make sure to press the Apply button to apply the changes.")
return return_batch
##-------------------start-of-refresh_translation_settings_fields()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def refresh_translation_settings_fields(input_translation_rules_file:str):
"""
Refreshes the translation settings fields with the values from the Translation Settings File.
"""
if(input_translation_rules_file is None):
raise gr.Error("No Translation Settings File Selected. Cannot refresh settings.")
try:
GuiJsonUtil.current_translation_settings = gui_get_json_from_file(input_translation_rules_file)
settings = [
("base translation settings", "prompt_assembly_mode", int),
("base translation settings", "number_of_lines_per_batch", str),
("base translation settings", "sentence_fragmenter_mode", int),
("base translation settings", "je_check_mode", int),
("base translation settings", "number_of_malformed_batch_retries", str),
("base translation settings", "batch_retry_timeout", str),
("base translation settings", "number_of_concurrent_batches", str),
("openai settings", "openai_model", str),
("openai settings", "openai_system_message", str),
("openai settings", "openai_temperature", float),
("openai settings", "openai_top_p", float),
("openai settings", "openai_n", str),
("openai settings", "openai_stream", str),
("openai settings", "openai_stop", str),
("openai settings", "openai_logit_bias", str),
("openai settings", "openai_max_tokens", str),
("openai settings", "openai_presence_penalty", float),
("openai settings", "openai_frequency_penalty", float),
("gemini settings", "gemini_model", str),
("gemini settings", "gemini_prompt", str),
("gemini settings", "gemini_temperature", float),
("gemini settings", "gemini_top_p", str),
("gemini settings", "gemini_top_k", str),
("gemini settings", "gemini_candidate_count", str),
("gemini settings", "gemini_stream", str),
("gemini settings", "gemini_stop_sequences", str),
("gemini settings", "gemini_max_output_tokens", str),
("deepl settings", "deepl_context", str),
("deepl settings", "deepl_split_sentences", str),
("deepl settings", "deepl_preserve_formatting", bool),
("deepl settings", "deepl_formality", str),
]
return_batch = [cast(GuiJsonUtil.fetch_translation_settings_key_values(setting, key)) for setting, key, cast in settings]
except:
GuiJsonUtil.current_translation_settings = JsonHandler.current_translation_settings
raise gr.Error("Invalid Custom Translation Settings File")
return return_batch
##-------------------start-of-clear_translation_settings_input_fields()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def clear_translation_settings_input_fields():
"""
Resets the translation settings input fields to None.
"""
settings = {
"input_translation_rules_file": None,
"prompt_assembly_mode_value": None,
"number_of_lines_per_batch_value": None,
"sentence_fragmenter_mode_value": None,
"je_check_mode_value": None,
"num_malformed_batch_retries_value": None,
"batch_retry_timeout_value": None,
"num_concurrent_batches_value": None,
"openai_model_value": None,
"openai_system_message_value": None,
"openai_temperature_value": None,
"openai_top_p_value": None,
"openai_n_value": None,
"openai_stream_value": None,
"openai_stop_value": None,
"openai_logit_bias_value": None,
"openai_max_tokens_value": None,
"openai_presence_penalty_value": None,
"openai_frequency_penalty_value": None,
"gemini_model_value": None,
"gemini_prompt_value": None,
"gemini_temperature_value": None,
"gemini_top_p_value": None,
"gemini_top_k_value": None,
"gemini_candidate_count_value": None,
"gemini_stream_value": None,
"gemini_stop_sequences_value": None,
"gemini_max_output_tokens_value": None,
"deepl_context": None,
"deepl_split_sentences": None,
"deepl_preserve_formatting": None,
"deepl_formality": None,
}
return settings
##-------------------start-of-fetch_log_content()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def fetch_debug_log_content() -> typing.Tuple[str, str]:
"""
Fetches the log content from the log file.
Returns:
log_text (str) : The log text.
logging_tab_error_log_output_field (str) : The error log.
"""
log_text = FileEnsurer.standard_read_file(FileEnsurer.debug_log_path)
logging_tab_error_log_output_field = FileEnsurer.standard_read_file(FileEnsurer.error_log_path)
return log_text, logging_tab_error_log_output_field
##-------------------start-of-send_to_kairyou_button()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def send_to_kairyou_button(input_text:str) -> str:
"""
Sends the indexed text to Kairyou.
Parameters:
input_text (str) : The input text.
Returns:
input_text (str) : The input text.
"""
if(input_text == ""):
gr.Warning("No indexed text to send to Preprocessor (Kairyou)")
return ""
else:
gr.Info("Indexed text copied to Preprocessor (Kairyou)")
return input_text
##-------------------start-of-send_to_translator_button()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def send_to_translator_button(input_text:str) -> str:
"""
Sends the preprocessed text to Translator.
Parameters:
input_text (str) : The input text.
Returns:
input_text (str) : The input text.
"""
if(input_text == ""):
gr.Warning("No preprocessed text to send to Translator")
return ""
else:
gr.Info("Preprocessed text copied to Translator")
return input_text
##-------------------start-of-switch_translator_api_key_type()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def switch_translator_api_key_type(translation_method:str) -> str:
"""
Switches the api key type between OpenAI, Gemini, and DeepL.
Parameters:
translation_method (str) : The translation method
Returns:
api_key (str) : The api key.
"""
return get_saved_api_key(translation_method.lower()) ## type: ignore
##-------------------start-of-Listener-Declaration---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
##-------------------start-of-load()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.gui.load(webgui_update_check)
self.gui.load(FileEnsurer.purge_storage)
##-------------------start-of-index()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.indexing_run_button.click(index,
inputs=[
self.input_txt_file_indexing, ## input txt file to index
self.input_json_file_indexing, ## input json file
self.input_knowledge_base_file, ## knowledge base file
self.input_knowledge_base_directory], ## knowledge base directory
outputs=[
self.indexing_output_field, ## indexed text
self.indexing_results_output_field, ## indexing results
self.debug_log_output_field_indexing_tab, ## debug log on indexing tab
self.logging_tab_debug_log_output_field, ## debug log on log tab
self.logging_tab_error_log_output_field])
##-------------------start-of-preprocess()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.preprocessing_run_button.click(fn=preprocess,
inputs=[
self.input_txt_file_preprocessing, ## input txt file to preprocess
self.input_json_file_preprocessing, ## replacements json file
self.input_text_kairyou], ## input text to preprocess
outputs=[
self.preprocessing_output_field, ## preprocessed text
self.preprocessing_results_output_field, ## kairyou results
self.debug_log_output_field_preprocess_tab, ## debug log on preprocess tab
self.logging_tab_debug_log_output_field, ## debug log on log tab
self.logging_tab_error_log_output_field]) ## error log on log tab
##-------------------start-of-translate_with_translator()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
## for the actual translation, and the je check text
translator_translate_process = self.translator_translate_button.click(translate_with_translator,
inputs=[
self.input_txt_file_translator, ## input txt file to translate
self.input_text_translator, ## input text to translate
self.translator_api_key_input, ## api key input
self.llm_option_dropdown, ## Translation Method dropdown
self.input_translation_rules_file], ## Translation Settings File
outputs=[
self.translator_translated_text_output_field, ## translated text
self.translator_je_check_text_output_field, ## je check text field on translator tab
self.logging_tab_debug_log_output_field , ## debug log on log tab
self.logging_tab_error_log_output_field]) ## error log on log tab
## for the debug log
self.translator_translate_button.click(fn=fetch_log_content,
inputs=[],
outputs=[self.translator_debug_log_output_field], ## debug log on translator tab
every=.1) ## update every 100ms
##-------------------start-of translator_calculate_costs_button_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.translator_calculate_cost_button.click(translator_calculate_costs_button_click,
inputs=[
self.input_txt_file_translator, ## input txt file to calculate costs
self.input_text_translator,
self.llm_option_dropdown,
self.translator_api_key_input,
self.input_translation_rules_file], ## Translation Settings File
outputs=[self.translator_translated_text_output_field]) ## functions as an output field for the cost output field
##-------------------start-of-clear_index_tab()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.indexing_clear_button.click(clear_index_tab,
inputs=[],
outputs=[
self.input_txt_file_indexing, ## input txt file
self.input_json_file_indexing, ## input json file
self.input_knowledge_base_file, ## knowledge base file
self.input_knowledge_base_directory, ## knowledge base directory
self.indexing_output_field, ## indexing output field
self.indexing_results_output_field, ## indexing results output field
self.debug_log_output_field_indexing_tab]) ## debug log on indexing tab
##-------------------start-of-clear_preprocessing_tab()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.preprocessing_clear_button.click(clear_preprocessing_tab,
inputs=[],
outputs=[
self.input_txt_file_preprocessing, ## input txt file
self.input_json_file_preprocessing, ## input json file
self.input_text_kairyou, ## input text
self.preprocessing_output_field, ## preprocessed text output field
self.preprocessing_results_output_field, ## preprocessing results output field
self.debug_log_output_field_preprocess_tab])## debug log on preprocess tab
##-------------------start-of-clear_button_translator_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.translator_clear_button.click(clear_translator_tab,
inputs=[],
outputs=[
self.input_txt_file_translator, ## input txt file
self.input_text_translator, ## input text
self.input_translation_rules_file, ## Translation Settings File
self.translator_translated_text_output_field, ## translation output field
self.translator_je_check_text_output_field, ## je check text field on translator tab
self.translator_debug_log_output_field], ## debug log on translator tab
cancels=translator_translate_process)
##-------------------start-of-clear_log_button_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.logging_clear_logs_button.click(clear_log_button_click,
inputs=[],
outputs=[
self.logging_tab_debug_log_output_field,
self.logging_tab_error_log_output_field])
##-------------------start-of-apply_new_translator_settings()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.translation_settings_apply_changes_button.click(apply_new_translator_settings,
inputs=[
self.input_translation_rules_file, ## Translation Settings File
self.prompt_assembly_mode_input_field, ## prompt assembly mode input field
self.number_of_lines_per_batch_input_field, ## num lines input field
self.sentence_fragmenter_mode_input_field, ## sentence fragmenter mode input field
self.je_check_mode_input_field, ## je check mode input field
self.number_of_malformed_batch_retries_input_field, ## num malformed batch retries input field
self.batch_retry_timeout_input_field, ## batch retry timeout input field
self.number_of_concurrent_batches_input_field, ## num concurrent batches input field
self.openai_model_input_field, ## openai model input field
self.openai_system_message_input_field, ## openai system message input field
self.openai_temperature_input_field, ## openai temperature input field
self.openai_top_p_input_field, ## openai top p input field
self.openai_n_input_field, ## openai n input field
self.openai_stream_input_field, ## openai stream input field
self.openai_stop_input_field, ## openai stop input field
self.openai_logit_bias_input_field, ## openai logit bias input field
self.openai_max_tokens_input_field, ## openai max tokens input field
self.openai_presence_penalty_input_field, ## openai presence penalty input field
self.openai_frequency_penalty_input_field, ## openai frequency penalty input field
self.gemini_model_input_field, ## gemini model input field
self.gemini_prompt_input_field, ## gemini prompt input field
self.gemini_temperature_input_field, ## gemini temperature input field
self.gemini_top_p_input_field, ## gemini top p input field
self.gemini_top_k_input_field, ## gemini top k input field
self.gemini_candidate_count_input_field, ## gemini candidate count input field
self.gemini_stream_input_field, ## gemini stream input field
self.gemini_stop_sequences_input_field, ## gemini stop sequences input field
self.gemini_max_output_tokens_input_field, ## gemini max output tokens input field
self.deepl_context_input_field, ## deepl context input field
self.deepl_split_sentences_input_field, ## deepl split sentences input field
self.deepl_preserve_formatting_input_field, ## deepl preserve formatting input field
self.deepl_formality_input_field], ## deepl formality input field
outputs=[])
##-------------------start-of-reset_to_default_translation_settings_button_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.translation_settings_reset_to_default_button.click(reset_to_default_translation_settings,
inputs=[self.input_translation_rules_file],
outputs=[
self.prompt_assembly_mode_input_field, ## prompt assembly mode input field
self.number_of_lines_per_batch_input_field, ## num lines input field
self.sentence_fragmenter_mode_input_field, ## sentence fragmenter mode input field
self.je_check_mode_input_field, ## je check mode input field
self.number_of_malformed_batch_retries_input_field, ## num malformed batch retries input field
self.batch_retry_timeout_input_field, ## batch retry timeout input field
self.number_of_concurrent_batches_input_field, ## num concurrent batches input field
self.openai_model_input_field, ## openai model input field
self.openai_system_message_input_field, ## openai system message input field
self.openai_temperature_input_field, ## openai temperature input field
self.openai_top_p_input_field, ## openai top p input field
self.openai_n_input_field, ## openai n input field
self.openai_stream_input_field, ## openai stream input field
self.openai_stop_input_field, ## openai stop input field
self.openai_logit_bias_input_field, ## openai logit bias input field
self.openai_max_tokens_input_field, ## openai max tokens input field
self.openai_presence_penalty_input_field, ## openai presence penalty input field
self.openai_frequency_penalty_input_field, ## openai frequency penalty input field
self.gemini_model_input_field, ## gemini model input field
self.gemini_prompt_input_field, ## gemini prompt input field
self.gemini_temperature_input_field, ## gemini temperature input field
self.gemini_top_p_input_field, ## gemini top p input field
self.gemini_top_k_input_field, ## gemini top k input field
self.gemini_candidate_count_input_field, ## gemini candidate count input field
self.gemini_stream_input_field, ## gemini stream input field
self.gemini_stop_sequences_input_field, ## gemini stop sequences input field
self.gemini_max_output_tokens_input_field, ## gemini max output tokens input field
self.deepl_context_input_field, ## deepl context input field
self.deepl_split_sentences_input_field, ## deepl split sentences input field
self.deepl_preserve_formatting_input_field, ## deepl preserve formatting input field
self.deepl_formality_input_field])
##-------------------start-of-discard_changes_button_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.translation_settings_discard_changes_button.click(refresh_translation_settings_fields,
inputs=[self.input_translation_rules_file],
outputs=[
self.prompt_assembly_mode_input_field, ## prompt assembly mode input field
self.number_of_lines_per_batch_input_field, ## num lines input field
self.sentence_fragmenter_mode_input_field, ## sentence fragmenter mode input field
self.je_check_mode_input_field, ## je check mode input field
self.number_of_malformed_batch_retries_input_field, ## num malformed batch retries input field
self.batch_retry_timeout_input_field, ## batch retry timeout input field
self.number_of_concurrent_batches_input_field, ## num concurrent batches input field
self.openai_model_input_field, ## openai model input field
self.openai_system_message_input_field, ## openai system message input field
self.openai_temperature_input_field, ## openai temperature input field
self.openai_top_p_input_field, ## openai top p input field
self.openai_n_input_field, ## openai n input field
self.openai_stream_input_field, ## openai stream input field
self.openai_stop_input_field, ## openai stop input field
self.openai_logit_bias_input_field, ## openai logit bias input field
self.openai_max_tokens_input_field, ## openai max tokens input field
self.openai_presence_penalty_input_field, ## openai presence penalty input field
self.openai_frequency_penalty_input_field, ## openai frequency penalty input field
self.gemini_model_input_field, ## gemini model input field
self.gemini_prompt_input_field, ## gemini prompt input field
self.gemini_temperature_input_field, ## gemini temperature input field
self.gemini_top_p_input_field, ## gemini top p input field
self.gemini_top_k_input_field, ## gemini top k input field
self.gemini_candidate_count_input_field, ## gemini candidate count input field
self.gemini_stream_input_field, ## gemini stream input field
self.gemini_stop_sequences_input_field, ## gemini stop sequences input field
self.gemini_max_output_tokens_input_field, ## gemini max output tokens input field
self.deepl_context_input_field, ## deepl context input field
self.deepl_split_sentences_input_field, ## deepl split sentences input field
self.deepl_preserve_formatting_input_field, ## deepl preserve formatting input field
self.deepl_formality_input_field]) ## deepl formality input field
##-------------------start-of-input_translator_rules_file_upload()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.input_translation_rules_file.upload(refresh_translation_settings_fields,
inputs=[self.input_translation_rules_file],
outputs=[
self.prompt_assembly_mode_input_field, ## prompt assembly mode input field
self.number_of_lines_per_batch_input_field, ## num lines input field
self.sentence_fragmenter_mode_input_field, ## sentence fragmenter mode input field
self.je_check_mode_input_field, ## je check mode input field
self.number_of_malformed_batch_retries_input_field, ## num malformed batch retries input field
self.batch_retry_timeout_input_field, ## batch retry timeout input field
self.number_of_concurrent_batches_input_field, ## num concurrent batches input field
self.openai_model_input_field, ## openai model input field
self.openai_system_message_input_field, ## openai system message input field
self.openai_temperature_input_field, ## openai temperature input field
self.openai_top_p_input_field, ## openai top p input field
self.openai_n_input_field, ## openai n input field
self.openai_stream_input_field, ## openai stream input field
self.openai_stop_input_field, ## openai stop input field
self.openai_logit_bias_input_field, ## openai logit bias input field
self.openai_max_tokens_input_field, ## openai max tokens input field
self.openai_presence_penalty_input_field, ## openai presence penalty input field
self.openai_frequency_penalty_input_field, ## openai frequency penalty input field
self.gemini_model_input_field, ## gemini model input field
self.gemini_prompt_input_field, ## gemini prompt input field
self.gemini_temperature_input_field, ## gemini temperature input field
self.gemini_top_p_input_field, ## gemini top p input field
self.gemini_top_k_input_field, ## gemini top k input field
self.gemini_candidate_count_input_field, ## gemini candidate count input field
self.gemini_stream_input_field, ## gemini stream input field
self.gemini_stop_sequences_input_field, ## gemini stop sequences input field
self.gemini_max_output_tokens_input_field, ## gemini max output tokens input field
self.deepl_context_input_field, ## deepl context input field
self.deepl_split_sentences_input_field, ## deepl split sentences input field
self.deepl_preserve_formatting_input_field, ## deepl preserve formatting input field
self.deepl_formality_input_field]) ## deepl formality input field
self.input_translation_rules_file.clear(clear_translation_settings_input_fields,
inputs=[],
outputs=[
self.input_translation_rules_file, ## Translation Settings File
self.prompt_assembly_mode_input_field, ## prompt assembly mode input field
self.number_of_lines_per_batch_input_field, ## num lines input field
self.sentence_fragmenter_mode_input_field, ## sentence fragmenter mode input field
self.je_check_mode_input_field, ## je check mode input field
self.number_of_malformed_batch_retries_input_field, ## num malformed batch retries input field
self.batch_retry_timeout_input_field, ## batch retry timeout input field
self.number_of_concurrent_batches_input_field, ## num concurrent batches input field
self.openai_model_input_field, ## openai model input field
self.openai_system_message_input_field, ## openai system message input field
self.openai_temperature_input_field, ## openai temperature input field
self.openai_top_p_input_field, ## openai top p input field
self.openai_n_input_field, ## openai n input field
self.openai_stream_input_field, ## openai stream input field
self.openai_stop_input_field, ## openai stop input field
self.openai_logit_bias_input_field, ## openai logit bias input field
self.openai_max_tokens_input_field, ## openai max tokens input field
self.openai_presence_penalty_input_field, ## openai presence penalty input field
self.openai_frequency_penalty_input_field, ## openai frequency penalty input field
self.gemini_model_input_field, ## gemini model input field
self.gemini_prompt_input_field, ## gemini prompt input field
self.gemini_temperature_input_field, ## gemini temperature input field
self.gemini_top_p_input_field, ## gemini top p input field
self.gemini_top_k_input_field, ## gemini top k input field
self.gemini_candidate_count_input_field, ## gemini candidate count input field
self.gemini_stream_input_field, ## gemini stream input field
self.gemini_stop_sequences_input_field, ## gemini stop sequences input field
self.gemini_max_output_tokens_input_field, ## gemini max output tokens input field
self.deepl_context_input_field, ## deepl context input field
self.deepl_split_sentences_input_field, ## deepl split sentences input field
self.deepl_preserve_formatting_input_field, ## deepl preserve formatting input field
self.deepl_formality_input_field]) ## deepl formality input field
##-------------------start-of-logging_tab.select()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.logging_tab.select(fetch_debug_log_content,
inputs=[],
outputs=[self.logging_tab_debug_log_output_field, self.logging_tab_error_log_output_field])
##-------------------start-of-translator_api_key_input.change()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.llm_option_dropdown.change(switch_translator_api_key_type,
inputs=[self.llm_option_dropdown],
outputs=[self.translator_api_key_input])
##-------------------start-of-save_to_file_indexed_text_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_indexed_text.click(lambda text: text, ## save text as is
inputs=[self.indexing_output_field], ## input text to save
outputs=[], ## no outputs
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "indexed_text.txt")
)
##-------------------start-of-save_to_file_indexing_results_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_indexing_results.click(lambda text: text, ## save text as is
inputs=[self.indexing_results_output_field], ## input text to save
outputs=[], ## no outputs
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "indexing_results.txt")
)
##-------------------start-of-save_to_file_indexing_debug_log_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_debug_log_indexing_tab.click(lambda text: text, ## save text as is
inputs=[self.debug_log_output_field_indexing_tab], ## input text to save
outputs=[], ## no outputs
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "indexing_debug_log.txt")
)
##-------------------start-of-save_to_file_preprocessing_results_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_preprocessed_text.click(lambda text: text, ## save text as is
inputs=[self.preprocessing_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "preprocessed_text.txt")
)
##-------------------start-of-save_to_file_preprocessing_results_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_preprocessing_results.click(lambda text: text, ## save text as is
inputs=[self.preprocessing_results_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "preprocessing_results.txt")
)
##-------------------start-of-save_to_file_debug_log_processing_tab_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_debug_log_preprocessing_tab.click(lambda text: text, ## save text as is
inputs=[self.debug_log_output_field_preprocess_tab],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "preprocessing_debug_log_.txt")
)
##-------------------start-of-save_to_file_translator_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_translator_translated_text.click(lambda text: text, ## save text as is
inputs=[self.translator_translated_text_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "translated_text_translator.txt")
)
##-------------------start-of-save_to_file_je_check_text_translator_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_translator_je_check_text.click(lambda text: text, ## save text as is
inputs=[self.translator_je_check_text_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "je_check_text_translator.txt")
)
##-------------------start-of-save_to_file_debug_log_translator_tab_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_debug_log_translator_tab.click(lambda text: text, ## save text as is
inputs=[self.translator_debug_log_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "debug_log_translator.txt")
)
##-------------------start-of-save_to_file_debug_log_logging_tab_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_debug_log_logging_tab.click(lambda text: text, ## save text as is
inputs=[self.logging_tab_debug_log_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "debug_log_all.txt")
)
##-------------------start-of-save_to_file_error_log_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.save_to_file_error_log_logging_tab.click(lambda text: text, ## save text as is
inputs=[self.logging_tab_error_log_output_field],
outputs=[],
## javascript code that allows us to save textbox contents to a file
js=(self.js_save_to_file).replace("downloaded_text.txt", "error_log.txt")
)
##-------------------start-of-send_to_x_click()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
self.send_to_kairyou_button.click(fn=send_to_kairyou_button,
inputs=[self.indexing_output_field],
outputs=[self.input_text_kairyou])
self.send_to_translator_button.click(fn=send_to_translator_button,
inputs=[self.preprocessing_output_field],
outputs=[self.input_text_translator])
##-------------------start-of-launch()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
def launch(self):
"""
Launches the GUI.
"""
Kudasai.boot()
GuiJsonUtil.current_translation_settings = JsonHandler.current_translation_settings
self.build_gui()
self.gui.queue().launch(inbrowser=True, show_error=True, show_api=False, favicon_path=FileEnsurer.favicon_path)
##-------------------start-of-main()---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
if(__name__ == '__main__'):
try:
kudasai_gui = KudasaiGUI()
kudasai_gui.launch()
except Exception as e:
FileEnsurer.handle_critical_exception(e)