| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| import argparse |
| import json |
| import os |
|
|
| from tensorflow.core.protobuf.saved_model_pb2 import SavedModel |
|
|
|
|
| |
| |
| REPO_PATH = "." |
|
|
| |
| INTERNAL_OPS = [ |
| "Assert", |
| "AssignVariableOp", |
| "EmptyTensorList", |
| "MergeV2Checkpoints", |
| "ReadVariableOp", |
| "ResourceGather", |
| "RestoreV2", |
| "SaveV2", |
| "ShardedFilename", |
| "StatefulPartitionedCall", |
| "StaticRegexFullMatch", |
| "VarHandleOp", |
| ] |
|
|
|
|
| def onnx_compliancy(saved_model_path, strict, opset): |
| saved_model = SavedModel() |
| onnx_ops = [] |
|
|
| with open(os.path.join(REPO_PATH, "utils", "tf_ops", "onnx.json")) as f: |
| onnx_opsets = json.load(f)["opsets"] |
|
|
| for i in range(1, opset + 1): |
| onnx_ops.extend(onnx_opsets[str(i)]) |
|
|
| with open(saved_model_path, "rb") as f: |
| saved_model.ParseFromString(f.read()) |
|
|
| model_op_names = set() |
|
|
| |
| for meta_graph in saved_model.meta_graphs: |
| |
| model_op_names.update(node.op for node in meta_graph.graph_def.node) |
|
|
| |
| for func in meta_graph.graph_def.library.function: |
| |
| model_op_names.update(node.op for node in func.node_def) |
|
|
| |
| model_op_names = sorted(model_op_names) |
| incompatible_ops = [] |
|
|
| for op in model_op_names: |
| if op not in onnx_ops and op not in INTERNAL_OPS: |
| incompatible_ops.append(op) |
|
|
| if strict and len(incompatible_ops) > 0: |
| raise Exception(f"Found the following incompatible ops for the opset {opset}:\n" + incompatible_ops) |
| elif len(incompatible_ops) > 0: |
| print(f"Found the following incompatible ops for the opset {opset}:") |
| print(*incompatible_ops, sep="\n") |
| else: |
| print(f"The saved model {saved_model_path} can properly be converted with ONNX.") |
|
|
|
|
| if __name__ == "__main__": |
| parser = argparse.ArgumentParser() |
| parser.add_argument("--saved_model_path", help="Path of the saved model to check (the .pb file).") |
| parser.add_argument( |
| "--opset", default=12, type=int, help="The ONNX opset against which the model has to be tested." |
| ) |
| parser.add_argument( |
| "--framework", choices=["onnx"], default="onnx", help="Frameworks against which to test the saved model." |
| ) |
| parser.add_argument( |
| "--strict", action="store_true", help="Whether make the checking strict (raise errors) or not (raise warnings)" |
| ) |
| args = parser.parse_args() |
|
|
| if args.framework == "onnx": |
| onnx_compliancy(args.saved_model_path, args.strict, args.opset) |
|
|