> ## Documentation Index
> Fetch the complete documentation index at: https://bloodhound.specterops.io/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/tTIczgde9H07oLXf/assets/enterprise-AND-community-edition-pill-tag.svg?fit=max&auto=format&n=tTIczgde9H07oLXf&q=85&s=ad49a576589f4d2a8081df77d07fdf56" 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.

<Info>
  At this time, OpenGraph nodes and edges are not supported in the Search or Pathfinding tab, so the Cypher tab **must** be used to query the data manually.
</Info>

# 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/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=875d1221ea0e944be36f4d2d7a97d136" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?w=280&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=4ea4d6fa29ffdaed6f1150abfec50080 280w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?w=560&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=2a0c1e5426a1b2f74daf66b703431aaa 560w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?w=840&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=abfe61a57a73a93c27cdd9a5d69d4651 840w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?w=1100&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=fca64f45a465dada08f9603f02877fef 1100w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?w=1650&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=5f8ebbb2cc1242e142f5727a86ce28e9 1650w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-1.png?w=2500&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=517949af66682e1e4f6f3ddac4eea131 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/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=a1a420e7c102e855b6f4d1d98ddb388c" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?w=280&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=645e031502eb7c6fdefb2b6031de7143 280w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?w=560&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=c78455d8075cb16f6e315fcdde71a783 560w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?w=840&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=298176a490c27379e94cf6ad5294b6ab 840w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?w=1100&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=4cf2c996e7306e475fa50c11c8bcda07 1100w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?w=1650&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=9e14572f9ef6da2076d418087c9dc5e5 1650w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-2.png?w=2500&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=1b9a93ac01268f298145e47475a9c3cc 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/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=c53a1c22d3aaa1264dfc1f561a53a466" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?w=280&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=605b918fc955ee99d44eb3e2321e4ef0 280w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?w=560&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=2ae19220e7e505d3a5ce92fd9170f692 560w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?w=840&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=4c0b3d273a0096ebc5d54b5deb2d308a 840w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?w=1100&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=6c808f94bcf7064d9ffa1763b21fa3c6 1100w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?w=1650&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=ac35f900b52bd04b4498e1cba6b0a8f1 1650w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-3.png?w=2500&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=a6d31d258b8c3f32d2f4c839ad1803a1 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/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=122a35311fff76546eeb79bafcbe53d1" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?w=280&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=6470da733bf4a8bf45edcf3a687beb91 280w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?w=560&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=83d48ee88a01272fc419319f31146f31 560w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?w=840&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=6cf31e664f884b1348823005553704fd 840w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?w=1100&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=712d5690888a7b32cb7143b9b21a1112 1100w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?w=1650&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=797ee8741c2bc9739428cd06e36dd2c5 1650w, https://mintcdn.com/specterops/t-Kb-qOfqpgsWjX4/assets/og-bp-4.png?w=2500&fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=36ef04ec6025fd7c5f037b4c7b476419 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/t-Kb-qOfqpgsWjX4/assets/og-bp-5.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=c6ae78b24a530de2901c1d7d1df1d9da" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-6.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=ffe864b0a03dc8ea3fd54bce83bfa501" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-7.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=3f3b457b4a642fb6cd8b8b581894411a" 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/t-Kb-qOfqpgsWjX4/assets/og-bp-8.png?fit=max&auto=format&n=t-Kb-qOfqpgsWjX4&q=85&s=8acd062a84e14176508279e4cdaa947c" 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.**

<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>
