1 /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.
2
3 Licensed under the Apache License, Version 2.0 (the "License");
4 you may not use this file except in compliance with the License.
5 You may obtain a copy of the License at
6
7 http://www.apache.org/licenses/LICENSE-2.0
8
9 Unless required by applicable law or agreed to in writing, software
10 distributed under the License is distributed on an "AS IS" BASIS,
11 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 See the License for the specific language governing permissions and
13 limitations under the License.
14 ==============================================================================*/
15
16 #include "tensorflow/cc/saved_model/reader.h"
17
18 #include <unordered_set>
19
20 #include "tensorflow/cc/saved_model/constants.h"
21 #include "tensorflow/core/lib/io/path.h"
22 #include "tensorflow/core/lib/strings/str_util.h"
23 #include "tensorflow/core/lib/strings/strcat.h"
24 #include "tensorflow/core/platform/env.h"
25 #include "tensorflow/core/protobuf/saved_model.pb.h"
26
27 namespace tensorflow {
28 namespace {
29
ReadSavedModel(const string & export_dir,SavedModel * saved_model_proto)30 Status ReadSavedModel(const string& export_dir, SavedModel* saved_model_proto) {
31 LOG(INFO) << "Reading SavedModel from: " << export_dir;
32
33 const string saved_model_pb_path =
34 io::JoinPath(export_dir, kSavedModelFilenamePb);
35 if (Env::Default()->FileExists(saved_model_pb_path).ok()) {
36 return ReadBinaryProto(Env::Default(), saved_model_pb_path,
37 saved_model_proto);
38 }
39 const string saved_model_pbtxt_path =
40 io::JoinPath(export_dir, kSavedModelFilenamePbTxt);
41 if (Env::Default()->FileExists(saved_model_pbtxt_path).ok()) {
42 return ReadTextProto(Env::Default(), saved_model_pbtxt_path,
43 saved_model_proto);
44 }
45 return Status(error::Code::NOT_FOUND,
46 "Could not find SavedModel .pb or .pbtxt at supplied export "
47 "directory path: " +
48 export_dir);
49 }
50
FindMetaGraphDef(const std::unordered_set<string> & tags,SavedModel * saved_model_proto,MetaGraphDef * meta_graph_def)51 Status FindMetaGraphDef(const std::unordered_set<string>& tags,
52 SavedModel* saved_model_proto,
53 MetaGraphDef* meta_graph_def) {
54 LOG(INFO) << "Reading meta graph with tags { " << absl::StrJoin(tags, " ")
55 << " }";
56 for (MetaGraphDef& graph_def : *saved_model_proto->mutable_meta_graphs()) {
57 // Get tags from the graph_def.
58 std::unordered_set<string> graph_tags;
59 for (const string& tag : graph_def.meta_info_def().tags()) {
60 graph_tags.insert(tag);
61 }
62 // Match with the set of tags provided.
63 if (graph_tags == tags) {
64 *meta_graph_def = std::move(graph_def);
65 return Status::OK();
66 }
67 }
68 return Status(
69 error::Code::NOT_FOUND,
70 strings::StrCat(
71 "Could not find meta graph def matching supplied tags: { ",
72 absl::StrJoin(tags, " "),
73 " }. To inspect available tag-sets in the SavedModel, please "
74 "use the SavedModel CLI: `saved_model_cli`"));
75 }
76
77 } // namespace
78
ReadMetaGraphDefFromSavedModel(const string & export_dir,const std::unordered_set<string> & tags,MetaGraphDef * const meta_graph_def)79 Status ReadMetaGraphDefFromSavedModel(const string& export_dir,
80 const std::unordered_set<string>& tags,
81 MetaGraphDef* const meta_graph_def) {
82 SavedModel saved_model_proto;
83 TF_RETURN_IF_ERROR(ReadSavedModel(export_dir, &saved_model_proto));
84 TF_RETURN_IF_ERROR(
85 FindMetaGraphDef(tags, &saved_model_proto, meta_graph_def));
86 return Status::OK();
87 }
88
89 } // namespace tensorflow
90