基于内容的召回在推荐系统中是比较常见的召回策略,常见有基于用户或物品的标签召回或者基于用户的年龄,地域等召回,通常该策略的实现是基于开源软件 Elasticseach 实现的。虽然召回的结果都比较合理,可是召回的新颖度,惊喜度等都比较低。好比经过标签“刘德华”进行召回,基本上召回的都是包含刘德华字眼的物品,不太可能召回出“黎明”,“张学友”等其余四大天王的物品。近年随着万物皆可 Embedding,特别是 word2vec,item2vec, graph2vec 等技术的成功应用,经过物品向量召回物品向量的方法也成为推荐系统中比较经常使用的召回策略。本文着重讲述经过开源软件 Vearch 来搭建一个向量搜索服务,并成功实现以图搜图的功能。node
最近一直在作小视频的推荐优化,优化的目标是人均入屏。按照以前资讯流的推荐经验,但愿根据用户的播放记录召回更多相关的小视频来给用户消费。小视频的推荐场景和如今比较火爆的抖音有点相似,就是自动播放一小段视频(5-30s左右),该小视频基本占据了一整屏,用户能够对这个小视频进行点赞,分享,评论;若是不喜欢能够经过上滑来观看下一个小视频。考虑到决定用户是否想看的因素不太多是视频下方两行的视频标题,更多的因素还在于视频的封面图可否引发用户的兴趣。基于这样的项目背景,因此才想搭建一个以图搜图的服务来进行小视频的封面图召回。python
由于以前使用过 gRPC 封装过 Faiss 来搭建向量召回服务,再加上以前作过图片的分类项目,都是将图片转换为向量做为分类器的输入,因此要作这个事基本要解决两件事:mysql
好比在博文Faiss 在项目中的使用中,做者就是使用 SIFT 算法进行图片特征的提取,这些特征对应一个 128 维的向量。将每张图片的特征向量输入给 Faiss 进行类似向量的召回。好比在博文基于gRPC的Faiss server实践中,MXPlayer的技术团队对原先基于 Flask 框架开发的用户/物品向量召回服务进行了 gRPC 的升级,单机压测 QPS 比以前高了2倍以上。原本打算是魔改以前 基于 gRPC 的 Faiss 服务来知足当期的业务场景需求的,可是偶然间发现京东开源的软件 Vearch,就将以前的念头掐灭了,并决定好好的学习这个开源软件。git
以图搜图的服务由两部分组成,一个是向量搜索服务,由 Vearch 提供;一个是将图片的特征提取成特征向量,由 Vearch 的插件 python-algorithm-plugin 提供。github
Vearch 是对大规模深度学习向量进行高性能类似搜索的弹性分布式系统。它的核心是向量搜索,是基于 Faiss 实现的叫 Gamma 引擎。不过除了向量搜索外, Gamma 还能够存储包含标量的文档,并对这些标量字段进行快速索引和过滤。说白了,标量和矢量都支持,而通常的像 Elasticsearch只支持标量。Faiss 只能构建单机的向量搜索服务,而Vearch 以 Gamma 为向量搜索引擎,使用 Raft 协议实现多副本存储,提供 Master 和 Router组件来构建向量类似搜索的弹性分布式系统。其架构图以下:
图中主要有三大组件: Master, Router,PartitionServer,其功能以下:web
从上面能够知道 Gamma 之于 Vearch,就至关于 Lucene 之于 Elasticsearch。算法
图像处理的服务 Vearch 也提供了对应的插件python-algorithm-plugin。Vearch 的目标是构建一个高性能类似搜索的弹性分布式系统。文本,图片和视频均可以转换成向量,因此 Vearch 团队提供了对应的插件来更好的集成到 Vearch 中。对于图片,该插件提供了目标检测,特征提取和类似搜索等功能。其处理逻辑以下:
其逻辑就是从图片中抽取向量特征存储进 Vearch 的 Gamma 引擎中,并提供检索服务。sql
以图搜图的服务由两个服务构成,一个是向量搜索服务,由 Vearch 提供;另外一个是图片特征提取为向量,由 Vearch 的插件python-algorithm-plugin提供。docker
Vearch 是用 Go 编写的,而其核心引擎 Gamma 是用 C++ 编写的(毕竟Faiss 也是 C++ 开发的),因此服务安装部署比较简单粗暴,只要设置好依赖的 lib 包(Faiss,Gamma,RocksDB),并有编译好的二进制文件 vearch,对于单机模式直接使用 ./vearch -conf config.toml
进行服务的启动,而对于集群服务,经过最后的命令参数 ./vearch -conf config.toml ps/router/master
进行配置。数据库
不过由于咱们线上服务器 Gcc 的版本过低,没有 Go 环境等因素,因此采用的是 Docker 方式。鉴于要详细了解 Faiss 服务是如何演变为 Vearch 这个弹性分布式系统的,因此采用的源码编译安装。
# 下载源码 git clone https://github.com/vearch/vearch # 切换到镜像编译目录 cd vearch/cloud # 打包环境镜像 vearch/vearch_env:3.2.2,将gcc,git,faiss,rocksdb,go等安装好 # 这步打包比较慢,能够直接使用官方镜像 docker pull vearch/vearch_env:3.2.2 sh compile_env.sh # 使用 vearch_env 编译二进制文件 vearch,主要是拉取 gamma 源码进行编译 sh compile.sh # 打包 vearch/vearch:3.2.2, 将打包好的二进制文件vearch和依赖的库放到镜像中。 # 能够直接使用官方镜像 docker pull vearch/vearch:3.2.2 sh build.sh
官方的镜像打包仍是有优化的空间的,打包建议使用centos源,准备好 Faiss, RocksDB, Go等源文件。
图像处理的服务没有现成的 Docker 镜像,并且 Github 仓库上提供的镜像打包有问题,可使用如下的仓库进行打包。
# 下载源码(使用修正后的 Dockfile 文件) git clone -b study https://github.com/haojunyu/python-algorithm-plugin # 切换到镜像目录并打包镜像 vearch/images:3.2.2 # 能够直接使用打包好的镜像 docker pull haojunyu/vimgs:3.2.2 cd python-algorithm-plugin && docker build -t haojunyu/vimgs:3.2.2 .
由于两个服务都已经打包成 Docker 镜像了,这里直接使用命令 docker stack deploy -c docker-compose.yml vearch
来启动服务, docker-compose.yml 内容以下:
version: '3.3' services: vearch: image: vearch/vearch:3.2.2 ports: - "8817:8817" - "9001:9001" volumes: - ./config.toml:/vearch/config.toml - ./data:/datas - ./logs:/logs deploy: mode: replicated replicas: 1 restart_policy: condition: on-failure delay: 10s max_attempts: 3 logging: driver: "json-file" options: max-size: "1g" imgs: image: haojunyu/vimgs:3.2.2 ports: - "4101:4101" volumes: - ./python-algorithm-plugin/src/config.py:/app/src/config.py - ./images/imgs:/app/src/imgs command: ["bash", "../bin/run.sh", "image"] deploy: mode: replicated replicas: 3 restart_policy: condition: on-failure delay: 10s max_attempts: 3
注意: 挂载文件 python-algorithm-plugin/src/config.py 是图片处理服务的配置文件,通常只须要针对本身的状况改动如下四个配置:
port
指图片处理服务的端口,默认 4101gpus
指定服务是否使用 gpu,默认不用为 -1master_address
和 router_address
指 Vearch 服务的 master 和 router服务由于图片服务和 Vearch 服务是高度集成的。通常是直接调用图片服务,而图片向量录入 Vearch交给图片服务本身处理。Vearch 详细的操做能够参考文档。
# 这里master_server指vearch主节点及其对应端口:localhost:8817 # 查看集群状态 curl -XGET http://master_server/_cluster/stats # 查看健康状态 curl -XGET http://master_server/_cluster/health # 查看端口状态 curl -XGET http://master_server/list/server # 清除锁(在建立表时会对集群加锁,若在此过程当中,服务异常,会致使锁不能释放,须要手动清除才能新建表。) curl -XGET http://master_server/clean_lock # 副本扩容缩容 curl -XPOST -H "content-type: application/json" -d' { "partition_id":1, "node_id": 1, "method": 0 } ' http://master_server/partition/change_member
库和空间的概念相似mysql里面的数据库和表的概念。
# 查看及群众全部的库 curl -XGET http://master_server/list/db # 建立库 curl -XPUT -H "content-type:application/json" -d '{ "name": "sv_month" } ' http://master_server/db/_create # 查看库 curl -XGET http://master_server/db/$db_name # 删除库(库下存在表空间则没法删除) curl -XDELETE http://master_server/db/$db_name # 查看指定库下全部表空间 curl -XGET http://master_server/list/space?db=$db_name
# 在库sv_month下建立表空间test(针对image) curl -XPUT -H "content-type: application/json" -d '{ "name":"test", "partition_num":1, "replica_num":1, "engine":{ "name":"gamma", "index_size":70000, "max_size":10000000, "id_type":"String", "retrieval_type":"IVFPQ", "retrieval_param":{ "metric_type":"InnerProduct", "ncentroids":256, "nsubvector":32 } }, "properties":{ "itemid":{ "type":"keyword", "index":true }, "feature1":{ "type":"vector", "dimension":512, "model_id":"vgg16", "format":"normalization" } } }' http://image_server:4101/space/sv_month/_create
# 插入本地图片数据到表空间中 curl -XPOST -H "content-type: application/json" -d' { "itemid":"COCO_val2014_000000123599", "feature1":{ "feature":"../images/COCO_val2014_000000123599.jpg" } } ' http://image_server:4101/sv_month/test/AW63W9I4JG6WicwQX_RC
# 查询类似结果 curl -H "content-type: application/json" -XPOST -d '{ "query": { "sum": [ { "feature":"../images/COCO_val2014_000000123599.jpg", "field":"feature1" }] } }' http://image_server:4101/sv_month/test/_search
在服务构建成功后,就须要查看一下以图搜图的效果,而对于效果的鉴别初步以人工为准,最终以线上的指标数据为准。对于一个服务效果的好坏在搭建服务之初就应该有个预期,好比:
如下就是以图搜图的效果截图:
整体来讲效果仍是挺不错的。
推荐策略的上线方式有如下几种:
导入了小视频最近 7 天新增和最近 30 天曝光的视频共 9 万数据导入到单机模式的 Vearch 服务中,没法支撑两个桶平均 48 QPS的冲击,后使用第二种方式解决上线问题。而对应用户向量(播放的图片向量均值)搜索图片向量策略经过第三种方式上线的。
若是该文章对您产生了帮助,或者您对技术文章感兴趣,能够关注微信公众号: 技术茶话会, 可以第一时间收到相关的技术文章,谢谢!
本篇文章由一文多发平台ArtiPub自动发布