From 9331ab4798f392ee0634a4194a6bae370afd435f Mon Sep 17 00:00:00 2001 From: oobabooga <112222186+oobabooga@users.noreply.github.com> Date: Mon, 11 Sep 2023 18:49:30 -0300 Subject: [PATCH] Read GGUF metadata (#3873) --- extensions/api/blocking_api.py | 7 +-- extensions/openai/models.py | 12 ++--- modules/evaluate.py | 7 +-- modules/metadata_gguf.py | 84 ++++++++++++++++++++++++++++++++++ modules/models.py | 10 ++-- modules/models_settings.py | 52 +++++++++++++++++---- modules/ui_model_menu.py | 4 +- server.py | 30 ++++++------ 8 files changed, 154 insertions(+), 52 deletions(-) create mode 100644 modules/metadata_gguf.py diff --git a/extensions/api/blocking_api.py b/extensions/api/blocking_api.py index 6b28205af3..8a643c1537 100644 --- a/extensions/api/blocking_api.py +++ b/extensions/api/blocking_api.py @@ -7,10 +7,7 @@ from modules.chat import generate_chat_reply from modules.LoRA import add_lora_to_model from modules.models import load_model, unload_model -from modules.models_settings import ( - get_model_settings_from_yamls, - update_model_parameters -) +from modules.models_settings import get_model_metadata, update_model_parameters from modules.text_generation import ( encode, generate_reply, @@ -132,7 +129,7 @@ def do_POST(self): shared.model_name = model_name unload_model() - model_settings = get_model_settings_from_yamls(shared.model_name) + model_settings = get_model_metadata(shared.model_name) shared.settings.update(model_settings) update_model_parameters(model_settings, initial=True) diff --git a/extensions/openai/models.py b/extensions/openai/models.py index 035996f6cc..25263e4ba6 100644 --- a/extensions/openai/models.py +++ b/extensions/openai/models.py @@ -1,11 +1,9 @@ -from modules import shared -from modules.utils import get_available_models -from modules.models import load_model, unload_model -from modules.models_settings import (get_model_settings_from_yamls, - update_model_parameters) - from extensions.openai.embeddings import get_embeddings_model_name from extensions.openai.errors import * +from modules import shared +from modules.models import load_model, unload_model +from modules.models_settings import get_model_metadata, update_model_parameters +from modules.utils import get_available_models def get_current_model_list() -> list: @@ -33,7 +31,7 @@ def load_model(model_name: str) -> dict: shared.model_name = model_name unload_model() - model_settings = get_model_settings_from_yamls(shared.model_name) + model_settings = get_model_metadata(shared.model_name) shared.settings.update(model_settings) update_model_parameters(model_settings, initial=True) diff --git a/modules/evaluate.py b/modules/evaluate.py index d94863d978..609419c818 100644 --- a/modules/evaluate.py +++ b/modules/evaluate.py @@ -8,10 +8,7 @@ from modules import shared from modules.models import load_model, unload_model -from modules.models_settings import ( - get_model_settings_from_yamls, - update_model_parameters -) +from modules.models_settings import get_model_metadata, update_model_parameters from modules.text_generation import encode @@ -69,7 +66,7 @@ def calculate_perplexity(models, input_dataset, stride, _max_length): if model != 'current model': try: yield cumulative_log + f"Loading {model}...\n\n" - model_settings = get_model_settings_from_yamls(model) + model_settings = get_model_metadata(model) shared.settings.update(model_settings) # hijacking the interface defaults update_model_parameters(model_settings) # hijacking the command-line arguments shared.model_name = model diff --git a/modules/metadata_gguf.py b/modules/metadata_gguf.py new file mode 100644 index 0000000000..c65f673957 --- /dev/null +++ b/modules/metadata_gguf.py @@ -0,0 +1,84 @@ +import struct +from enum import IntEnum + + +class GGUFValueType(IntEnum): + UINT8 = 0 + INT8 = 1 + UINT16 = 2 + INT16 = 3 + UINT32 = 4 + INT32 = 5 + FLOAT32 = 6 + BOOL = 7 + STRING = 8 + ARRAY = 9 + UINT64 = 10 + INT64 = 11 + FLOAT64 = 12 + + +_simple_value_packing = { + GGUFValueType.UINT8: " 0: + loader = 'AutoGPTQ' + + model_settings['loader'] = loader + + # Read GGUF metadata + if model_settings['loader'] in ['llama.cpp', 'llamacpp_HF', 'ctransformers']: + path = Path(f'{shared.args.model_dir}/{model}') + if path.is_file(): + model_file = path + else: + model_file = list(path.glob('*.gguf'))[0] + + metadata = metadata_gguf.load_metadata(model_file) + if 'llama.context_length' in metadata: + model_settings['n_ctx'] = metadata['llama.context_length'] + return model_settings -def infer_loader(model_name): +def infer_loader(model_name, model_settings): path_to_model = Path(f'{shared.args.model_dir}/{model_name}') - model_settings = get_model_settings_from_yamls(model_name) if not path_to_model.exists(): loader = None elif Path(f'{shared.args.model_dir}/{model_name}/quantize_config.json').exists() or ('wbits' in model_settings and type(model_settings['wbits']) is int and model_settings['wbits'] > 0): @@ -85,11 +119,9 @@ def update_model_parameters(state, initial=False): # UI: update the state variable with the model settings def apply_model_settings_to_state(model, state): - model_settings = get_model_settings_from_yamls(model) - if 'loader' not in model_settings: - loader = infer_loader(model) - if 'wbits' in model_settings and type(model_settings['wbits']) is int and model_settings['wbits'] > 0: - loader = 'AutoGPTQ' + model_settings = get_model_metadata(model) + if 'loader' in model_settings: + loader = model_settings.pop('loader') # If the user is using an alternative loader for the same model type, let them keep using it if not (loader == 'AutoGPTQ' and state['loader'] in ['GPTQ-for-LLaMa', 'ExLlama', 'ExLlama_HF']) and not (loader == 'llama.cpp' and state['loader'] in ['llamacpp_HF', 'ctransformers']): diff --git a/modules/ui_model_menu.py b/modules/ui_model_menu.py index 423911d570..6d74f247d9 100644 --- a/modules/ui_model_menu.py +++ b/modules/ui_model_menu.py @@ -15,7 +15,7 @@ from modules.models import load_model, unload_model from modules.models_settings import ( apply_model_settings_to_state, - get_model_settings_from_yamls, + get_model_metadata, save_model_settings, update_model_parameters ) @@ -196,7 +196,7 @@ def load_model_wrapper(selected_model, loader, autoload=False): if shared.model is not None: output = f"Successfully loaded `{selected_model}`." - settings = get_model_settings_from_yamls(selected_model) + settings = get_model_metadata(selected_model) if 'instruction_template' in settings: output += '\n\nIt seems to be an instruction-following model with template "{}". In the chat tab, instruct or chat-instruct modes should be used.'.format(settings['instruction_template']) diff --git a/server.py b/server.py index 0beed5ad6f..597b47be4b 100644 --- a/server.py +++ b/server.py @@ -1,8 +1,8 @@ import os import warnings -from modules.logging_colors import logger from modules.block_requests import OpenMonkeyPatch, RequestBlocker +from modules.logging_colors import logger os.environ['GRADIO_ANALYTICS_ENABLED'] = 'False' os.environ['BITSANDBYTES_NOWELCOME'] = '1' @@ -12,6 +12,7 @@ import gradio as gr import matplotlib + matplotlib.use('Agg') # This fixes LaTeX rendering on some systems import json @@ -37,13 +38,14 @@ ui_notebook, ui_parameters, ui_session, - utils, + utils ) from modules.extensions import apply_extensions from modules.LoRA import add_lora_to_model from modules.models import load_model from modules.models_settings import ( - get_model_settings_from_yamls, + get_fallback_settings, + get_model_metadata, update_model_parameters ) from modules.utils import gradio @@ -169,17 +171,7 @@ def create_interface(): shared.settings.update(new_settings) # Fallback settings for models - shared.model_config['.*'] = { - 'wbits': 'None', - 'model_type': 'None', - 'groupsize': 'None', - 'pre_layer': 0, - 'skip_special_tokens': shared.settings['skip_special_tokens'], - 'custom_stopping_strings': shared.settings['custom_stopping_strings'], - 'truncation_length': shared.settings['truncation_length'], - 'rope_freq_base': 0, - } - + shared.model_config['.*'] = get_fallback_settings() shared.model_config.move_to_end('.*', last=False) # Move to the beginning # Activate the extensions listed on settings.yaml @@ -213,12 +205,18 @@ def create_interface(): # If any model has been selected, load it if shared.model_name != 'None': - model_settings = get_model_settings_from_yamls(shared.model_name) + p = Path(shared.model_name) + if p.exists(): + model_name = p.parts[-1] + else: + model_name = shared.model_name + + model_settings = get_model_metadata(model_name) shared.settings.update(model_settings) # hijacking the interface defaults update_model_parameters(model_settings, initial=True) # hijacking the command-line arguments # Load the model - shared.model, shared.tokenizer = load_model(shared.model_name) + shared.model, shared.tokenizer = load_model(model_name) if shared.args.lora: add_lora_to_model(shared.args.lora)