Project Icon

lm-format-enforcer

开源库实现语言模型输出格式的精确控制

lm-format-enforcer是一个Python开源库,用于确保语言模型输出符合特定格式,如JSON Schema或正则表达式。该库通过在生成过程中过滤tokens来实现精确控制,同时保留模型的灵活性。它支持多种主流LLM框架,提供批处理生成和波束搜索等功能,可处理JSON schema中的复杂结构。这为开发者提供了一种简便方法来获得结构化的AI生成内容。

lm-format-enforcer

LMFE标志

强制执行语言模型的输出格式(JSON Schema、正则表达式等)

在Colab中打开

代码覆盖率 测试

解决方案概览

语言模型能够生成文本,但在需要精确的输出格式时,它们并不总是按照指示执行。 各种提示工程技术被引入以提高生成文本的稳健性,但它们并不总是足够的。 这个项目通过在每个时间步过滤语言模型允许生成的标记来解决这些问题,从而确保输出格式得到遵守,同时最小化对语言模型的限制。

安装

pip install lm-format-enforcer

基础教程

# 如果在Google Colab上使用T4 GPU运行,需要安装以下依赖
!pip install transformers torch lm-format-enforcer huggingface_hub optimum
!pip install auto-gptq --extra-index-url https://huggingface.github.io/autogptq-index/whl/cu118/ 

from pydantic import BaseModel
from lmformatenforcer import JsonSchemaParser
from lmformatenforcer.integrations.transformers import build_transformers_prefix_allowed_tokens_fn
from transformers import pipeline

class AnswerFormat(BaseModel):
    first_name: str
    last_name: str
    year_of_birth: int
    num_seasons_in_nba: int

# 创建一个transformers pipeline
hf_pipeline = pipeline('text-generation', model='TheBloke/Llama-2-7b-Chat-GPTQ', device_map='auto')
prompt = f'以下是Michael Jordan的信息,符合以下JSON模式: {AnswerFormat.schema_json()} :\n'

# 创建一个字符级解析器并从中构建transformers前缀函数
parser = JsonSchemaParser(AnswerFormat.schema())
prefix_function = build_transformers_prefix_allowed_tokens_fn(hf_pipeline.tokenizer, parser)

# 使用前缀函数调用pipeline
output_dict = hf_pipeline(prompt, prefix_allowed_tokens_fn=prefix_function)

# 提取结果
result = output_dict[0]['generated_text'][len(prompt):]
print(result)
# {'first_name': 'Michael', 'last_name': 'Jordan', 'year_of_birth': 1963, 'num_seasons_in_nba': 15}

功能/优势

  • 适用于任何Python语言模型和分词器。已经支持transformersLangChainLlamaIndexllama.cppvLLMHaystackNVIDIA TensorRT-LLMExLlamaV2
  • 支持批处理生成和束搜索 - 每个输入/束在每个时间步可以有不同的标记过滤
  • 支持JSON Schema、JSON模式(无模式)和正则表达式格式
  • 支持JSON模式中的必需和可选字段
  • 支持JSON模式中的嵌套字段、数组和字典
  • 允许语言模型控制JSON模式中的空白和字段顺序,减少幻觉
  • 不修改transformers API的高级循环,因此可以在任何场景中使用

与其他库的比较

功能LM Format EnforcerGuidanceJsonformerOutlines
正则表达式
JSON Schema🟡 (可进行部分转换)
批处理生成
束搜索
集成到现有管道
可选JSON字段
LLM控制JSON字段排序和空白
带递归类的JSON Schema

发现错误?库更新了新功能?提出问题!

详细示例

我们创建了一个Google Colab笔记本,其中包含了如何使用这个库来强制执行llama2的输出格式的完整示例,包括解释中间结果。这个笔记本可以在Colab的免费GPU支持运行时上运行。

在Colab中打开

你也可以在GitHub上查看笔记本

关于与huggingface transformers集成的不同方式,请查看单元测试

vLLM服务器集成

LM Format Enforcer已集成到vLLM推理服务器中。vLLM包含一个兼容OpenAI的服务器,具有额外的功能,允许使用LM Format Enforcer而无需编写自定义推理代码。

通过以下方式在vLLM OpenAI服务器中使用LM Format Enforcer,可以添加vLLM命令行参数

python -m vllm.entrypoints.openai.api_server \
  --model mistralai/Mistral-7B-Instruct-v0.2 \
  --guided-decoding-backend lm-format-enforcer

或者在每个请求的基础上,通过将guided_decoding_backend参数与引导解码参数一起添加到请求中:

completion = client.chat.completions.create(
  model="mistralai/Mistral-7B-Instruct-v0.2",
  messages=[
    {"role": "user", "content": "对这个情感进行分类:LMFE很棒!"}
  ],
  extra_body={
    "guided_regex": "[Pp]ositive|[Nn]egative",
    "guided_decoding_backend": "lm-format-enforcer"
  }
)

JSON模式和选择解码也通过guided_jsonguided_choice 额外参数支持。

它是如何工作的?

该库通过将字符级解析器和分词器前缀树组合成一个智能标记过滤机制来工作。

某个时间步中字符级解析器和分词器前缀树的示例

字符级解析器

将字符串解析为任何类型的格式器可以看作是一个隐式的树结构 - 在解析过程的任何时刻,都有一组允许的下一个字符,如果选择其中任何一个,就会有一组新的允许的下一个字符,以此类推。

CharacterLevelParser是一个根据这种隐式结构进行解析的接口。add_character()get_allowed_characters()可以看作是树遍历方法。

这个接口有几种实现:

  • JsonSchemaParser - 根据JSON模式进行解析(或纯JSON输出 - JsonSchemaParser(None)将允许任何JSON对象)。
  • StringParser - 强制执行精确的字符串(主要用于诊断)
  • RegexParser - 根据正则表达式进行解析。请注意,这不能使用内置的Python正则表达式,而是使用手动实现的(通过interegular库),因此它不能覆盖100%的正则表达式标准。

分词器前缀树

给定某个语言模型使用的分词器,我们可以构建该语言模型可以生成的所有标记的前缀树。这是通过生成所有可能的标记序列并将它们添加到树中来完成的。 参见TokenizerPrefixTree

结合两者

给定一个字符级解析器和一个分词器前缀树,我们可以优雅而高效地过滤语言模型在下一个时间步允许生成的标记: 我们只遍历同时存在于字符级解析节点和分词器前缀树节点中的字符。这使我们能够找到所有允许的标记(包括复杂的子词标记,如",",这在JSON解析中至关重要)。 我们在两棵树上递归地执行此操作,并返回所有允许的标记。当语言模型生成一个标记时,我们根据新字符推进字符级解析器,为过滤下一个时间步做好准备。

这种方法有何不同?为什么它很好?

这不是第一个强制执行语言模型输出格式的库。然而,其他类似的库(如Guidance、JsonFormer和Outlines)强制执行精确的输出格式。这意味着语言模型不能控制空白、字段可选性和字段顺序(在JSON用例中)。虽然这对人类来说似乎无关紧要,但它意味着语言模型可能无法生成它"想要"生成的JSON格式,并可能将其内部状态置于次优值,降低后续时间步输出的质量。

这迫使语言模型用户了解他们所使用的语言模型的细节(例如 - 在预训练之前JSON是否被最小化?)并修改库以生成精确的格式。

我们通过扫描潜在的下一个标记并允许任何将被解析为输出格式的标记序列来避免这个问题。这意味着语言模型可以控制所有这些方面,并以最自然的方式输出与其风格匹配的标记序列,而不需要开发人员了解细节。

诊断 - 我总是能得到好的结果吗?

使用这个库可以保证输出格式的匹配,但不能保证输出在语义上是正确的。强制语言模型遵循某种特定输出格式可能会增加产生幻觉的可能性。通过提示工程来引导模型仍然可能改善结果。

为了帮助你理解格式强制执行造成的激进性,如果你在 generate_enforced()kwargs 中传递 output_scores=Truereturn_dict_in_generate=True(这些是 transformers 库中现有的可选参数),你还会得到一个逐token的数据框,显示选择了哪个token、其分数,以及如果不应用格式强制执行会选择的token。如果你看到格式强制器迫使语言模型选择权重很低的token,这很可能是导致结果不佳的原因。尝试修改提示以引导语言模型,避免格式强制器过于激进。

使用正则表达式格式 Michael Jordan was Born in (\d)+. 的示例:

idxgenerated_tokengenerated_token_idxgenerated_scoreleading_tokenleading_token_idxleading_score
0298711.000000298711.000000
1Michael240830.000027▁Sure185850.959473
2▁Jordan182841.000000▁Jordan182841.000000
3▁was4711.000000▁was4711.000000
4▁Born192980.000008▁born63451.000000
5▁in2970.994629▁in2970.994629
6298710.982422298710.982422
71298961.0000001298961.000000
89299291.0000009299291.000000
96299531.0000006299531.000000
103299411.0000003299411.000000
11.298890.999512.298890.999512
12</s>20.981445</s>20.981445

你可以看到,模型"想要"用 Sure 开始回答,但格式强制器迫使它使用 Michael - 在token 1中存在很大差距。之后,几乎所有的leading scores都在允许的token集合内,这意味着模型可能不会因为token强制而产生幻觉。唯一的例外是时间步骤4 - " Born"被强制使用,而LLM想要选择"born"。这为提示工程师提供了一个提示,建议将提示中的B改为小写。

配置选项

LM Format Enforcer使用几种启发式方法来避免LLM生成结构化输出时可能出现的边缘情况。有两种方法可以控制这些启发式方法:

选项1:通过环境变量

有几个可以设置的环境变量,会影响库的运行。当你不想修改代码时,这种方法很有用,例如通过vLLM OpenAI服务器使用库时。

  • LMFE_MAX_CONSECUTIVE_WHITESPACES - 解析JsonSchemaObjects时允许的最大连续空白字符数。默认值:12。
  • LMFE_STRICT_JSON_FIELD_ORDER - JsonSchemaParser是否应该强制属性按照JsonSchema的'required'列表中出现的顺序排列?(注:这与Pydantic模型中的声明顺序一致)。默认值:False。
  • LMFE_MAX_JSON_ARRAY_LENGTH - 如果未在架构中指定,JSON数组的最大长度是多少。帮助LLM避免无限循环。默认值:20。

选项2:通过CharacterLevelParserConfig类

通过代码使用库时,任何 CharacterLevelParserJsonSchemaParserRegexParser 等)构造函数都接收一个可选的 CharacterLevelParserConfig 对象。

因此,要配置单个解析器的启发式方法,实例化一个 CharacterLevelParserConfig 对象,修改其值,并将其传递给 CharacterLevelParser 的构造函数。

已知问题和限制

  • LM Format Enforcer需要一个Python API来处理语言模型的输出logits。这意味着在API扩展之前,它无法与OpenAI ChatGPT和类似的基于API的解决方案一起使用。
  • 不完全支持正则表达式语法。更多详情请参见interegular
  • LM Format Enforcer正则表达式解析器只能生成存在于分词器词汇表中的字符。这可能会在以后的版本中解决,请参见GitHub上的问题

贡献者和贡献

查看CONTRIBUTORS.md获取贡献者列表。

项目侧边栏1项目侧边栏2
推荐项目
Project Cover

豆包MarsCode

豆包 MarsCode 是一款革命性的编程助手,通过AI技术提供代码补全、单测生成、代码解释和智能问答等功能,支持100+编程语言,与主流编辑器无缝集成,显著提升开发效率和代码质量。

Project Cover

AI写歌

Suno AI是一个革命性的AI音乐创作平台,能在短短30秒内帮助用户创作出一首完整的歌曲。无论是寻找创作灵感还是需要快速制作音乐,Suno AI都是音乐爱好者和专业人士的理想选择。

Project Cover

白日梦AI

白日梦AI提供专注于AI视频生成的多样化功能,包括文生视频、动态画面和形象生成等,帮助用户快速上手,创造专业级内容。

Project Cover

有言AI

有言平台提供一站式AIGC视频创作解决方案,通过智能技术简化视频制作流程。无论是企业宣传还是个人分享,有言都能帮助用户快速、轻松地制作出专业级别的视频内容。

Project Cover

Kimi

Kimi AI助手提供多语言对话支持,能够阅读和理解用户上传的文件内容,解析网页信息,并结合搜索结果为用户提供详尽的答案。无论是日常咨询还是专业问题,Kimi都能以友好、专业的方式提供帮助。

Project Cover

讯飞绘镜

讯飞绘镜是一个支持从创意到完整视频创作的智能平台,用户可以快速生成视频素材并创作独特的音乐视频和故事。平台提供多样化的主题和精选作品,帮助用户探索创意灵感。

Project Cover

讯飞文书

讯飞文书依托讯飞星火大模型,为文书写作者提供从素材筹备到稿件撰写及审稿的全程支持。通过录音智记和以稿写稿等功能,满足事务性工作的高频需求,帮助撰稿人节省精力,提高效率,优化工作与生活。

Project Cover

阿里绘蛙

绘蛙是阿里巴巴集团推出的革命性AI电商营销平台。利用尖端人工智能技术,为商家提供一键生成商品图和营销文案的服务,显著提升内容创作效率和营销效果。适用于淘宝、天猫等电商平台,让商品第一时间被种草。

Project Cover

AIWritePaper论文写作

AIWritePaper论文写作是一站式AI论文写作辅助工具,简化了选题、文献检索至论文撰写的整个过程。通过简单设定,平台可快速生成高质量论文大纲和全文,配合图表、参考文献等一应俱全,同时提供开题报告和答辩PPT等增值服务,保障数据安全,有效提升写作效率和论文质量。

投诉举报邮箱: service@vectorlightyear.com
@2024 懂AI·鲁ICP备2024100362号-6·鲁公网安备37021002001498号