项目传送门:https://github.com/wzzzd/FAQ_system
构建了一个FAQ智能问答系统。
使用多种方法,实现FAQ的问题-模板匹配功能。
使用Tornado框架,部署成轻量级的Web服务应用。
整体框架如下。

1.读取QA数据集
2.创建Elasticsearch的index索引
3.将QA语料导入Elasticsearch
输入query文本 -> 分词 -> 召回(ES) -> 粗序(PreRank) -> 精排(Rank) -> result
BM25的思路,计算query与候选数据集之间的得分,作为排序分数。2-gram、3-gram、4-gram提取词片段,再使用jaccard计算相似度,作为排序分数。gensim框架训练得到的word2vec模型。file/model/下。项目默认使用【Original size: 1.8G; tar.gz size: 763M】的词向量,若使用其他词向量文件,需要修改Config.py下的path_w2v_tx变量值。sentence-transformers,Base model为MiniLM,使用的模型参数为all-MiniLM-L12-v2。给出句子
sentence1: 车子不小心刮花了,保险流程怎么办?
sentence2: 不小心刮花车子了,怎么走保险流程?
label: 1(语义相同)
转化为
input: 车子不小心刮花了,保险流程怎么办?[SEP]不小心刮花车子了,怎么走保险流程?
label: 1
1.保险行业语料
is_best=1筛选出回答正确的数据,获取其中的title和reply字段,处理成两个数据集,位于目录data/insurance_zhidao_test/下
corpus.txt: 语料库,包含question和answer两个字段。
question:与原始文件的title字段对应answer:与原始文件的reply对应。test.txt : 测试数据,包含query和question两个字段。
question:根据question字段使用回译的方法(中->英->中),获取语义相同但表述不同的句子。answer:表示原始问题的标准答案。2.自定义语料
corpus.txt和test.txt。本项目基于python==3.8实现。
若安装了anaconda,建议创建一套虚拟环境,再安装环境依赖
$ conda create -n your_env_name python=3.8 # 创建虚拟环境
$ conda activate your_env_name # 激活环境
$ pip install -r requirements.txt # 安装依赖
若无安装anaconda,直接安装依赖
$ pip install -r requirements.txt
另外,为了在语料较大的情况下,减少召回模块消耗的时间,本项目使用Elasticsearch分布式搜索引擎来存储所有的语料。使用版本为(7-10-2)。
按照完毕后,启动ES服务
cd /your_path/elasticsearch-7.10.2/bin;
./elasticsearch -d;
在模型配置,粗排方法为bm25,精排方法为有监督的bert方法下,部分效果如下:
Query:重疾险有年龄限制吗?
PreRank:
- question:商业养老保险是否有投保年龄限制 - score:12.33
- question:康惠保重疾险是所有人都可以买吗?对年龄有什么限制? - score:9.42
- question:重疾险对购买人年龄有没有要求? - score:9.01
- question:多大年龄开始购买重疾险比较好? - score:9.01
- question:哆啦A保对年龄有没有限制,10岁的小女孩可以买吗 - score:7.74
- question:交强险有报案时间限制吗 - score:7.38
- question:生育险报销有时间限制吗 - score:7.00
- question:生育险到账有时间限制去取吗 - score:6.66
- question:有商业保险可年龄大可以贷款吗 - score:6.51
- question:16周岁的孩子买人身保险有什么限制吗 - score:6.07
Rank:
- question:重疾险对购买人年龄有没有要求? - score:0.98
- question:有年交2000块的重疾险吗 - score:0.94
- question:康惠保重疾险是所有人都可以买吗?对年龄有什么限制? - score:0.76
- question:多大年龄开始购买重疾险比较好? - score:0.69
- question:缴费年限是什么? - score:0.53
- question:重大疾病保险有必要买吗? - score:0.52
- question:重疾险有多少种疾病 - score:0.50
在100个样本的测试集中,分别统计top1、top3、top10的召回结果准确率。
| 粗排模型 | 精排模型 | Top1 Acc | Top3 Acc | Top10 Acc |
|---|---|---|---|---|
| bm25 | - | 0.77 | 0.86 | 0.92 |
| ngram | - | 0.52 | 0.69 | 0.84 |
| word2vec(tencent) | - | 0.76 | 0.83 | 0.88 |
| bm25 | lm-mini(Unsup) | 0.45 | 0.54 | 0.64 |
| bm25 | simcse-bert(Unsup) | 0.92 | 0.98 | 0.99 |
| bm25 | bert(sup) | 0.99 | 1.0 | 1.0 |
根目录下的Config.py文件,是配置文件。可自行修改相关的参数:
es_ip:ES搜索引擎的地址,默认是部署在同一台设备环境里,如果ES是部署在其他服务器,那么需要改成其他服务器的地址。es_index:表示语料存储在ES中的index名字。model_name:表示排序的方法,分别包含bm25/ngram/word2vec/word2vec-tx。dataset:表示数据集的名称,对应目录./data/下的数据文件夹名。use_rank:是否使用精排模块。True开启,False关闭。use_supervise:精排模块使用有监督方法,还是无监督方法。True为监督方法,False为无监督方法。unsup_rank_name:表示无监督精排中,使用的模型类型,可选:simcse-distilbert/simcse-bert/lm-minisup_rank_name:表示监督方法精排中,使用的模型类型,可选:distilbert/bert考虑到性能问题,原始的数据会被存储到Elasticsearch。
在开始以下Demo之前,需要将加工数据,并存储到ES。
$ python insert_data_to_es.py
若配置文件config.py中,字段use_supervise=True,则表示在rank阶段,使用有监督的方法来实现。
那么就需要提前训练好一个rank模型。
本项目提供了已经处理好的保险行业rank数据集,可参考
└── data
└── insurance_zhidao_rank
├── train.txt
├── dev.txt
└── test.txt
模型方面,选了本人另一个项目text_classifier_pytorch中的bert模型来完成排序任务。
可直接将本数据集,替换掉该项目的数据集,再进行训练,即可得到排序模型。
若想直接使用本数据集训练的模型参数,可直接下载模型文件rank-bert,密码:khpb。并将下载的所有文件(非文件夹)放在目录file/supervise/bert/下。
若配置文件config.py中,字段use_supervise=False,则表示在rank阶段,使用有无监督的方法来实现。字段unsup_rank_name=simcse-bert时,表示使用SimCSE训练的预训练模型来进行句子语义提取。
具体的无监督SimCSE模型及训练,可参考论文源码SimCSE。
本项目提供了已经处理好的保险行业的无监督训练数据集,可参考data/insurance_zhidao_unsup/corpus.txt
若想直接使用本数据集训练的模型参数,可直接下载模型文件simcse-bert,密码:65av。并将下载的所有文件(非文件夹)放在目录file/unsupervise/simcse_bert/下。
直接测试FAQ效果
$ python FAQ.py
可以将FAQ部署成一个Web服务。
Step1:启动FAQ问答Web服务。
$ python service.py
Step2:以http的方式访问FAQ问答Web服务。
可在linux shell命令行下,发送http请求,其中参数说明
data 里面包含了两个参数question 表示输入的问题size 表示返回的结果数量,返回结果不一定与输入的size值一致,但是返回结果长度会小于或等于size值。$ curl --request POST \
--url http://localhost:5000/api/v1/faq/ \
--header 'content-type: application/json' \
--data '{"question": "什么是综保保险?","size": 3}'
结果会按照相似得分倒序排序,服务会返回类似以下的结果格式:
{
"status": 200,
"answer": [
{
"question": "综保是什么保险",
"answer": "综合保险是为了维护外来从业人员的合法权益,规范单位用工行为,维护劳动力市场秩序。综合保险包括工伤(或者意外伤害)、住院医疗和老年补贴等三项保险待遇。外来从业人员在参加综合保险期间发生工伤事故或患职业病的,可以得到一次性支付的工伤保险金...",
"score": "0.98"
},
{
"question": "什么是万能型保险?",
"answer": "万能型保险是指包含保险保障功能并至少在一个投资账户拥有一定资产价值的人身保险产品。万能型保险除了同传统寿险一样给予保护生命保障外,还可以让客户直接参与由保险公司为投保人建立的投资帐户内资金的投资活动,将保单的价值与保险公司独立运作的投保人投资帐户资金的业绩联系起来...",
"score": "0.97"
},
{
"question": "保险到底是什么?",
"answer": "保险是指投保人根据合同约定,向保险人支付保险费,保险人对于合同约定的可能发生的事故因其发生所造成的财产损失承担赔偿保险金责任,或者被保险人死亡、伤残、疾病或者达到合同约定的年龄、期限等条件时承担给付保险金责任的商业保险行为...",
"score": "0.96"
}
]
}
说明
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何
我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>
如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象
关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?
最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru
在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo
我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案