serving开发 如何使用官方的hashmap sourceadapter

本文主要介绍如何使用tensorflow serving官方提供的hashmap sourceadapter。其实理解了如何使用这个hashmapsourceadapter,也就真正能对serving进行二次开发,网上几乎找不到任何相关资料(stackoverflow上的问题Tensorflow-serving: Serving a hashmap,而为了真正让这个hashmapsourceadapter有用,我把serving的代码全部看了一遍。

这里一共分为6个步骤,第7步为接口测试。

首先,官方的代码只是定义了 HashmapSourceAdapter。没有注册。

1 定义 HashmapSourceAdapterCreator

如下

// register this source adapter
class HashmapSourceAdapterCreator {
  public:
    static Status Create(
        const HashmapSourceAdapterConfig& config,
        std::unique_ptr<SourceAdapter<StoragePath, std::unique_ptr<Loader>>>*
            adapter) {
      adapter->reset(new HashmapSourceAdapter(config));
      return Status::OK();
    }
};

同时,把这个类加为 HashmapSourceAdapter 的友元。

 private:
  friend class HashmapSourceAdapterCreator;

2 注册 HashmapSourceAdapter

REGISTER_STORAGE_PATH_SOURCE_ADAPTER(HashmapSourceAdapterCreator,
                                     HashmapSourceAdapterConfig);

3 使用hashmap servable

这一步非常重要,当然这一步不是非要我这么做,但这是最简单的方法。添加这个步骤的原因在于,标准的C++编译程序时,如果一个文件中的代码如果没有被调用,它就会被编译器优化掉。所以,这其实是一个hack。

第一步,在 hashmap_source_adapter.h 定义一个函数

void loadHashmapServable();

第二步,在hashmap_source_adapter.cc 中实现这个函数。

void loadHashmapServable() {
  LOG(INFO) << "load hashmap servable...";
}

第三步,在main函数的开头调用这个函数

tensorflow::serving::loadHashmapServable();

4 http中添加使用hashmap servable的接口

这一步,我们会在原有http接口的基础上,添加一个lookup接口。

一, 在ProcessRequest中添加分支lookup

    } else if (method == "lookup") {
      status = ProcessLookupRequest(model_name, model_version, request_body,
                                     output);
    }

二,定义函数 ProcessLookupRequest

Status HttpRestApiHandler::ProcessLookupRequest(
    const absl::string_view model_name,
    const absl::optional<int64>& model_version,
    const absl::string_view request_body, string* output) {
  ModelSpec model_spec;
  model_spec.set_name(string(model_name));
  if (model_version.has_value()) {
    model_spec.mutable_version()->set_value(model_version.value());
  }
  ServableHandle<std::unordered_map<string, string>> bundle;
  TF_RETURN_IF_ERROR(core_->GetServableHandle(model_spec, &bundle));
  std::unordered_map<std::string, std::string>::const_iterator got = bundle->find(request_body.data());
  if (got == bundle->end()) {
    output->assign(string("None"));
  } else {
    output->assign(got->second);
  }
  return Status::OK();
}

三,放开URL正则匹配的限制

prediction_api_regex_(
          R"((?i)/v1/models/([^/:]+)(?:/versions/(\d+))?:(classify|regress|predict|lookup))"),

5 给hashmap servable加载的文件添加一个文件名

我们程序启动时会去模型目录下加载一个名为 data.csv的文件。

const string fpath = io::JoinPath(path, "data.csv");
      std::unique_ptr<RandomAccessFile> file;
      TF_RETURN_IF_ERROR(Env::Default()->NewRandomAccessFile(fpath, &file));

该文件的格式如下:

key0,value0
key1,value1
tom,jerry
pete,henry
hello,world
good,bye

6 从配置文件启动TF serving

因为加入了hashmap servable,tensorflow serving不止支持一个platform,当tensorflow serving支持多个platform的时候需要从配置文件启动,命令如下:

tensorflow_model_server --port=8500 --rest_api_port=8501 --platform_config_file=./etc/platform.conf --model_config_file=./etc/models.conf

其中,platform.conf内容如下:

platform_configs {
  key: "hashmap"
  value {
    source_adapter_config {
      type_url: "type.googleapis.com/tensorflow.serving.HashmapSourceAdapterConfig"
    }
  }
}
platform_configs {
  key: "tensorflow"
  value {
    source_adapter_config {
      type_url: "type.googleapis.com/tensorflow.serving.SavedModelBundleSourceAdapterConfig"
      value: "\302>\002\022\000"
    }
  }
}

models.conf内容如下:

model_config_list: {
    config: {
                name: "tensorflow",
        base_path: "/data/models/tensorflow",
        model_platform: "tensorflow",
        model_version_policy: {
            all: {}
        }
    }
    config: {
                name: "hash",
        base_path: "/data/models/hash",
        model_platform: "hashmap",
        model_version_policy: {
            all: {}
        }
    }
}

platform.conf的编写可以参考这个issue
how to write a tensorflow serving platform_config_file

7 测试

访问接口

curl -d 'hello' -X POST http://localhost:8501/v1/models/hash/versions/1:lookup

输出

world
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,490评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,581评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,830评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,957评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,974评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,754评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,464评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,357评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,847评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,995评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,137评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,819评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,482评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,023评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,149评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,409评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,086评论 2 355

推荐阅读更多精彩内容