> ## Documentation Index
> Fetch the complete documentation index at: https://specterops-bp-2638-jira-forge.mintlify.site/llms.txt
> Use this file to discover all available pages before exploring further.

# OpenGraph Graph Theory

> Attack Graph Model Design Requirements and Examples

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/5Uew_kPPp8q6O153/assets/enterprise-AND-community-edition-pill-tag.svg?fit=max&auto=format&n=5Uew_kPPp8q6O153&q=85&s=1a659c7fe8da7cec98d35c0705fb8b7e" alt="Applies to BloodHound Enterprise and CE" width="482" height="45" data-path="assets/enterprise-AND-community-edition-pill-tag.svg" />

# Introduction

For several years, one of the biggest pain-points with contributing to BloodHound has been in getting nodes and edges ingested and correctly displayed in the GUI. BloodHound OpenGraph changes that. Now it is easy for anyone to add nodes and edges into BloodHound through the easy-to-use `/file-upload/` endpoint.

However, while the process of adding nodes and edges to the product is greatly simplified, the product will not function as expected without a well-designed attack graph model. This document seeks to educate users on attack graph model design theory, best-practices, and requirements.

An attack graph is a tool - a powerful force multiplier when wielded correctly, a frustrating and confusing hazard when not. This document aims to equip you with the knowledge and skills necessary to effectively wield this tool.

# Basic Attack Graph Vocabulary and Design Theory

Graphs are [well-understood](https://en.wikipedia.org/wiki/Graph_%28discrete_mathematics%29), well-studied mathematical constructs. You can find thousands of guides, tools, and academic papers that make use of graphs. This document will not replace a proper education or time spent working with graphs. But in this section we will touch on the most fundamental aspects of a graph you must understand in order to effectively get BloodHound to work with your nodes and edges.

Every graph is constructed from two fundamental components: vertices (nodes) and edges (relationships):

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=7fe70a495c3b3a94fb713541cdf0c33a" alt="Node1 -- Edge1 --> Node2" data-og-width="1012" width="1012" data-og-height="508" height="508" data-path="assets/og-bp-1.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?w=280&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=a5f32b7612dfc164cdfc01114d10be91 280w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?w=560&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=12b97e4716190ea0f1729ba270820b1a 560w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?w=840&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=50c53efd572f761a2557ab5bed0b428e 840w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?w=1100&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=d9b3666f1377a17f40b67ed9b0763669 1100w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?w=1650&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=f174f0801b55d416ef5ba4309dcdecae 1650w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-1.png?w=2500&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=bd2b92649ccd87dd187557cc5c9be086 2500w" />

The above graph has two nodes and one edge. The edge is **directed**. The source node of the edge is “Node 1”. The destination node of the edge is “Node 2”.

**Every** edge in a BloodHound attack graph is **directed**, and is **one-way**. There are no bi-directional (“two-way”) edges in a BloodHound graph.

In a BloodHound attack graph, the direction of the **edge** must match the direction of **access** or **attack**. Let's look at an example with Active Directory group memberships.

In the BloodHound attack graph, we model Active Directory security group memberships like this:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=0361caf2489a9e33113e415a38a84047" alt="User -- MemberOf --> Group" data-og-width="1024" width="1024" data-og-height="432" height="432" data-path="assets/og-bp-2.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?w=280&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=17701e524052ec6b121adc93ec9475ed 280w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?w=560&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=555262edeec5c07429b681a6116d6fde 560w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?w=840&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=ad93fc3e4b26a447ddbb62461d67ecf3 840w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?w=1100&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=edb1a5fcda720f283101b8485eaa003f 1100w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?w=1650&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=e6a2657d71c57effd7ef87038d7eb05c 1650w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-2.png?w=2500&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=b4450fff57520f3ec9387e57bd7b41ba 2500w" />

Think about the direction of the edge. Now think for a moment and try to figure out why we don't model AD security group memberships like this instead:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=0c2f5f482e1046a1f9aa85adf9e510c6" alt="Group -- HasMember --> User" data-og-width="1018" width="1018" data-og-height="424" height="424" data-path="assets/og-bp-3.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?w=280&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=9b70022afa3c85c4ee3b86875d79db5d 280w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?w=560&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=c1c063ec056480537eedf2afa5b4c32e 560w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?w=840&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=135421393ba000efd2eff4ccc4171367 840w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?w=1100&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=7c1bacac6aea10ff4ee9158e1d235b67 1100w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?w=1650&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=26dfcc1fa1a95fbee0c52240c561cc18 1650w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-3.png?w=2500&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=27bd3dec566c35ed3ae6becd928449cb 2500w" />

This seems perfectly reasonable at first glance, does it not? But remember that we are constructing an **attack graph** in order to discover **attack paths**. Edge directionality must serve attack path discovery.

The direction of the edge going from the group to the user does not expose any attack path. Just because a user is a member of a group does not mean the group has any “control” of the user. But when the direction of the edge is from the user to the group, that DOES serve attack path discovery.

Why? Because in Windows and Active Directory, members of security groups gain the privileges held by those groups. Let's extend the model a bit to make this easier to see:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=3086d801b211ce20b43a864d17fe0f2b" alt="User -- MemberOf -> Group -- GenericAll --> Domain" data-og-width="1582" width="1582" data-og-height="414" height="414" data-path="assets/og-bp-4.png" data-optimize="true" data-opv="3" srcset="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?w=280&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=9bd8200426f02b6676660771af34717a 280w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?w=560&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=5a7fb0977e4fb2e25c6cde545902ba3e 560w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?w=840&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=fd2b7d9b8ad1b81012fab01d23ad693a 840w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?w=1100&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=f9f3c0b5f7e650ecdead31873fdb8b8c 1100w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?w=1650&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=d54dc6ce674f63f7483f14d58a8866a3 1650w, https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-4.png?w=2500&fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=9eb85c92785894745c2294dbe4b62826 2500w" />

The user is a member of a group, and the group has full control of the domain. When the user authenticates to Active Directory, their Kerberos ticket will include the SID of the group. When the user uses that ticket to perform some action against the domain object, the security reference monitor will inspect the ticket, see the group SID, and grant the user all the permissions against the domain that the group has.

**In reality the process is much more involved than this, but work with me here, people.**

The above diagram shows a **path** connecting two **non-adjacent** nodes. **Adjacent** nodes are those that are connected together by an edge. In the above diagram, the adjacent nodes are:

1. “User” and “Group” via the “MemberOf” edge

2. “Group” and “Domain” via the “GenericAll” edge

The “User” and “Domain” nodes are non-adjacent, yet there is a **path** connecting the “User” node to the “Domain” node.

When designing your attack graph model, you **must** be aware of the **patterns** that will emerge from your design. There are many examples out there of people who want to make a contribution to the BloodHound graph who do not seem to be aware of this. Instead of proposing nodes/edges that create multi-node patterns, they propose nodes/edges that result **only** in one-to-one patterns:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-5.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=5d3dd9c91532333ae0cd7ba010994408" alt="Badly connected nodes" width="1012" height="772" data-path="assets/og-bp-5.png" />

In the above graph there are two patterns:

1. From the red (top left) to the pink (top right) node

2. From the blue (bottom left) to the green (bottom right) node

What's wrong with this design?

Think of the graph as a map of **one-way streets**. In the above graph we have two one-way streets. But this map kinda sucks, doesn't it? You can only start in two places and you can only go to two places. You can't go from the red (top left) node to the blue (bottom left) node because there is no **path** connecting those nodes.

This is a much better map:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-6.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=4e7e4f6b18c4ff01955813543f57150e" alt="Well connected nodes" width="1002" height="770" data-path="assets/og-bp-6.png" />

Now is there a **path** from the red (top left) node to the blue (bottom left) node? Yes! It goes **through** the green (bottom right) node!

The difference in the two graphs is the level of **connectedness**, or how well-linked the nodes are to one another.

Let's belabor the point a little more to make it even more clear. The top model would be analogous to having a node represent both a **person** and the **address** where they live, with the edge representing the fact that they live at that address:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-7.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=8703fe8e8fb3b2e949727128c931adba" alt="Badly connected nodes" width="980" height="808" data-path="assets/og-bp-7.png" />

While the bottom graph would be analogous to having the nodes represent the **addresses** and the edges represent **streets**:

<img noZoom src="https://mintcdn.com/specterops-bp-2638-jira-forge/T2K6v3-pm25xZeG3/assets/og-bp-8.png?fit=max&auto=format&n=T2K6v3-pm25xZeG3&q=85&s=2deb3dd5c9658893133fe8f7998e9cbf" alt="Well connected nodes" width="1004" height="826" data-path="assets/og-bp-8.png" />

It should be obvious that for the sake of **pathfinding**, the **second** model is the **only** model that will work.

**This is actually how Google Maps works under the hood — it is a graph where locations are nodes and streets are edges.**

<Warning>
  If your graph model does not create paths connecting non-adjacent nodes, you should use a relational database instead. A graph database is the wrong tool for data that only produces one-to-one patterns.
</Warning>

<Note>
  This article is adapted from [Andy Robbins](https://www.linkedin.com/in/robbinsandy/)' blog post, “[Attack Graph Model Design Requirements and Examples](https://specterops.io/blog/2025/08/01/attack-graph-model-design-requirements-and-examples/),” which goes beyond what's described here.
</Note>
