2023-05-05 23:14:56 -03:00
|
|
|
import os
|
|
|
|
import re
|
2023-06-13 20:34:35 -03:00
|
|
|
from datetime import datetime
|
2023-05-05 23:14:56 -03:00
|
|
|
from pathlib import Path
|
|
|
|
|
2023-09-25 19:48:30 -07:00
|
|
|
from modules import github, shared
|
2023-06-11 12:19:18 -03:00
|
|
|
from modules.logging_colors import logger
|
|
|
|
|
|
|
|
|
2023-07-04 00:03:30 -03:00
|
|
|
# Helper function to get multiple values from shared.gradio
|
|
|
|
def gradio(*keys):
|
2023-08-13 01:12:15 -03:00
|
|
|
if len(keys) == 1 and type(keys[0]) in [list, tuple]:
|
2023-07-04 00:03:30 -03:00
|
|
|
keys = keys[0]
|
|
|
|
|
|
|
|
return [shared.gradio[k] for k in keys]
|
|
|
|
|
|
|
|
|
2023-06-11 12:19:18 -03:00
|
|
|
def save_file(fname, contents):
|
|
|
|
if fname == '':
|
|
|
|
logger.error('File name is empty!')
|
|
|
|
return
|
|
|
|
|
|
|
|
root_folder = Path(__file__).resolve().parent.parent
|
|
|
|
abs_path = Path(fname).resolve()
|
|
|
|
rel_path = abs_path.relative_to(root_folder)
|
|
|
|
if rel_path.parts[0] == '..':
|
|
|
|
logger.error(f'Invalid file path: {fname}')
|
|
|
|
return
|
|
|
|
|
|
|
|
with open(abs_path, 'w', encoding='utf-8') as f:
|
|
|
|
f.write(contents)
|
|
|
|
|
|
|
|
logger.info(f'Saved {abs_path}.')
|
|
|
|
|
|
|
|
|
|
|
|
def delete_file(fname):
|
|
|
|
if fname == '':
|
|
|
|
logger.error('File name is empty!')
|
|
|
|
return
|
|
|
|
|
|
|
|
root_folder = Path(__file__).resolve().parent.parent
|
|
|
|
abs_path = Path(fname).resolve()
|
|
|
|
rel_path = abs_path.relative_to(root_folder)
|
|
|
|
if rel_path.parts[0] == '..':
|
|
|
|
logger.error(f'Invalid file path: {fname}')
|
|
|
|
return
|
|
|
|
|
|
|
|
if abs_path.exists():
|
|
|
|
abs_path.unlink()
|
|
|
|
logger.info(f'Deleted {fname}.')
|
2023-05-05 23:14:56 -03:00
|
|
|
|
|
|
|
|
2023-06-13 20:34:35 -03:00
|
|
|
def current_time():
|
|
|
|
return f"{datetime.now().strftime('%Y-%m-%d-%H%M%S')}"
|
|
|
|
|
|
|
|
|
2023-05-05 23:14:56 -03:00
|
|
|
def atoi(text):
|
|
|
|
return int(text) if text.isdigit() else text.lower()
|
|
|
|
|
|
|
|
|
2023-05-10 01:34:04 -03:00
|
|
|
# Replace multiple string pairs in a string
|
|
|
|
def replace_all(text, dic):
|
|
|
|
for i, j in dic.items():
|
|
|
|
text = text.replace(i, j)
|
|
|
|
|
|
|
|
return text
|
|
|
|
|
|
|
|
|
2023-05-05 23:14:56 -03:00
|
|
|
def natural_keys(text):
|
|
|
|
return [atoi(c) for c in re.split(r'(\d+)', text)]
|
|
|
|
|
|
|
|
|
|
|
|
def get_available_models():
|
2023-10-10 22:20:49 -03:00
|
|
|
model_list = ['None']
|
2023-08-10 10:01:12 -07:00
|
|
|
for item in list(Path(f'{shared.args.model_dir}/').glob('*')):
|
|
|
|
if not item.name.endswith(('.txt', '-np', '.pt', '.json', '.yaml', '.py')) and 'llama-tokenizer' not in item.name:
|
|
|
|
model_list.append(re.sub('.pth$', '', item.name))
|
|
|
|
|
|
|
|
return sorted(model_list, key=natural_keys)
|
2023-05-05 23:14:56 -03:00
|
|
|
|
|
|
|
|
|
|
|
def get_available_presets():
|
2023-05-28 22:34:12 -03:00
|
|
|
return sorted(set((k.stem for k in Path('presets').glob('*.yaml'))), key=natural_keys)
|
2023-05-05 23:14:56 -03:00
|
|
|
|
|
|
|
|
|
|
|
def get_available_prompts():
|
|
|
|
prompts = []
|
2023-05-10 02:24:09 -03:00
|
|
|
files = set((k.stem for k in Path('prompts').glob('*.txt')))
|
|
|
|
prompts += sorted([k for k in files if re.match('^[0-9]', k)], key=natural_keys, reverse=True)
|
|
|
|
prompts += sorted([k for k in files if re.match('^[^0-9]', k)], key=natural_keys)
|
2023-05-05 23:14:56 -03:00
|
|
|
prompts += ['None']
|
|
|
|
return prompts
|
|
|
|
|
|
|
|
|
|
|
|
def get_available_characters():
|
|
|
|
paths = (x for x in Path('characters').iterdir() if x.suffix in ('.json', '.yaml', '.yml'))
|
2023-09-21 17:19:32 -03:00
|
|
|
return sorted(set((k.stem for k in paths)), key=natural_keys)
|
2023-05-05 23:14:56 -03:00
|
|
|
|
|
|
|
|
|
|
|
def get_available_instruction_templates():
|
2023-08-06 17:50:07 -07:00
|
|
|
path = "instruction-templates"
|
2023-05-05 23:14:56 -03:00
|
|
|
paths = []
|
|
|
|
if os.path.exists(path):
|
|
|
|
paths = (x for x in Path(path).iterdir() if x.suffix in ('.json', '.yaml', '.yml'))
|
2023-05-10 01:34:04 -03:00
|
|
|
|
2023-05-05 23:14:56 -03:00
|
|
|
return ['None'] + sorted(set((k.stem for k in paths)), key=natural_keys)
|
|
|
|
|
|
|
|
|
|
|
|
def get_available_extensions():
|
2023-09-25 19:48:30 -07:00
|
|
|
extensions = sorted(set(map(lambda x: x.parts[1], Path('extensions').glob('*/script.py'))), key=natural_keys)
|
|
|
|
extensions = [v for v in extensions if v not in github.new_extensions]
|
|
|
|
return extensions
|
2023-05-05 23:14:56 -03:00
|
|
|
|
|
|
|
|
|
|
|
def get_available_loras():
|
2023-10-10 22:20:49 -03:00
|
|
|
return ['None'] + sorted([item.name for item in list(Path(shared.args.lora_dir).glob('*')) if not item.name.endswith(('.txt', '-np', '.pt', '.json'))], key=natural_keys)
|
2023-05-05 23:14:56 -03:00
|
|
|
|
|
|
|
|
|
|
|
def get_datasets(path: str, ext: str):
|
2023-07-12 17:44:30 +03:00
|
|
|
# include subdirectories for raw txt files to allow training from a subdirectory of txt files
|
|
|
|
if ext == "txt":
|
2023-07-12 11:33:25 -07:00
|
|
|
return ['None'] + sorted(set([k.stem for k in list(Path(path).glob('txt')) + list(Path(path).glob('*/')) if k.stem != 'put-trainer-datasets-here']), key=natural_keys)
|
2023-07-12 17:44:30 +03:00
|
|
|
|
2023-05-05 23:14:56 -03:00
|
|
|
return ['None'] + sorted(set([k.stem for k in Path(path).glob(f'*.{ext}') if k.stem != 'put-trainer-datasets-here']), key=natural_keys)
|
2023-05-08 12:35:03 -03:00
|
|
|
|
|
|
|
|
|
|
|
def get_available_chat_styles():
|
|
|
|
return sorted(set(('-'.join(k.stem.split('-')[1:]) for k in Path('css').glob('chat_style*.css'))), key=natural_keys)
|
2023-09-24 07:08:41 -07:00
|
|
|
|
|
|
|
|
|
|
|
def get_available_grammars():
|
|
|
|
return ['None'] + sorted([item.name for item in list(Path('grammars').glob('*.gbnf'))], key=natural_keys)
|