English | 简体中文
To edit an ONNX model, One common way is to visualize the model graph, and edit it using ONNX Python API. This works fine. However, we have to code to edit, then visualize to check. The two processes may iterate for many times, which is time-consuming. 👋
What if we have a tool, which allow us to edit and preview the editing effect in a totally visualization fashion?
Then onnx-modifier
comes. With it, we can focus on editing the model graph in the visualization pannel. All the editing information will be summarized and processed by Python ONNX API automatically at last. Then our time can be saved! 🚀
onnx-modifier
is built based on the popular network viewer Netron and the lightweight web application framework flask.
Currently, the following editing operations are supported:
- Delete/recover nodes
- Delete a single node.
- Delete a node and all the nodes rooted on it.
- Recover a deleted node.
- Rename the name of node inputs/outputs
- Edit the attribute of nodes
- Add new nodes (experimental)
Hope it helps!
Clone the repo and install the required Python packages by
git clone [email protected]:ZhangGe6/onnx-modifier.git
cd onnx-modifier
pip install onnx
pip install flask
Then run
python app.py
Click the url in the output info generated by flask (http://127.0.0.1:5000/
for example), then onnx-modifier
will be launched in the web browser.
Click Open Model...
to upload the ONNX model to edit. The model will be parsed and shown on the page.
Graph-level-operation elements are placed on the left-top of the page. Currently, there are four buttons: Refresh
, Reset
, Download
and Add node
. They can do:
-
Refresh
: Refresh the model graph to preview editing effects.In this version, the model graph is refreshed automatically as soon as an editing operation is invoked. So this button can be used much fewer than earlier versions.
-
Reset
: Reset the whole model graph to its initial state; -
Download
: Save the modified model into disk. -
Add node
: Add a new node into the model.
Node-level-operation elements are all in the sidebar, which can be invoked by clicking a specific node.
Let's take a closer look.
There are two modes for deleting node: Delete With Children
and Delete Single Node
. Delete Single Node
only deletes the clicked node, while Delete With Children
also deletes all the node rooted on the clicked node, which is convenient and natural if we want to delete a long path of nodes.
The implementation of
Delete With Children
is based on the backtracking algorithm.
For previewing, The deleted nodes are in grey mode at first. If a node is deleted by mistake, Recover Node
button can help us recover it back to graph. Click Enter
button to take the deleting operation into effect, then the updated graph will show on the page automatically.
The following figure shows a typical deleting process:
By changing the input/output name of nodes, we can change the model forward path. It can also be helpful if we want to rename the model output(s).
Using onnx-modifier
, we can achieve this by simply enter a new name for node inputs/outputs in its corresponding input placeholder. The graph topology is updated automatically and instantly, according to the new names.
For example, Now we want remove the preprocess operators (Sub->Mul->Sub->Transpose
) shown in the following figure. We can
- Click on the 1st
Conv
node, rename its input (X) as serving_default_input:0 (the output of nodedata_0
). - The model graph is updated automatically and we can see the input node links to the 1st
Conv
directly. In addition, the preprocess operators have been split from the main routine. Delete them. - We are done! (click
Download
, then we can get the modified ONNX model).
Note: To link node
$A$ (data_0
in the above example) to node$B$ (the 1stConv
in the above example), it is suggested to edit the input of node$B$ to the output of nodeA
, rather than edit the output of node$A$ to the input of nodeB
. Because the input of$B$ can also be other node's output (Transpose
in the above example ) and unexpected result will happen.
The process is shown in the following figure:
Change the original attribute to a new value, then we are done.
By clicking the
+
in the right side of placeholder, we can get some helpful reference.
Sometimes we want to add new nodes into the existed model. onnx-modifier
supports this feature experimentally now.
Note there is an Add node
button, following with a selector elements on the top-left of the index page. To do this, what we need to do is as easy as 3 steps:
-
Choose a node type in the selector, and click
Add node
button. Then an empty node of the chosen type will emerge on the graph.The selector contains all the supported operator types in domains of
ai.onnx
(171),ai.onnx.preview.training
(4),ai.onnx.ml
(18) andcom.microsoft
(1). -
Click the new node and edit it in the invoked siderbar. What we need to fill are the node Attributes (
undefined
by default) and its Inputs/Outputs (which decide where the node will be inserted in the graph). -
We are done.
The following are some notes for this feature:
-
⚠️ Currently, adding nodes with initializer (such as weight parameters) is not supported (such asConv
,BatchNormalization
). Adding nodes without initializer are tested and work as expected in my tested case (such asFlatten
,ArgMax
,Concat
). -
Click the selector and type the first letter for the new node type (
f
forFlatten
node for example), we can be quickly navigated to the node type. -
By clicking the
?
in theNODE PROPERTIES -> type
element, or the+
in eachAttribute
element, we can get some reference to help us fill the node information. -
It is suggested to fill all of the
Attribute
, without leaving them asundefined
. The default value may not be supported well in the current version. -
For the
Attribute
with typelist
, items are split with ',
' (comma) -
For the
Inputs/Outputs
with typelist
, it is forced to be at most 8 elements in the current version. If the actual inputs/outputs number is less than 8, we can leave the unused items with the name starting withlist_custom
, and they will be automatically omitted. -
This feature is experimentally supported now and may be not very robust. So any issues are warmly welcomed if some unexpected result is encountered.
For quick testing, some typical sample models are provided as following. Most of them are from onnx model zoo
- squeezeNet Link (4.72MB)
- MobileNet Link (13.3MB)
- ResNet50-int8 Link (24.6MB)
- movenet-lightning Link (9.01MB)
- Converted from the pretrained tflite model using tensorflow-onnx;
- There are preprocess nodes and a big bunch of postprocessing nodes in the model.
onnx-modifier
is under active development 🛠. Welcome to use, create issues and pull requests! 🥰
- Netron
- flask
- ONNX IR Official doc
- ONNX Python API Official doc, Leimao's Blog
- ONNX IO Stream Leimao's Blog
- onnx-utils
- sweetalert