PT-2025-36601 · Npm · @Escape.Tech/Graphql-Armor-Max-Depth

Published

2025-08-26

·

Updated

2025-08-26

CVSS v3.1

5.3

Medium

VectorAV:N/AC:L/PR:N/UI:N/S:U/C:N/I:N/A:L

Summary

A query depth restriction using the max-depth can be bypassed if ignoreIntrospection is enabled (which is the default configuration) by naming your query/fragment schema.

Details

In the countDepth function, we have the following code that calculates the depth of a used fragment:
typescript
  } else if (node.kind == Kind.FRAGMENT SPREAD) {
   if (this.visitedFragments.has(node.name.value)) {
    return this.visitedFragments.get(node.name.value) ?? 0;
   } else {
    this.visitedFragments.set(node.name.value, -1);
   }
   const fragment = this.context.getFragment(node.name.value);
   if (fragment) {
    let fragmentDepth;
    if (this.config.flattenFragments) {
     fragmentDepth = this.countDepth(fragment, parentDepth);
    } else {
     fragmentDepth = this.countDepth(fragment, parentDepth + 1);
    }
    depth = Math.max(depth, fragmentDepth);
    if (this.visitedFragments.get(node.name.value) === -1) {
     this.visitedFragments.set(node.name.value, fragmentDepth);
    }
   }
  }
which will calculate the depth of the fragment used in the current node, store the value in this.visitedFragments and re-use it in the future to avoid re-calculating the depth for the same fragment.
The issue arises when the same fragment is used multiple times, at different depths. The current caching takes into account the depth of the first occurrence, which means if the fragment is re-used later in a higher depth, this cached value is not updated.
So, for example, sending the following query with a max depth of 6:
graphql
query {
 books {
  author {
   ...Test
  }
 }
 books {
  author {
   books {
    author {
     ...Test
    }
   }
  }
 }
}
fragment Test on Author {
 books {
  title
 }
}
The first use of Test fragment does not exceed the defined limit, and this depth will be cached.
In the second use, the fragment is reused in a greater depth, but the countDepth function will still use the depth cached, without accounting for the increased depth.

PoC

Max depth: 6
graphql
query {
 books {
  author {
   ...Test
  }
 }
 books {
  author {
   books {
    author {
     ...Test
    }
   }
  }
 }
}
fragment Test on Author {
 books {
  title
 }
}

Impact

This issue affects applications using the GraphQL Armor Depth Limit plugin.

Fix

This is fixed in PR#824. We now store only the additional depth contributed by the fragment and add it to the parent depth where the fragment is used (parentDepth).

Fix

Resource Exhaustion

Found an issue in the description? Have something to add? Feel free to write us 👾

Weakness Enumeration

Related Identifiers

GHSA-224P-V68G-5G8F

Affected Products

@Escape.Tech/Graphql-Armor-Max-Depth