| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607 |
- // Tests chat handling, including grammar generation and parsing for tool calling, for various templates.
- //
- // Also acts as a CLI to generate a Markdown summary of the formats of Jinja templates,
- // e.g. given Minja (http://github.com/google/minja) checked out in parent dir:
- //
- // cmake -B build && cmake --build build --parallel && ./build/bin/test-chat ../minja/build/tests/*.jinja 2>/dev/null
- //
- #include <fstream>
- #include <iostream>
- #include <json.hpp>
- #include <string>
- #include "chat-template.hpp"
- #include "chat.hpp"
- #include "llama-grammar.h"
- #include "unicode.h"
- using json = nlohmann::ordered_json;
- static common_chat_msg msg_from_json(const json & message) {
- common_chat_msg ret;
- ret.role = "assistant";
- if (message.contains("content") && !message.at("content").is_null()) {
- ret.content = message.at("content");
- }
- if (message.contains("tool_plan")) {
- ret.tool_plan = message.at("tool_plan");
- }
- auto has_tool_calls = message.contains("tool_calls");
- if (has_tool_calls) {
- for (const auto & tc : message.at("tool_calls")) {
- const auto & arguments = tc.at("function").at("arguments");
- ret.tool_calls.push_back({
- tc.at("function").at("name").get<std::string>(),
- arguments.is_string() ? arguments.get<std::string>() : arguments.dump(),
- tc.contains("id") ? tc.at("id").get<std::string>() : "",
- });
- }
- }
- return ret;
- }
- template <class T> static void assert_equals(const T & expected, const T & actual) {
- if (expected != actual) {
- std::cerr << "Expected: " << expected << std::endl;
- std::cerr << "Actual: " << actual << std::endl;
- std::cerr << std::flush;
- throw std::runtime_error("Test failed");
- }
- }
- static std::string read_file(const std::string & path) {
- std::cerr << "# Reading: " << path << std::endl << std::flush;
- std::ifstream fs(path, std::ios_base::binary);
- if (!fs.is_open()) {
- fs = std::ifstream("../" + path, std::ios_base::binary);
- if (!fs.is_open()) {
- throw std::runtime_error("Failed to open file: " + path);
- }
- }
- fs.seekg(0, std::ios_base::end);
- auto size = fs.tellg();
- fs.seekg(0);
- std::string out;
- out.resize(static_cast<size_t>(size));
- fs.read(&out[0], static_cast<std::streamsize>(size));
- return out;
- }
- static std::unique_ptr<llama_grammar> build_grammar(const std::string & grammar_str) {
- return std::unique_ptr<llama_grammar>(
- llama_grammar_init_impl(nullptr, grammar_str.c_str(), "root", false, nullptr, 0, nullptr, 0));
- }
- // TODO: extract to common helper (copied from test-grammar-integration.cpp)
- static bool match_string(const std::string & input, llama_grammar * grammar) {
- const auto cpts = unicode_cpts_from_utf8(input);
- auto & stacks_cur = llama_grammar_get_stacks(grammar);
- for (const auto & cpt : cpts) {
- llama_grammar_accept(grammar, cpt);
- if (stacks_cur.empty()) {
- // no stacks means that the grammar failed to match at this point
- return false;
- }
- }
- for (const auto & stack : stacks_cur) {
- if (stack.empty()) {
- // An empty stack means that the grammar has been completed
- return true;
- }
- }
- return false;
- }
- // Dumps `{"a": 1}` as `"{\"a\": 1}"`, unlike nlohmann::json::dump which would dump it as `"{\"a\":1}"`.
- static std::string dump(const json & j) {
- return minja::Value(j).dump(-1, /* to_json= */ true);
- }
- static void assert_msg_equals(const common_chat_msg & expected, const common_chat_msg & actual) {
- assert_equals(expected.role, actual.role);
- assert_equals(expected.content, actual.content);
- assert_equals(expected.tool_calls.size(), actual.tool_calls.size());
- for (size_t i = 0; i < expected.tool_calls.size(); i++) {
- const auto & expected_tool_call = expected.tool_calls[i];
- const auto & actual_tool_call = actual.tool_calls[i];
- assert_equals(expected_tool_call.name, actual_tool_call.name);
- assert_equals(dump(json::parse(expected_tool_call.arguments)), dump(json::parse(actual_tool_call.arguments)));
- assert_equals(expected_tool_call.id, actual_tool_call.id);
- }
- }
- const auto special_function_tool = json::parse(R"({
- "type": "function",
- "function": {
- "name": "special_function",
- "description": "I'm special",
- "parameters": {
- "type": "object",
- "properties": {
- "arg1": {
- "type": "integer",
- "description": "The arg."
- }
- },
- "required": ["arg1"]
- }
- }
- })");
- const auto python_tool = json::parse(R"({
- "type": "function",
- "function": {
- "name": "python",
- "description": "an ipython interpreter",
- "parameters": {
- "type": "object",
- "properties": {
- "code": {
- "type": "string",
- "description": "Python code to execute."
- }
- },
- "required": ["code"]
- }
- }
- })");
- const auto code_interpreter_tool = json::parse(R"({
- "type": "function",
- "function": {
- "name": "code_interpreter",
- "description": "an ipython interpreter",
- "parameters": {
- "type": "object",
- "properties": {
- "code": {
- "type": "string",
- "description": "Python code to execute."
- }
- },
- "required": ["code"]
- }
- }
- })");
- const json tools = { special_function_tool, python_tool };
- const json llama_3_1_tools = { special_function_tool, code_interpreter_tool };
- struct delta_data {
- std::string delta;
- common_chat_params params;
- };
- static delta_data init_delta(const common_chat_template & tmpl, const std::vector<std::string> & end_tokens,
- const json & user_message, const json & delta_message, const json & tools,
- const json & tool_choice) {
- common_chat_inputs inputs;
- inputs.parallel_tool_calls = true;
- inputs.messages = json::array();
- inputs.messages.push_back(user_message);
- inputs.tools = tools;
- inputs.tool_choice = tool_choice;
- auto params_prefix = common_chat_params_init(tmpl, inputs);
- inputs.messages.push_back(delta_message);
- inputs.add_generation_prompt = false;
- auto params_full = common_chat_params_init(tmpl, inputs);
- std::string prefix = params_prefix.prompt;
- std::string full = params_full.prompt;
- // Check full starts with prefix
- if (full.find(prefix) != 0) {
- fprintf(stderr, "Full:\n%s\n\nPrefix:\n%s\n\n", full.c_str(), prefix.c_str());
- throw std::runtime_error("Full message does not start with prefix");
- }
- if (full == prefix) {
- throw std::runtime_error("Full message is the same as the prefix");
- }
- auto delta = full.substr(prefix.size());
- // Strip end tokens
- for (const auto & end_token : end_tokens) {
- // rfind to find the last occurrence
- auto pos = delta.rfind(end_token);
- if (pos != std::string::npos) {
- delta = delta.substr(0, pos);
- break;
- }
- }
- return { delta, params_full };
- }
- /*
- Applies the template to 1 user message w/ add_generation_prompt=true, then w/ the test message w/ add_generation_prompt=false,
- gets the diff, removes any end tokens and parses the result w/ the grammar, checking that
- the parsed message is the same as the test_message
- */
- static void test_template(const common_chat_template & tmpl, const std::vector<std::string> & end_tokens,
- const json & test_message, const json & tools = {}, const std::string & expected_delta = "",
- bool expect_grammar_triggered = true) {
- common_chat_msg expected_msg = msg_from_json(test_message);
- auto user_message = json{
- { "role", "user" },
- { "content", "Hello, world!" }
- };
- for (const auto & tool_choice : json({ "auto", "required" })) {
- auto data = init_delta(tmpl, end_tokens, user_message, test_message, tools, tool_choice);
- if (!expected_delta.empty()) {
- assert_equals(expected_delta, data.delta);
- }
- if (expect_grammar_triggered) {
- const auto msg = common_chat_parse(data.delta, data.params.format);
- assert_msg_equals(expected_msg, msg);
- }
- if (!expected_msg.tool_calls.empty()) {
- GGML_ASSERT(!data.params.grammar.empty());
- }
- if (!data.params.grammar.empty()) {
- auto grammar = build_grammar(data.params.grammar);
- if (!grammar) {
- throw std::runtime_error("Failed to build grammar");
- }
- auto earliest_trigger_pos = std::string::npos;
- auto constrained = data.delta;
- for (const auto & trigger : data.params.grammar_triggers) {
- auto pos = constrained.find(trigger.word);
- if (pos == std::string::npos) {
- continue;
- }
- if (pos > 0 && trigger.at_start) {
- fprintf(stderr, "Trigger %s not at start of message, skipping:\n\n%s\n\n", trigger.word.c_str(), constrained.c_str());
- continue;
- }
- if (earliest_trigger_pos == std::string::npos || pos < earliest_trigger_pos) {
- earliest_trigger_pos = pos;
- }
- }
- auto grammar_triggered = false;
- if (earliest_trigger_pos != std::string::npos) {
- constrained = constrained.substr(earliest_trigger_pos);
- grammar_triggered = true;
- }
- if (data.params.grammar_lazy) {
- assert_equals(expect_grammar_triggered, grammar_triggered);
- }
- if (grammar_triggered && !match_string(constrained, grammar.get())) {
- throw std::runtime_error("Failed to match delta against grammar:\n\n" + data.delta +
- "\n\nGrammar: " + data.params.grammar);
- }
- }
- }
- }
- static void test_template_output_parsers() {
- json text_message {
- { "role", "assistant" },
- { "content", "Hello, world!\nWhat's up?" },
- };
- json tool_calls = json::array({{
- { "type", "function" },
- { "function", { { "name", "special_function" }, { "arguments", "{\"arg1\": 1}" } } },
- }});
- json tool_call_message {
- { "role", "assistant"},
- { "content", {}},
- { "tool_calls", {
- {
- { "type", "function" },
- { "function", {
- { "name", "special_function" },
- { "arguments", "{\"arg1\": 1}" },
- }},
- },
- }},
- };
- json tool_call_message_with_id {
- { "role", "assistant"},
- { "content", {}},
- { "tool_calls", {
- {
- { "type", "function" },
- { "function", {
- { "name", "special_function" },
- { "arguments", "{\"arg1\": 1}" },
- }},
- {"id", "123456789"},
- },
- }},
- { "role", "assistant" },
- { "content", {} },
- { "tool_calls", tool_calls }
- };
- json tool_call_plan_message_with_idx {
- { "role", "assistant"},
- { "content", {}},
- { "tool_plan", "I'm not so sure"},
- { "tool_calls", {
- {
- { "type", "function" },
- { "function", {
- { "name", "special_function" },
- { "arguments", "{\"arg1\": 1}" },
- }},
- // Index of the tool call in the tool_calls array
- {"id", "0"},
- },
- }},
- { "role", "assistant" },
- { "content", {} },
- { "tool_calls", tool_calls }
- };
- auto python_tool_call_message = json{
- { "role", "assistant" },
- { "content", {} },
- { "tool_calls", json{ {
- { "type", "function" },
- { "function",
- {
- { "name", "python" },
- { "arguments",
- {
- { "code", "print('hey')" },
- } },
- } },
- } } }
- };
- auto code_interpreter_tool_call_message = json{
- { "role", "assistant" },
- { "content", {} },
- { "tool_calls", json{ {
- { "type", "function" },
- { "function",
- {
- { "name", "code_interpreter" },
- { "arguments",
- {
- { "code", "print('hey')" },
- } },
- } },
- } } }
- };
- common_chat_inputs inputs_no_tools;
- inputs_no_tools.messages = {
- { { "role", "user" }, { "content", "Hey\nThere" } }
- };
- common_chat_inputs inputs_tools = inputs_no_tools;
- inputs_tools.tools = json::array();
- inputs_tools.tools.push_back(special_function_tool);
- common_chat_inputs inputs_tools_builtin = inputs_no_tools;
- inputs_tools_builtin.tools = json::array();
- inputs_tools_builtin.tools.push_back(python_tool);
- {
- // Not supported yet
- const common_chat_template tmpl(read_file("models/templates/CohereForAI-c4ai-command-r-plus-tool_use.jinja"), "<s>", "</s>");
- assert_equals(COMMON_CHAT_FORMAT_GENERIC, common_chat_params_init(tmpl, inputs_tools).format);
- }
- {
- const common_chat_template tmpl(read_file("models/templates/CohereForAI-c4ai-command-r7b-12-2024-tool_use.jinja"), "<s>", "</s>");
- std::vector<std::string> end_tokens{ "<|END_OF_TURN_TOKEN|>" };
- assert_equals(COMMON_CHAT_FORMAT_CONTENT_ONLY, common_chat_params_init(tmpl, inputs_no_tools).format);
- assert_equals(COMMON_CHAT_FORMAT_COMMAND_R7B, common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, tool_call_plan_message_with_idx, tools,
- "<|START_THINKING|>I'm not so sure<|END_THINKING|>"
- "<|START_ACTION|>[\n"
- " {\"tool_call_id\": \"0\", \"tool_name\": \"special_function\", \"parameters\": {\"arg1\": 1}}\n"
- "]<|END_ACTION|>");
- test_template(tmpl, end_tokens, text_message, tools,
- "<|START_RESPONSE|>Hello, world!\n"
- "What's up?<|END_RESPONSE|>",
- /* expect_grammar_triggered= */ false);
- }
- {
- const common_chat_template tmpl(read_file("models/templates/google-gemma-2-2b-it.jinja"), "<s>", "</s>");
- std::vector<std::string> end_tokens{ "<end_of_turn>" };
- assert_equals(COMMON_CHAT_FORMAT_CONTENT_ONLY, common_chat_params_init(tmpl, inputs_no_tools).format);
- assert_equals(COMMON_CHAT_FORMAT_GENERIC, common_chat_params_init(tmpl, inputs_tools).format);
- assert_equals(COMMON_CHAT_FORMAT_GENERIC,
- common_chat_params_init(
- common_chat_template(read_file("models/templates/microsoft-Phi-3.5-mini-instruct.jinja"),
- "<s>", "</s>"),
- inputs_tools)
- .format);
- // Generic tool calls doesn't generate / parse content-only messages symmetrically.
- assert_msg_equals(msg_from_json(text_message),
- common_chat_parse("{\n"
- " \"response\": \"Hello, world!\\nWhat's up?\"\n"
- "}",
- common_chat_params_init(tmpl, inputs_tools).format));
- test_template(tmpl, end_tokens, tool_call_message_with_id, tools,
- "{\n"
- " \"tool_calls\": [\n"
- " {\n"
- " \"name\": \"special_function\",\n"
- " \"arguments\": {\n"
- " \"arg1\": 1\n"
- " },\n"
- " \"id\": \"123456789\"\n"
- " }\n"
- " ]\n"
- "}");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/mistralai-Mistral-Nemo-Instruct-2407.jinja"), "<s>",
- "</s>");
- std::vector<std::string> end_tokens{ "</s>" };
- assert_equals(COMMON_CHAT_FORMAT_MISTRAL_NEMO, common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, text_message, tools, "Hello, world!\nWhat's up?", /* expect_grammar_triggered= */ false);
- test_template(
- tmpl, end_tokens, tool_call_message_with_id, tools,
- "[TOOL_CALLS][{\"name\": \"special_function\", \"arguments\": {\"arg1\": 1}, \"id\": \"123456789\"}]");
- }
- {
- const common_chat_template tmpl(
- read_file("models/templates/NousResearch-Hermes-2-Pro-Llama-3-8B-tool_use.jinja"), "<s>", "</s>");
- std::vector<std::string> end_tokens{ "<|im_end|>" };
- assert_equals(COMMON_CHAT_FORMAT_HERMES_2_PRO, common_chat_params_init(tmpl, inputs_tools).format);
- assert_equals(
- COMMON_CHAT_FORMAT_HERMES_2_PRO,
- common_chat_params_init(
- common_chat_template(read_file("models/templates/NousResearch-Hermes-3-Llama-3.1-8B-tool_use.jinja"),
- "<s>", "</s>"),
- inputs_tools)
- .format);
- assert_equals(
- COMMON_CHAT_FORMAT_HERMES_2_PRO,
- common_chat_params_init(
- common_chat_template(read_file("models/templates/Qwen-Qwen2.5-7B-Instruct.jinja"), "<s>", "</s>"),
- inputs_tools)
- .format);
- test_template(tmpl, end_tokens, text_message, tools, "Hello, world!\nWhat's up?", /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, tool_call_message, tools,
- "<tool_call>\n"
- "{\"name\": \"special_function\", \"arguments\": {\"arg1\": 1}}\n"
- "</tool_call>");
- test_template(tmpl, end_tokens, python_tool_call_message, tools,
- "<tool_call>\n"
- "{\"name\": \"python\", \"arguments\": {\"code\": \"print('hey')\"}}\n"
- "</tool_call>");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/meta-llama-Llama-3.1-8B-Instruct.jinja"), "<s>",
- "</s>");
- std::vector<std::string> end_tokens{ "<|eom_id|>", "<|eot_id|>" };
- assert_equals(COMMON_CHAT_FORMAT_LLAMA_3_X, common_chat_params_init(tmpl, inputs_tools).format);
- assert_equals(COMMON_CHAT_FORMAT_LLAMA_3_X_WITH_BUILTIN_TOOLS,
- common_chat_params_init(tmpl, inputs_tools_builtin).format);
- assert_equals(COMMON_CHAT_FORMAT_LLAMA_3_X_WITH_BUILTIN_TOOLS,
- common_chat_params_init(
- common_chat_template(read_file("models/templates/meta-llama-Llama-3.3-70B-Instruct.jinja"),
- "<s>", "</s>"),
- inputs_tools_builtin)
- .format);
- // test_template(tmpl, end_tokens, text_message, tools, R"(?)", /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, code_interpreter_tool_call_message, llama_3_1_tools,
- "<|python_tag|>code_interpreter.call(code=\"print('hey')\")");
- test_template(tmpl, end_tokens, python_tool_call_message, tools,
- "<|python_tag|>python.call(code=\"print('hey')\")");
- test_template(tmpl, end_tokens, tool_call_message, tools,
- "{\"name\": \"special_function\", \"parameters\": {\"arg1\": 1}}");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/meta-llama-Llama-3.2-3B-Instruct.jinja"), "<s>",
- "</s>");
- std::vector<std::string> end_tokens{ "<|eom_id|>", "<|eot_id|>" };
- assert_equals(COMMON_CHAT_FORMAT_LLAMA_3_X, common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, text_message, tools, "Hello, world!\nWhat's up?", /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, tool_call_message, tools,
- "{\"name\": \"special_function\", \"parameters\": {\"arg1\": 1}}");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/meetkai-functionary-medium-v3.1.jinja"), "<s>",
- "</s>");
- std::vector<std::string> end_tokens{ "<|eom_id|>", "<|eot_id|>" };
- assert_equals(COMMON_CHAT_FORMAT_FUNCTIONARY_V3_1_LLAMA_3_1,
- common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, text_message, tools, "Hello, world!\nWhat's up?", /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, tool_call_message, tools,
- "<function=special_function>{\"arg1\": 1}</function>");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/meetkai-functionary-medium-v3.2.jinja"), "<s>",
- "</s>");
- std::vector<std::string> end_tokens{ "<|eom_id|>", "<|eot_id|>" };
- assert_equals(COMMON_CHAT_FORMAT_FUNCTIONARY_V3_2, common_chat_params_init(tmpl, inputs_no_tools).format);
- assert_equals(COMMON_CHAT_FORMAT_FUNCTIONARY_V3_2, common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, text_message, {},
- "all\n"
- "Hello, world!\n"
- "What's up?",
- /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, tool_call_message, tools,
- "special_function\n"
- "{\"arg1\": 1}");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/fireworks-ai-llama-3-firefunction-v2.jinja"), "<s>",
- "</s>");
- std::vector<std::string> end_tokens{ "<|eot_id|>" };
- assert_equals(COMMON_CHAT_FORMAT_FIREFUNCTION_V2, common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, text_message, tools, "Hello, world!\nWhat's up?", /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, tool_call_message, tools,
- " functools[{\"name\": \"special_function\", \"arguments\": {\"arg1\": 1}}]");
- }
- {
- const common_chat_template tmpl(read_file("models/templates/deepseek-ai-DeepSeek-R1-Distill-Llama-8B.jinja"),
- "<s>", "</s>");
- std::vector<std::string> end_tokens{ "<|end▁of▁sentence|>" };
- assert_equals(COMMON_CHAT_FORMAT_DEEPSEEK_R1, common_chat_params_init(tmpl, inputs_tools).format);
- test_template(tmpl, end_tokens, text_message, tools, "Hello, world!\nWhat's up?", /* expect_grammar_triggered= */ false);
- test_template(tmpl, end_tokens, tool_call_message, tools,
- "<|tool▁calls▁begin|><|tool▁call▁begin|>function<|tool▁sep|>special_function\n"
- "```json\n"
- "{\"arg1\": 1}\n"
- "```<|tool▁call▁end|>");
- }
- }
- int main(int argc, char ** argv) {
- #ifndef _WIN32
- if (argc > 1) {
- common_chat_inputs inputs;
- inputs.messages = {
- { { "role", "user" }, { "content", "Hey" } }
- };
- inputs.tools = json::array({ special_function_tool });
- std::cout << "| Template | Format |\n";
- std::cout << "|----------|--------|\n";
- for (int i = 1; i < argc; i++) {
- std::string path = argv[i];
- if (path.rfind(".jinja") != path.size() - 6) {
- std::cerr << "Skipping non-jinja file: " << path << std::endl;
- continue;
- }
- common_chat_template tmpl(read_file(path), "", "");
- auto parts = string_split(path, "/");
- auto name = parts[parts.size() - 1];
- std::cout << "| " << name << " | " << common_chat_format_name(common_chat_params_init(tmpl, inputs).format)
- << " |\n";
- }
- } else
- #endif
- {
- test_template_output_parsers();
- std::cout << "\n[chat] All tests passed!" << std::endl;
- }
- return 0;
- }
|