From ce57885f54a131cc0bbed54f3e480501e16f4930 Mon Sep 17 00:00:00 2001 From: Oleg Pipikin Date: Mon, 29 Jul 2024 09:06:28 +0000 Subject: [PATCH 1/3] Fix to throw exception in case of empty chat template in chat scenario --- src/cpp/include/openvino/genai/tokenizer.hpp | 6 ++++++ src/cpp/src/llm_pipeline.cpp | 5 +++++ src/cpp/src/tokenizer.cpp | 9 ++++++++- 3 files changed, 19 insertions(+), 1 deletion(-) diff --git a/src/cpp/include/openvino/genai/tokenizer.hpp b/src/cpp/include/openvino/genai/tokenizer.hpp index 425c30128b..0b71ccfb9e 100644 --- a/src/cpp/include/openvino/genai/tokenizer.hpp +++ b/src/cpp/include/openvino/genai/tokenizer.hpp @@ -83,6 +83,12 @@ class OPENVINO_GENAI_EXPORTS Tokenizer { bool add_generation_prompt, const std::string& chat_template="") const; + /** + * @brief Returns true if chat tempate exists and is ready to be applied; false otherwise + * @return is chat tempate ready + */ + bool is_chat_template_ready() const; + // information about , tokens should be public, // they are used at least in StreamerBase descendants int64_t get_bos_token_id() const; diff --git a/src/cpp/src/llm_pipeline.cpp b/src/cpp/src/llm_pipeline.cpp index 507d988a6a..e86ada33c5 100644 --- a/src/cpp/src/llm_pipeline.cpp +++ b/src/cpp/src/llm_pipeline.cpp @@ -266,11 +266,16 @@ class StatefulLLMPipeline final : public LLMPipelineImplBase { m_history = {}; m_templated_chat_history = ""; } + OPENVINO_ASSERT(m_tokenizer.is_chat_template_ready(), + "There is no existing chat template for actual model. LLMPipeline cannot work in chat mode." + " Please add chat template to tokenizer_config.json or use another model."); + if (system_message.empty()) return; m_history.push_back({{"role", "system"}, {"content", system_message}}); constexpr bool add_generation_prompt = false; + m_templated_chat_history = m_tokenizer.apply_chat_template(m_history, add_generation_prompt); } diff --git a/src/cpp/src/tokenizer.cpp b/src/cpp/src/tokenizer.cpp index b1e36033ee..98a9d1a027 100644 --- a/src/cpp/src/tokenizer.cpp +++ b/src/cpp/src/tokenizer.cpp @@ -434,7 +434,9 @@ class Tokenizer::TokenizerImpl { } } - + bool is_chat_template_ready() { + return !m_chat_template.empty(); + } }; Tokenizer::Tokenizer(const std::string& tokenizer_path, const ov::AnyMap& plugin_config) { @@ -500,6 +502,11 @@ std::string Tokenizer::apply_chat_template(ChatHistory history, return m_pimpl->apply_chat_template(history, add_generation_prompt, chat_template); } +bool Tokenizer::is_chat_template_ready() const { + return m_pimpl->is_chat_template_ready(); +}; + + Tokenizer::~Tokenizer() = default; } // namespace genai } // namespace ov From bf4f9f331f5bb4f4b9f52f3cc097500f6809c970 Mon Sep 17 00:00:00 2001 From: Oleg Pipikin Date: Mon, 29 Jul 2024 16:32:28 +0000 Subject: [PATCH 2/3] Apply comments --- samples/cpp/chat_sample/README.md | 8 ++++++++ samples/python/chat_sample/README.md | 10 ++++++++++ src/cpp/include/openvino/genai/tokenizer.hpp | 6 ------ src/cpp/src/llm_pipeline.cpp | 4 ---- src/cpp/src/tokenizer.cpp | 14 +++++--------- 5 files changed, 23 insertions(+), 19 deletions(-) diff --git a/samples/cpp/chat_sample/README.md b/samples/cpp/chat_sample/README.md index a2eccb4d3d..46d9358180 100644 --- a/samples/cpp/chat_sample/README.md +++ b/samples/cpp/chat_sample/README.md @@ -34,3 +34,11 @@ UnicodeEncodeError: 'charmap' codec can't encode character '\u25aa' in position If you encounter the error described in the example when sample is printing output to the Windows console, it is likely due to the default Windows encoding not supporting certain Unicode characters. To resolve this: 1. Enable Unicode characters for Windows cmd - open `Region` settings from `Control panel`. `Administrative`->`Change system locale`->`Beta: Use Unicode UTF-8 for worldwide language support`->`OK`. Reboot. 2. Enable UTF-8 mode by setting environment variable `PYTHONIOENCODING="utf8"`. + +#### Missing chat template + +If you encounter an exception indicating a missing "chat template" when launching the `ov::genai::LLMPipeline` in chat mode, it likely means the model was not tuned for chat functionality. To resolve this, manually add the chat template to tokenizer_config.json of your model. +The following template can be used as a default, but it may not work properly with every model: +``` +"chat_template": "{% for message in messages %}{% if (message['role'] == 'user') %}{{'<|im_start|>user\n' + message['content'] + '<|im_end|>\n<|im_start|>assistant\n'}}{% elif (message['role'] == 'assistant') %}{{message['content'] + '<|im_end|>\n'}}{% endif %}{% endfor %}", +``` diff --git a/samples/python/chat_sample/README.md b/samples/python/chat_sample/README.md index 983789d0eb..a137cddce8 100644 --- a/samples/python/chat_sample/README.md +++ b/samples/python/chat_sample/README.md @@ -22,3 +22,13 @@ To enable Unicode characters for Windows cmd open `Region` settings from `Contro Discrete GPUs (dGPUs) usually provide better performance compared to CPUs. It is recommended to run larger models on a dGPU with 32GB+ RAM. For example, the model meta-llama/Llama-2-13b-chat-hf can benefit from being run on a dGPU. Modify the source code to change the device for inference to the GPU. See https://github.com/openvinotoolkit/openvino.genai/blob/master/src/README.md#supported-models for the list of supported models. + + +## Troubleshooting +### Missing chat template + +If you encounter an exception indicating a missing "chat template" when launching the `ov::genai::LLMPipeline` in chat mode, it likely means the model was not tuned for chat functionality. To resolve this, manually add the chat template to tokenizer_config.json of your model. +The following template can be used as a default, but it may not work properly with every model: +``` +"chat_template": "{% for message in messages %}{% if (message['role'] == 'user') %}{{'<|im_start|>user\n' + message['content'] + '<|im_end|>\n<|im_start|>assistant\n'}}{% elif (message['role'] == 'assistant') %}{{message['content'] + '<|im_end|>\n'}}{% endif %}{% endfor %}", +``` \ No newline at end of file diff --git a/src/cpp/include/openvino/genai/tokenizer.hpp b/src/cpp/include/openvino/genai/tokenizer.hpp index 0b71ccfb9e..425c30128b 100644 --- a/src/cpp/include/openvino/genai/tokenizer.hpp +++ b/src/cpp/include/openvino/genai/tokenizer.hpp @@ -83,12 +83,6 @@ class OPENVINO_GENAI_EXPORTS Tokenizer { bool add_generation_prompt, const std::string& chat_template="") const; - /** - * @brief Returns true if chat tempate exists and is ready to be applied; false otherwise - * @return is chat tempate ready - */ - bool is_chat_template_ready() const; - // information about , tokens should be public, // they are used at least in StreamerBase descendants int64_t get_bos_token_id() const; diff --git a/src/cpp/src/llm_pipeline.cpp b/src/cpp/src/llm_pipeline.cpp index e86ada33c5..1594dbd583 100644 --- a/src/cpp/src/llm_pipeline.cpp +++ b/src/cpp/src/llm_pipeline.cpp @@ -266,10 +266,6 @@ class StatefulLLMPipeline final : public LLMPipelineImplBase { m_history = {}; m_templated_chat_history = ""; } - OPENVINO_ASSERT(m_tokenizer.is_chat_template_ready(), - "There is no existing chat template for actual model. LLMPipeline cannot work in chat mode." - " Please add chat template to tokenizer_config.json or use another model."); - if (system_message.empty()) return; diff --git a/src/cpp/src/tokenizer.cpp b/src/cpp/src/tokenizer.cpp index 98a9d1a027..748daa5875 100644 --- a/src/cpp/src/tokenizer.cpp +++ b/src/cpp/src/tokenizer.cpp @@ -368,6 +368,11 @@ class Tokenizer::TokenizerImpl { bool add_generation_prompt, const std::string& chat_template) const { auto chat_tpl = chat_template.empty() ? m_chat_template : chat_template; + OPENVINO_ASSERT(!chat_tpl.empty(), + "Chat template wasn't found. This may indicate that the model wasn't trained for chat scenario." + " Please add 'chat_template' to tokenizer_config.json to use the model in chat scenario." + " For more information see the section Troubleshooting in README.md"); + // Jinja2Cpp does not support slicing, e.g. [1:]. // In templates slicing is used typically in the header to find system prompt. // If header containts that typical expression we update template and @@ -433,10 +438,6 @@ class Tokenizer::TokenizerImpl { "For exmaple: user{user_prompt}model"); } } - - bool is_chat_template_ready() { - return !m_chat_template.empty(); - } }; Tokenizer::Tokenizer(const std::string& tokenizer_path, const ov::AnyMap& plugin_config) { @@ -502,11 +503,6 @@ std::string Tokenizer::apply_chat_template(ChatHistory history, return m_pimpl->apply_chat_template(history, add_generation_prompt, chat_template); } -bool Tokenizer::is_chat_template_ready() const { - return m_pimpl->is_chat_template_ready(); -}; - - Tokenizer::~Tokenizer() = default; } // namespace genai } // namespace ov From 318a9f8c8ac7780a1dd3662220e60b4329458e0a Mon Sep 17 00:00:00 2001 From: Oleg Pipikin Date: Tue, 30 Jul 2024 15:06:16 +0000 Subject: [PATCH 3/3] Apply comments --- samples/cpp/chat_sample/README.md | 2 +- samples/python/chat_sample/README.md | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/samples/cpp/chat_sample/README.md b/samples/cpp/chat_sample/README.md index 46d9358180..3f736985c2 100644 --- a/samples/cpp/chat_sample/README.md +++ b/samples/cpp/chat_sample/README.md @@ -37,7 +37,7 @@ If you encounter the error described in the example when sample is printing outp #### Missing chat template -If you encounter an exception indicating a missing "chat template" when launching the `ov::genai::LLMPipeline` in chat mode, it likely means the model was not tuned for chat functionality. To resolve this, manually add the chat template to tokenizer_config.json of your model. +If you encounter an exception indicating a missing "chat template" when launching the `ov::genai::LLMPipeline` in chat mode, it likely means the model was not tuned for chat functionality. To work this around, manually add the chat template to tokenizer_config.json of your model. The following template can be used as a default, but it may not work properly with every model: ``` "chat_template": "{% for message in messages %}{% if (message['role'] == 'user') %}{{'<|im_start|>user\n' + message['content'] + '<|im_end|>\n<|im_start|>assistant\n'}}{% elif (message['role'] == 'assistant') %}{{message['content'] + '<|im_end|>\n'}}{% endif %}{% endfor %}", diff --git a/samples/python/chat_sample/README.md b/samples/python/chat_sample/README.md index a137cddce8..c07023391f 100644 --- a/samples/python/chat_sample/README.md +++ b/samples/python/chat_sample/README.md @@ -27,7 +27,7 @@ See https://github.com/openvinotoolkit/openvino.genai/blob/master/src/README.md# ## Troubleshooting ### Missing chat template -If you encounter an exception indicating a missing "chat template" when launching the `ov::genai::LLMPipeline` in chat mode, it likely means the model was not tuned for chat functionality. To resolve this, manually add the chat template to tokenizer_config.json of your model. +If you encounter an exception indicating a missing "chat template" when launching the `ov::genai::LLMPipeline` in chat mode, it likely means the model was not tuned for chat functionality. To work this around, manually add the chat template to tokenizer_config.json of your model. The following template can be used as a default, but it may not work properly with every model: ``` "chat_template": "{% for message in messages %}{% if (message['role'] == 'user') %}{{'<|im_start|>user\n' + message['content'] + '<|im_end|>\n<|im_start|>assistant\n'}}{% elif (message['role'] == 'assistant') %}{{message['content'] + '<|im_end|>\n'}}{% endif %}{% endfor %}",