Create customized callgraph directly from your favorite editor.
- Add a node to the callgraph directly from your editor.
- You choose what to add and where.
- Open the selected node in the editor using right click.
- Add text nodes, and comments
- Export the graph to mermaid
- The graph support scrolling and zooming
- Auto save to localstorage, can export to file.
- Multiple tabs
- Rename in the editor? the change will propagate to the graph.
Graffiti was built with the following assumptions:
- You might use more than a single editor for a project.
- You might want to run everything locally.
- It should be easy to use.
Graffiti consists of 3 separate components
- Backend - the editor used to browse code. The editor might be native, therefore supporting TCP sockets. However, some editors are inside a browser (for example: OpenGrok). Chrome doesn't support TCP Sockets, so Backend should be able to communicate with WebSocket also. Backend should implement the following functionality:
- Add to graph - Send the current focused symbol.
- Pull - Get a symbol's address from the socket and open it in the editor
- (Optional) Rename - detect rename in the editor and notify the socket.
- Frontend - Shows the call graph and allow you to interact with it. Should support:
- Layout the nodes
- Navigating the graph
- Import and export graph
- Undo, Redo
- Server - A middleware between the backend and the frontend. Support multiple of them in the same time, by multiplexing all the requests.
Need to support:
- TCP editor connection
- WebSocket editor connection
- Websocket frontend connection
As a user, you should run the server locally. It is a single python file which depends on websockets
library.
The frontend is a website which you can server using python -m http.server
.
As for the editors, you should install an extension or the equivalance for your editor.
In order to run Graffiti using the web frontend, you should:
- Run the python server
pip3 install -r server/requirements.txt
python3 server/main.py
- start a localhost server from web dir:
cd frontend/Web
python3 -m http.server 80
- From your web, enter http://localhost . Press connect to connect to the python server. The button will be green if successfully connected.
- Follow the usage instructions for the specific backend below.
Editor | Languages | add to graph | open in editor | Rename support | Field support | Add line to graph | Socket type |
---|---|---|---|---|---|---|---|
JEB | Java | ✅ | ✅ | ✅ | ✅ | ❌ | TCP |
Intellij | Java, Kotlin | ✅ | ✅ | ❌ | ✅ | ❌ | TCP |
VSCode | Depends on available language server | ✅ | ✅ | ❌ | ❌ | ✅ | TCP |
OpenGrok | * | ✅ | ✅ | ❌ | ❌ | ❌ | Websocket |
IDA | * | ✅ | ✅ | ✅ | ❌ | ❌ | TCP |
Jadx | Java | ✅ | ✅ | ✅ | ✅ | ❌ | TCP |
The common shortcuts are:
- Ctrl+Shift+A - Add a new node to the graph.
- Ctrl+Shift+X - Add a new node to the graph with a custom text on the edge.
Your cursor should be inside the function (or field in supported platforms) you want to add to the graph.
You can build all the backends using make backends
.
- Copy the scripts from
backend/jeb
into$JEB_INSTALLATION/scripts/graffiti
- Restart JEB
Press F2 and then double click graffiti
to connect the graffiti server. Now, you can use the shortcuts.
- An additional short is: Ctrl+Shift+Q - Add all of the xrefs of a node to the graph.
A Rename will be reflected in the opened graphs.
- Run
gradle jar
insidebackends/intellij
. - The generated plugin will be in
build/libs/
- In Intellij settings, go to plugins, then choose the setting icon and "Install plugin from Disk..."
- In Intellij settings, go to keymap. Press the find actions by shortcut (on the right) and search for Ctrl+Shift+A. Remove the non-Graffiti shortcut.
Go to Menu->Tools->"Graffiti: Connect to server". Now, you can use the shortcuts.
npm install -g @vscode/vsce
- Run
vsce package
insidebackends/vscode
- A vsix file will be generated on the same folder
- In the extensions section in the side menu, click the menu icon, then "Install from VSIX..."
- Install language servers for the languages you want to support
- Anycode is a language server based on treesitter, which supports: C#, CPP (C), Go, Java, Kotlin, PHP, Python, Rust, Typescript (Javascript)
- Go to the command palate (Ctrl+Shift+P). Choose "Graffiti: Connect to server". Now, you can use the shortcuts.
- Additional shortcut is (Ctrl+Shift+Q) for adding the current line as a node to the graph.
- Go to chrome://extensions
- Enable developer mode
- Select "Pack extension", and use
backends/opengrok
as the folder. - You will be informed where the crx is saved
- Go to chrome://extensions
- Enable developer mode
- Try drag and droping the Crx
- If it doesn't work, choose load unpacked, and select
backends/opengrok
as the folder. - Pin the graffiti extension to the toolbar
- Go to chrome://extensions/shortcuts and set "Ctrl+Shift+A" for "Add to graph", and "Ctrl+Shift+X" for "Add to graph with edge info"
- Press The graffiti icon in the toolbar, and then connect the server. Now you can use the shortcuts.
- In the previous popup window, you can change the settings of the plugin.
Inside a project, run file->execute script
, choose the graffiti.py
script.
Go to Options->"Graffiti: Connect to server". Now, you can use the shortcuts.
A Rename will be reflected in the opened graphs.
Note: Currently requires building Jadx from source to run the script.
Inside a project, Right click scripts
and choose Add scripts
. Select graffiti.py
script.
Go to Plguins->"Graffiti: Connect to server". Now, you can use the shortcut A (the only supported shortcut for now) to add to the graph.
A Rename will be reflected in the opened graphs.
Current relevant issues: #1905
The project uses a the following patch to support comments on the web frontend:
diff --git a/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js b/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js
index 5ed06723..dc0fde0e 100644
--- a/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js
+++ b/packages/mermaid/src/diagrams/flowchart/elk/flowRenderer-elk.js
@@ -902,6 +902,7 @@ export const draw = async function (text, id, _version, diagObj) {
});
insertChildren(graph.children, parentLookupDb);
+ if (window.elk_beforeCallback) window.elk_beforeCallback(id, graph)
log.info('after layout', JSON.stringify(graph, null, 2));
const g = await elk.layout(graph);
drawNodes(0, 0, g.children, svg, subGraphsEl, diagObj, 0);
The logo icons created by Freepik - Flaticon.