diff --git a/.github/workflows/docs.yml b/.github/workflows/docs.yml new file mode 100644 index 000000000..4e53a13e4 --- /dev/null +++ b/.github/workflows/docs.yml @@ -0,0 +1,34 @@ +name: generate-docs +on: + workflow_dispatch: {} # allow manually triggering the workflow + push: + branches: + - main + paths: + - "docs/**" + - mkdocs.yml + +permissions: + contents: read + +jobs: + deploy: + permissions: + contents: write # for mkdocs gh-deploy to publish docs + runs-on: ubuntu-latest + steps: + - name: Harden Runner + uses: step-security/harden-runner@cba0d00b1fc9a034e1e642ea0f1103c282990604 + with: + egress-policy: block + allowed-endpoints: > + files.pythonhosted.org:443 + github.com:443 + pypi.org:443 + + - uses: actions/checkout@c85c95e3d7251135ab7dc9ce3241c5835cc595a9 + - uses: actions/setup-python@61a6322f88396a6271a6ee3565807d608ecaddd1 + with: + python-version: 3.x + - run: pip install mkdocs-material mkdocs-awesome-pages-plugin + - run: mkdocs gh-deploy --force \ No newline at end of file diff --git a/docs/application/DEVELOPER.md b/DEVELOPER.md similarity index 100% rename from docs/application/DEVELOPER.md rename to DEVELOPER.md diff --git a/docs/application/Architecture.excalidraw b/docs/Architecture.excalidraw similarity index 100% rename from docs/application/Architecture.excalidraw rename to docs/Architecture.excalidraw diff --git a/docs/architecture.md b/docs/architecture.md new file mode 100644 index 000000000..a9e9c7dde --- /dev/null +++ b/docs/architecture.md @@ -0,0 +1,23 @@ +# Architecture + +KubeHound works in 3 steps: + +1. Connect to your Kubernetes cluster and read API resources (pods, nodes, permissions...) +2. Compute attack paths +3. Write the results to a local graph database (JanusGraph) + +After the initial ingestion is done, you use a compatible client (such as [gdotv](https://gdotv.com/)) to visualize and query attack paths in your cluster. + + +
+[![KubeHound architecture](./images/kubehound-high-level.png)](./images/kubehound-high-level.png) +

KubeHound architecture (click to enlarge)

+
+ +Under the hood, KubeHound leverages a caching and persistence layer (Redis and MongoDB) while computing attack paths. As an end user, this is mostly transparent to you. + + +
+[![KubeHound architecture](./images/kubehound-detailed.png)](./images/kubehound-detailed.png) +

KubeHound detailed architecture (click to enlarge)

+
diff --git a/docs/comparison.md b/docs/comparison.md new file mode 100644 index 000000000..a5be0926a --- /dev/null +++ b/docs/comparison.md @@ -0,0 +1,34 @@ +# Comparison with other tools + +## Lyft's [Cartography](https://github.com/lyft/cartography) + +Cartography has a [Kubernetes module](https://lyft.github.io/cartography/modules/kubernetes/index.html). While useful to vizualize a cluster, it only has a few types of resources and relationships. Consequently, it cannot be used for mapping attack paths in a Kubernetes cluster. + +## [BloodHound](https://github.com/SpecterOps/BloodHound) + +BloodHound is one of the first projects (and certainly the most popular) that introduced attack graphs mapping. It is currently focused on Active Directory and [Azure](https://bloodhound.readthedocs.io/en/latest/data-collection/azurehound.html) environments, and does not support Kubernetes. + +## [BOtB](https://github.com/brompwnie/botb) + +BOtB is a pentesting tool that attempts to exploit common weaknesses. It runs from inside a compromised container. While very useful when performing a blackbox assessment, it doesn't have a full view of the cluster and does not attempt to find cluster-wide attack paths. + + +## [BOtB](https://github.com/brompwnie/botb) + +BOtB is a pentesting tool that attempts to exploit common weaknesses. It runs from inside a compromised container. While very useful when performing a blackbox assessment, it doesn't have a full view of the cluster and does not attempt to find cluster-wide attack paths. + +## [peirates](https://github.com/inguardians/peirates) + +Similarly to BOtB, peirates is an offensive tool running from inside a pod. It doesn't have a full view of the cluster and does not attempt to find cluster-wide attack paths. + +## [rbac-police](https://github.com/PaloAltoNetworks/rbac-police) + +rbac-police allows you to retrieve the permissions associated to a specific identity in the cluster, which makes it easier to understand who has access to what. However, it does not look for attack paths in the cluster - it's focused on showing effective permissions of an identity. + +## [KubiScan](https://github.com/cyberark/KubiScan) + +KubeScan scans a Kubernetes cluster for risky permissions that allow an identity to escalate its privileges inside the cluster. It does not look for other types of attacks in the cluster, nor does it attempt to build an attack graph. + +## [kdigger](https://github.com/quarkslab/kdigger) + +Similarly to BOtB and peirates, kdigger runs from a compromised pod and attempts to retrieve information about the cluster and potential weaknesses. It does not have a full cluster view, nor does it attempt to build attack paths on a graph. \ No newline at end of file diff --git a/docs/contributing.md b/docs/contributing.md new file mode 100644 index 000000000..82a41e3ad --- /dev/null +++ b/docs/contributing.md @@ -0,0 +1,3 @@ +# Contributing + +Contributions are welcome! Refer to the [CONTRIBUTING](https://github.com/DataDog/KubeHound/blob/main/CONTRIBUTING.md) guide on how you can contribute to KubeHound, and what to expect. \ No newline at end of file diff --git a/docs/faq.md b/docs/faq.md new file mode 100644 index 000000000..0f106984c --- /dev/null +++ b/docs/faq.md @@ -0,0 +1,3 @@ +# Frequently Asked Questions + +TODO \ No newline at end of file diff --git a/docs/images/kubehound-detailed.excalidraw b/docs/images/kubehound-detailed.excalidraw new file mode 100644 index 000000000..3b9e343f1 --- /dev/null +++ b/docs/images/kubehound-detailed.excalidraw @@ -0,0 +1,1038 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://marketplace.visualstudio.com/items?itemName=pomdtr.excalidraw-editor", + "elements": [ + { + "type": "rectangle", + "version": 162, + "versionNonce": 212841042, + "isDeleted": false, + "id": "YijJ7E4UNzzVIJhvenmCh", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "dashed", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 333.2221984863281, + "y": 238.3791046142578, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 660.0268249511719, + "height": 386.2890319824219, + "seed": 17202382, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "diamond", + "version": 1039, + "versionNonce": 488166350, + "isDeleted": false, + "id": "zUmguPGAP2vTB_V4EW4_A", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 602.6229421293422, + "y": 344.19214153940055, + "strokeColor": "#c92a2a", + "backgroundColor": "#fd8888", + "width": 112.64736525303451, + "height": 36.77344700318558, + "seed": 112715238, + "groupIds": [ + "WKktN-osppvFakfT9AgQB" + ], + "roundness": null, + "boundElements": [ + { + "id": "52v-YF8sx9XcALQAikpWa", + "type": "arrow" + } + ], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "diamond", + "version": 1078, + "versionNonce": 2087294994, + "isDeleted": false, + "id": "8P6cE1eP8oyX8iy3tm43d", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 602.6229421293422, + "y": 335.320361611142, + "strokeColor": "#c92a2a", + "backgroundColor": "#fd8888", + "width": 112.64736525303451, + "height": 36.77344700318558, + "seed": 1301173114, + "groupIds": [ + "WKktN-osppvFakfT9AgQB" + ], + "roundness": null, + "boundElements": [ + { + "id": "52v-YF8sx9XcALQAikpWa", + "type": "arrow" + } + ], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "diamond", + "version": 1156, + "versionNonce": 178858510, + "isDeleted": false, + "id": "sS3kPIsIpHNM52XDjW3FY", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 602.6229421293422, + "y": 322.68980201642995, + "strokeColor": "#c92a2a", + "backgroundColor": "#fd8888", + "width": 112.64736525303451, + "height": 36.77344700318558, + "seed": 1473132838, + "groupIds": [ + "WKktN-osppvFakfT9AgQB" + ], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "diamond", + "version": 1198, + "versionNonce": 794871250, + "isDeleted": false, + "id": "Y0KS8wt1zivKG_AL7nMwL", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 602.6229421293422, + "y": 310.1581297191326, + "strokeColor": "#c92a2a", + "backgroundColor": "#fd8888", + "width": 112.64736525303451, + "height": 36.77344700318558, + "seed": 2116670522, + "groupIds": [ + "WKktN-osppvFakfT9AgQB" + ], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "line", + "version": 4593, + "versionNonce": 139769934, + "isDeleted": false, + "id": "z7cEO4y046S-TNzT7BOKA", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 819.762044259025, + "y": 308.72438456621495, + "strokeColor": "#0a11d3", + "backgroundColor": "#228be6", + "width": 60.7382680010908, + "height": 78.39237751899326, + "seed": 1884246331, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0.20028361605803496, + 59.24853403628056 + ], + [ + 0.009372803058215485, + 65.99372235184508 + ], + [ + 3.1281551336420153, + 68.90823481425232 + ], + [ + 13.989158982750121, + 71.37485607708486 + ], + [ + 32.347332056685445, + 72.1429250910517 + ], + [ + 49.88735243619048, + 70.91659040084292 + ], + [ + 59.20649440816879, + 67.98383316912404 + ], + [ + 60.52083327976382, + 65.51148805709545 + ], + [ + 60.7054274163297, + 60.08098613254131 + ], + [ + 60.56054248355994, + 4.970621305146061 + ], + [ + 60.23392533882228, + -0.23629276019543785 + ], + [ + 56.33391578766731, + -3.1464712835087236 + ], + [ + 48.12132796908886, + -4.831892693901661 + ], + [ + 29.405949953842192, + -6.249452427941555 + ], + [ + 14.400963346206083, + -5.404158366627446 + ], + [ + 2.599643283560067, + -2.537024049408259 + ], + [ + -0.03284058476110599, + -0.03560029776626749 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 2327, + "versionNonce": 1588345746, + "isDeleted": false, + "id": "ohw8PGTCNgicQYX0jkIyS", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 820.2353016855415, + "y": 353.67748994903775, + "strokeColor": "#0a11d3", + "backgroundColor": "transparent", + "width": 60.80127009119773, + "height": 6.745993515778707, + "seed": 2079699957, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 1.6018523995224774, + 2.6890657551102657 + ], + [ + 8.509979555390247, + 4.945167211490023 + ], + [ + 17.702324706273377, + 6.31144861163304 + ], + [ + 32.10328143659777, + 6.435951684337329 + ], + [ + 48.908703758524034, + 5.566101969596352 + ], + [ + 58.72326031034886, + 2.401608343351324 + ], + [ + 60.80127009119773, + -0.31004183144137804 + ] + ] + }, + { + "type": "line", + "version": 2414, + "versionNonce": 1576304270, + "isDeleted": false, + "id": "YnYxUPbYLZw85Zllj-aD0", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 819.4822463778527, + "y": 330.9055028153093, + "strokeColor": "#0a11d3", + "backgroundColor": "transparent", + "width": 60.80127009119773, + "height": 6.745993515778707, + "seed": 778531291, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 1.6018523995224774, + 2.6890657551102657 + ], + [ + 8.509979555390247, + 4.945167211490023 + ], + [ + 17.702324706273377, + 6.31144861163304 + ], + [ + 32.10328143659777, + 6.435951684337329 + ], + [ + 48.908703758524034, + 5.566101969596352 + ], + [ + 58.72326031034886, + 2.401608343351324 + ], + [ + 60.80127009119773, + -0.31004183144137804 + ] + ] + }, + { + "type": "ellipse", + "version": 5434, + "versionNonce": 142801234, + "isDeleted": false, + "id": "McNYp6GMfDJE24wy776X_", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 818.7105453283045, + "y": 303.19927646730434, + "strokeColor": "#0a11d3", + "backgroundColor": "#fff", + "width": 60.34868280304699, + "height": 12.205067075133375, + "seed": 1667591509, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 801, + "versionNonce": 1883339982, + "isDeleted": false, + "id": "yLzkifW9o_pPJq3RR6d43", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 867.9459294355252, + "y": 320.12003753676214, + "strokeColor": "#0a11d3", + "backgroundColor": "#fff", + "width": 8.844677956739957, + "height": 9.599183137539441, + "seed": 32669307, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 850, + "versionNonce": 1279000338, + "isDeleted": false, + "id": "BQysi4ABUsDr_OF23i4hb", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 867.9459294355252, + "y": 341.1902264732981, + "strokeColor": "#0a11d3", + "backgroundColor": "#fff", + "width": 8.844677956739957, + "height": 9.599183137539441, + "seed": 727120565, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "ellipse", + "version": 904, + "versionNonce": 617919246, + "isDeleted": false, + "id": "wbHzA5bMb9HwU5QOowF_I", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 867.9459294355252, + "y": 364.090722235984, + "strokeColor": "#0a11d3", + "backgroundColor": "#fff", + "width": 8.844677956739957, + "height": 9.599183137539441, + "seed": 364050203, + "groupIds": [ + "Q9SaOana8lZmAZmlfmiyh" + ], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 313, + "versionNonce": 479410177, + "isDeleted": false, + "id": "dMMKC3N4WzJFN4q0zOtiG", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 588.3804321289062, + "y": 269.3304443359375, + "strokeColor": "#000000", + "backgroundColor": "#fd8888", + "width": 154.4921875, + "height": 23, + "seed": 844306638, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691756929216, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 2, + "text": "In-memory cache", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "In-memory cache", + "lineHeight": 1.15, + "baseline": 18 + }, + { + "type": "text", + "version": 501, + "versionNonce": 1817240210, + "isDeleted": false, + "id": "cn-MGnUL6NODmYNkVRwLM", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 765.6129150390625, + "y": 269.3623046875, + "strokeColor": "#000000", + "backgroundColor": "#fd8888", + "width": 177.861328125, + "height": 23, + "seed": 844306638, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691671685184, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 2, + "text": "Storage (MongoDB)", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "Storage (MongoDB)", + "lineHeight": 1.15, + "baseline": 18 + }, + { + "type": "diamond", + "version": 335, + "versionNonce": 357790414, + "isDeleted": false, + "id": "YNXvVaNMg8wGXD55ABs1d", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 347.86181640625, + "y": 445.6231689453125, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 190.18695068359375, + "height": 95.24267578125, + "seed": 592178962, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "iGnUCaKa6Dsd1W7mC7RJV" + } + ], + "updated": 1691671830024, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 222, + "versionNonce": 837576978, + "isDeleted": false, + "id": "iGnUCaKa6Dsd1W7mC7RJV", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 402.33238220214844, + "y": 481.933837890625, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 81.15234375, + "height": 23, + "seed": 1215986062, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691671830025, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 2, + "text": "ingestors", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "YNXvVaNMg8wGXD55ABs1d", + "originalText": "ingestors", + "lineHeight": 1.15, + "baseline": 18 + }, + { + "type": "text", + "version": 219, + "versionNonce": 1201300498, + "isDeleted": false, + "id": "C6RcEjDEBsRgNoPbKwfHe", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 580.0796508789062, + "y": 186.6113494873047, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 147.90234375, + "height": 32.199999999999996, + "seed": 1997638162, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691671784536, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 2, + "text": "KubeHound", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "KubeHound", + "lineHeight": 1.15, + "baseline": 26 + }, + { + "type": "rectangle", + "version": 253, + "versionNonce": 2052968658, + "isDeleted": false, + "id": "CHmgrq0ai57DieZpBpP-A", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 618.920166015625, + "y": 454.7411804199219, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 284, + "height": 68, + "seed": 725437710, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "71oRt_ozcb2XVp01OA-KJ" + }, + { + "id": "UOYege4suyYxnHAQhNyIS", + "type": "arrow" + }, + { + "id": "e7fIu70Co648e4-_nHO-q", + "type": "arrow" + }, + { + "id": "00j74CecUimyC8XrJnRus", + "type": "arrow" + } + ], + "updated": 1691671791332, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 315, + "versionNonce": 2011318798, + "isDeleted": false, + "id": "71oRt_ozcb2XVp01OA-KJ", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 648.635009765625, + "y": 477.2411804199219, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 224.5703125, + "height": 23, + "seed": 1333424398, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691671736178, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 2, + "text": "Attack paths computation", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "CHmgrq0ai57DieZpBpP-A", + "originalText": "Attack paths computation", + "lineHeight": 1.15, + "baseline": 18 + }, + { + "type": "arrow", + "version": 223, + "versionNonce": 1690435086, + "isDeleted": false, + "id": "UOYege4suyYxnHAQhNyIS", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 531.4299926757812, + "y": 493.5558776855469, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 86.34759521484375, + "height": 0.16207023815331922, + "seed": 1938194002, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671746602, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "CHmgrq0ai57DieZpBpP-A", + "focus": -0.12793696178654337, + "gap": 1.142578125 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 86.34759521484375, + -0.16207023815331922 + ] + ] + }, + { + "type": "arrow", + "version": 370, + "versionNonce": 1054260878, + "isDeleted": false, + "id": "52v-YF8sx9XcALQAikpWa", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 660.0524135839318, + "y": 385.87452519338433, + "strokeColor": "#495057", + "backgroundColor": "transparent", + "width": 0, + "height": 68.95414912302192, + "seed": 267686354, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671764823, + "link": null, + "locked": false, + "startBinding": { + "elementId": "8P6cE1eP8oyX8iy3tm43d", + "focus": -0.019632750851979457, + "gap": 13.4435055785132 + }, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": "triangle", + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 0, + 68.95414912302192 + ] + ] + }, + { + "type": "arrow", + "version": 433, + "versionNonce": 523329870, + "isDeleted": false, + "id": "e7fIu70Co648e4-_nHO-q", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 852.5105590820314, + "y": 384.1728582998172, + "strokeColor": "#495057", + "backgroundColor": "transparent", + "width": 0, + "height": 68.95414912302192, + "seed": 267686354, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671775504, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "CHmgrq0ai57DieZpBpP-A", + "focus": 0.6450027680732842, + "gap": 1.6141729970827328 + }, + "lastCommittedPoint": null, + "startArrowhead": "triangle", + "endArrowhead": "triangle", + "points": [ + [ + 0, + 0 + ], + [ + 0, + 68.95414912302192 + ] + ] + }, + { + "type": "diamond", + "version": 2, + "versionNonce": 2137374482, + "isDeleted": false, + "id": "FC5dvllDxDBtclEx-cTZT", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 854.1347045898438, + "y": 394.9568176269531, + "strokeColor": "#495057", + "backgroundColor": "transparent", + "width": 0.08404541015625, + "height": 0.231964111328125, + "seed": 2061221070, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671769490, + "link": null, + "locked": false + }, + { + "type": "arrow", + "version": 380, + "versionNonce": 1249618706, + "isDeleted": false, + "id": "00j74CecUimyC8XrJnRus", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 905.4414367675781, + "y": 489.00274331892626, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 140.88473510742188, + "height": 0.30871324352858664, + "seed": 1938194002, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691671822447, + "link": null, + "locked": false, + "startBinding": { + "elementId": "CHmgrq0ai57DieZpBpP-A", + "focus": 0.003452398665398077, + "gap": 2.521270751953125 + }, + "endBinding": { + "elementId": "Ph0DcnJVgqfDBTazrLIPw", + "focus": -0.05833994666563345, + "gap": 13.0677490234375 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 140.88473510742188, + 0.30871324352858664 + ] + ] + }, + { + "type": "text", + "version": 94, + "versionNonce": 1615340882, + "isDeleted": false, + "id": "Ph0DcnJVgqfDBTazrLIPw", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "dashed", + "roughness": 0, + "opacity": 100, + "angle": 0, + "x": 1059.3939208984375, + "y": 477.3401184082031, + "strokeColor": "#000000", + "backgroundColor": "transparent", + "width": 165.68359375, + "height": 23, + "seed": 1261480338, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "00j74CecUimyC8XrJnRus", + "type": "arrow" + } + ], + "updated": 1691671822446, + "link": null, + "locked": false, + "fontSize": 20, + "fontFamily": 2, + "text": "To graph database", + "textAlign": "left", + "verticalAlign": "top", + "containerId": null, + "originalText": "To graph database", + "lineHeight": 1.15, + "baseline": 18 + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": {} +} \ No newline at end of file diff --git a/docs/images/kubehound-detailed.png b/docs/images/kubehound-detailed.png new file mode 100644 index 000000000..ad825ccf7 Binary files /dev/null and b/docs/images/kubehound-detailed.png differ diff --git a/docs/images/kubehound-high-level.excalidraw b/docs/images/kubehound-high-level.excalidraw new file mode 100644 index 000000000..311de5492 --- /dev/null +++ b/docs/images/kubehound-high-level.excalidraw @@ -0,0 +1,2439 @@ +{ + "type": "excalidraw", + "version": 2, + "source": "https://marketplace.visualstudio.com/items?itemName=pomdtr.excalidraw-editor", + "elements": [ + { + "type": "line", + "version": 590, + "versionNonce": 884923627, + "isDeleted": false, + "id": "1q7NTtjuzdQ2AuN5MM2vA", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 609.9926247584258, + "y": 464.7894662263899, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 170.5479126392067, + "height": 165.8878192491774, + "seed": 1920910379, + "groupIds": [ + "9J-kkt8Ft7Da_84YzLM7a", + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + -4.396962054402086, + 1.1646874339146518 + ], + [ + -63.9402715195358, + 29.56596636773484 + ], + [ + -63.9402715195358, + 29.56596636773484 + ], + [ + -67.88778686931525, + 32.68804950762777 + ], + [ + -70.0842642471441, + 37.216572224302446 + ], + [ + -84.65888566945968, + 100.95916519983552 + ], + [ + -84.65888566945968, + 100.95916519983552 + ], + [ + -84.76907687141536, + 105.46565159474414 + ], + [ + -83.05610475455529, + 109.65158466072563 + ], + [ + -82.41499236471051, + 110.54913805780342 + ], + [ + -41.2769501625808, + 161.69922653927426 + ], + [ + -41.2769501625808, + 161.69922653927426 + ], + [ + -37.31206815320828, + 164.80126809034192 + ], + [ + -32.397540410633006, + 165.88781924917737 + ], + [ + 33.58360912709668, + 165.88781924917697 + ], + [ + 33.58360912709668, + 165.88781924917697 + ], + [ + 38.509483664717834, + 164.7592029175085 + ], + [ + 42.47369821555802, + 161.6511389543416 + ], + [ + 83.590366618572, + 110.49037113635723 + ], + [ + 83.590366618572, + 110.49037113635723 + ], + [ + 85.77883576779129, + 105.97187048787097 + ], + [ + 85.7274057950955, + 100.94848586332202 + ], + [ + 71.05662649840332, + 37.15246053949655 + ], + [ + 71.05662649840332, + 37.15246053949655 + ], + [ + 68.85613290539939, + 32.62393782282196 + ], + [ + 64.91262485436425, + 29.501854682928673 + ], + [ + 5.465486737767181, + 1.1700299681670852 + ], + [ + 5.465486737767181, + 1.1700299681670852 + ], + [ + -0.0534221580855956, + 0 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 1114, + "versionNonce": 551156549, + "isDeleted": false, + "id": "NAKzxrkH2FHtA4wZYAHiB", + "fillStyle": "solid", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 609.9906980892406, + "y": 485.91078838407805, + "strokeColor": "#326ce5", + "backgroundColor": "#fff", + "width": 124.88167478224784, + "height": 122.53590960399487, + "seed": 302712523, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + -2.5591124386745037, + 1.2615221012119529 + ], + [ + -3.4994086573241248, + 3.9535275716717213 + ], + [ + -3.4994086573241248, + 4.957935475127422 + ], + [ + -3.4994086573241248, + 4.957935475127422 + ], + [ + -3.007889137213411, + 8.339805194456696 + ], + [ + -2.6552774183318633, + 14.804355072319312 + ], + [ + -3.819967081354406, + 16.647551998958733 + ], + [ + -3.9001028660344446, + 18.154166401693924 + ], + [ + -3.9001028660344446, + 18.154166401693924 + ], + [ + -10.32725468568539, + 19.147892421084155 + ], + [ + -16.83713569759547, + 21.134422246167567 + ], + [ + -22.949822836761424, + 24.04505605161777 + ], + [ + -28.566654522909538, + 27.826198445836983 + ], + [ + -33.58894879535297, + 32.42425913233075 + ], + [ + -34.87117230126708, + 31.516016208532427 + ], + [ + -34.87117230126708, + 31.516016208532427 + ], + [ + -36.981502946299365, + 31.302312290880103 + ], + [ + -41.77381954115481, + 26.974801589540924 + ], + [ + -44.10853598790492, + 24.490495457496134 + ], + [ + -44.89923997370828, + 23.860068136156226 + ], + [ + -44.89923997370828, + 23.860068136156226 + ], + [ + -47.37286422168759, + 22.925111585763457 + ], + [ + -50.209787167717224, + 24.185966228443267 + ], + [ + -50.8575786010987, + 26.9921657015825 + ], + [ + -49.333600086321944, + 29.437743953454792 + ], + [ + -48.59632080615586, + 30.02542845322292 + ], + [ + -48.59632080615586, + 30.02542845322292 + ], + [ + -45.6472036854915, + 31.751088989419035 + ], + [ + -40.315285080696654, + 35.49091519099007 + ], + [ + -39.604721974676664, + 37.54247381947327 + ], + [ + -38.440032311654235, + 38.610993407734945 + ], + [ + -38.440032311654235, + 38.610993407734945 + ], + [ + -42.453414609326, + 46.02661055262045 + ], + [ + -45.03080063598091, + 53.98366394463282 + ], + [ + -46.12661978771645, + 62.27630631540628 + ], + [ + -45.69528872287185, + 70.69866492105825 + ], + [ + -47.19121614643827, + 71.12607275636286 + ], + [ + -47.19121614643827, + 71.12607275636286 + ], + [ + -48.71920195995935, + 72.66474300150108 + ], + [ + -55.09826874223004, + 73.71723848988879 + ], + [ + -58.517536519770886, + 73.98436456562676 + ], + [ + -59.46317495422912, + 74.19807357838249 + ], + [ + -59.57002691305527, + 74.19807357838249 + ], + [ + -59.57002691305527, + 74.19807357838249 + ], + [ + -61.323967820110155, + 74.94968283657575 + ], + [ + -62.39454328255286, + 76.39698768936387 + ], + [ + -61.90248113394084, + 79.94255798257318 + ], + [ + -61.63761473651582, + 80.24590005255747 + ], + [ + -59.93547829224923, + 81.20498191706895 + ], + [ + -57.983274178088365, + 81.12742425465802 + ], + [ + -57.91382027747376, + 81.12742425465802 + ], + [ + -56.95215264803821, + 81.02057993848678 + ], + [ + -56.95215264803821, + 81.02057993848678 + ], + [ + -53.7572757673185, + 79.85589027546439 + ], + [ + -47.54917008112836, + 78.03940442786777 + ], + [ + -45.508295884262296, + 78.76065706060814 + ], + [ + -43.905516501869705, + 78.49352079466358 + ], + [ + -43.905516501869705, + 78.49352079466358 + ], + [ + -40.60528817024521, + 86.23996299672599 + ], + [ + -35.999798823142164, + 93.20705819823723 + ], + [ + -30.2182679228674, + 99.23250697872516 + ], + [ + -23.389922574382894, + 104.15405067854242 + ], + [ + -24.041718758957046, + 105.51640996913648 + ], + [ + -24.041718758957046, + 105.51640996913648 + ], + [ + -23.721160334926928, + 107.52522577604792 + ], + [ + -26.94275338979269, + 113.35935342767391 + ], + [ + -28.855404217046726, + 116.20695571021739 + ], + [ + -29.309525678945842, + 117.16862588720463 + ], + [ + -29.309525678945842, + 117.16862588720463 + ], + [ + -29.792684134543613, + 119.01465314372642 + ], + [ + -29.307171741221893, + 120.74817032920896 + ], + [ + -28.046964176660957, + 122.0337464130935 + ], + [ + -26.206021833231233, + 122.53590960399487 + ], + [ + -25.836830648216146, + 122.51657878208012 + ], + [ + -24.033276172791297, + 121.78030578481567 + ], + [ + -22.88237385929482, + 120.20856831926949 + ], + [ + -22.438936829012828, + 119.30032539547108 + ], + [ + -22.438936829012828, + 119.30032539547108 + ], + [ + -21.41315496721958, + 116.0413342823939 + ], + [ + -20.167662060985002, + 112.55195278836115 + ], + [ + -18.66171766433351, + 109.65158561605767 + ], + [ + -17.13373185081244, + 108.90362445182609 + ], + [ + -16.332343433392012, + 107.44508999136787 + ], + [ + -16.332343433392012, + 107.44508999136787 + ], + [ + -8.233534077406446, + 109.73426895238698 + ], + [ + 0.07808755315407723, + 110.51101745025544 + ], + [ + 8.393408228709504, + 109.77484635505681 + ], + [ + 16.50329433926609, + 107.52522577604792 + ], + [ + 17.21386763549272, + 108.8875952568486 + ], + [ + 17.21386763549272, + 108.8875952568486 + ], + [ + 19.030343292882677, + 109.98282847170483 + ], + [ + 21.466577125305406, + 115.9665381659706 + ], + [ + 22.503043418715677, + 119.23086894730498 + ], + [ + 22.946480448997775, + 120.1391118711032 + ], + [ + 22.946480448997775, + 120.1391118711032 + ], + [ + 24.06736241385906, + 121.68340711028317 + ], + [ + 25.715597760329914, + 122.4074009085981 + ], + [ + 29.06454801724695, + 121.14341787249288 + ], + [ + 29.288141530401873, + 120.8389956406094 + ], + [ + 29.859924212775155, + 118.97848810051305 + ], + [ + 29.3736271735454, + 117.09382977078147 + ], + [ + 28.90882128002913, + 116.13215959379427 + ], + [ + 28.90882128002913, + 116.13215959379427 + ], + [ + 26.99617045277519, + 113.29523664776464 + ], + [ + 23.79060659288676, + 107.64810947740314 + ], + [ + 24.15390783848862, + 105.51107030087962 + ], + [ + 23.560878575360448, + 104.0739047036557 + ], + [ + 30.37127897002722, + 99.12240179644283 + ], + [ + 36.131354390311685, + 93.07148768966871 + ], + [ + 40.71238214647833, + 86.08394074351675 + ], + [ + 43.98564973899843, + 78.32255969858304 + ], + [ + 45.50294093069587, + 78.58969596452754 + ], + [ + 45.50294093069587, + 78.58969596452754 + ], + [ + 47.49573773283649, + 77.85241413680993 + ], + [ + 53.70384596657826, + 79.6688999844065 + ], + [ + 56.8987202997463, + 80.89769623713153 + ], + [ + 57.817642560058374, + 81.09537605490989 + ], + [ + 57.88709900822471, + 81.09537605490989 + ], + [ + 57.88709900822471, + 81.09537605490989 + ], + [ + 59.7933300053224, + 81.18200300119241 + ], + [ + 61.387088507324904, + 80.34487752923569 + ], + [ + 62.38084764488653, + 78.84379895623448 + ], + [ + 62.487131499694996, + 76.93855640917657 + ], + [ + 62.38022604228418, + 76.54882176766148 + ], + [ + 61.264918120431545, + 74.94530104773905 + ], + [ + 59.47385683829488, + 74.16601518842774 + ], + [ + 58.43739054488442, + 73.92025797592403 + ], + [ + 58.43739054488442, + 73.92025797592403 + ], + [ + 55.01811767224032, + 73.65312170997949 + ], + [ + 48.639058532624574, + 72.60063641179842 + ], + [ + 47.10038828748644, + 71.06196616666014 + ], + [ + 45.657883022005635, + 70.63455833135549 + ], + [ + 45.657883022005635, + 70.63455833135549 + ], + [ + 46.03678547390773, + 62.22105501523648 + ], + [ + 44.89768341965083, + 53.94499720569996 + ], + [ + 42.28747218999483, + 46.00916491892614 + ], + [ + 38.253036925493305, + 38.616338171095194 + ], + [ + 39.51389156817304, + 37.45164850807274 + ], + [ + 39.51389156817304, + 37.45164850807274 + ], + [ + 40.17636708926102, + 35.43214317444094 + ], + [ + 45.46554287248419, + 31.713690931207466 + ], + [ + 48.41465999314843, + 29.988030395011364 + ], + [ + 49.19467954733483, + 29.3576030736714 + ], + [ + 49.19467954733483, + 29.3576030736714 + ], + [ + 50.88472531147386, + 26.20214955472445 + ], + [ + 50.38192013755673, + 24.47356442923497 + ], + [ + 48.99704049038279, + 23.16080596925247 + ], + [ + 48.48411644131477, + 22.935796017380596 + ], + [ + 46.50069462924344, + 22.828946606106054 + ], + [ + 44.76566929318453, + 23.7959564513502 + ], + [ + 43.974960212277935, + 24.42638377269011 + ], + [ + 43.974960212277935, + 24.42638377269011 + ], + [ + 41.64024121797614, + 26.910694999838206 + ], + [ + 36.976153087836124, + 31.307657054240284 + ], + [ + 34.81773485787174, + 31.542729835126885 + ], + [ + 33.46606509399801, + 32.504400012114104 + ], + [ + 33.46606509399801, + 32.504400012114104 + ], + [ + 27.138578589970592, + 26.96979310299842 + ], + [ + 19.97866429922756, + 22.687363875645108 + ], + [ + 12.169613467832335, + 19.74726508781922 + ], + [ + 3.8947581026741886, + 18.239646949734194 + ], + [ + 3.8146121277875693, + 16.652894214767297 + ], + [ + 2.649922464765183, + 14.879151188742481 + ], + [ + 3.055961436835645, + 8.430627958305543 + ], + [ + 3.5474860520496065, + 5.04876078652791 + ], + [ + 3.5474860520496065, + 3.9802411982661803 + ], + [ + 3.5474860520496065, + 3.9802411982661803 + ], + [ + 2.6091922089963617, + 1.288235727806406 + ], + [ + 0.048077394725396516, + 0.02671362659445572 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 754, + "versionNonce": 1850345355, + "isDeleted": false, + "id": "oAVWoM9FTOJ84Xb3pJTBR", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 605.5507359961043, + "y": 513.9438709117968, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 21.43452383045392, + "height": 21.35438295067055, + "seed": 137753963, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + -1.0685195882617524, + 18.66171256923035 + ], + [ + -1.148660468045076, + 18.661712569230446 + ], + [ + -1.148660468045076, + 18.661712569230446 + ], + [ + -2.9224034940699255, + 21.354382950670548 + ], + [ + -6.1279673539584385, + 21.065882916595058 + ], + [ + -21.434523830453912, + 10.22040017930772 + ], + [ + -21.434523830453912, + 10.22040017930772 + ], + [ + -13.28278144629947, + 4.181714320450315 + ], + [ + -3.78790869139976, + 0.60905336300238 + ], + [ + -0.010684431617111318, + 1.7758364223574574e-14 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 752, + "versionNonce": 2028227237, + "isDeleted": false, + "id": "KUqmLYnYb0Oa29MD5rEcD", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 614.703060131676, + "y": 513.5106776267022, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 21.316984892458994, + "height": 21.439863498710842, + "seed": 1682575371, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + 11.501951131586363, + 3.3885443046504036 + ], + [ + 21.316984892459004, + 10.279167100753504 + ], + [ + 6.15468098055286, + 21.055198484977964 + ], + [ + 6.15468098055286, + 21.055198484977964 + ], + [ + 2.8529521410069645, + 21.439863498710867 + ], + [ + 1.5546994401155487, + 20.29654779402594 + ], + [ + 1.0578351566446038, + 18.640338610892833 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 757, + "versionNonce": 777444907, + "isDeleted": false, + "id": "n2spbU_7IbYY3qnznsd6N", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 578.8292850836692, + "y": 530.9116323914815, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 20.189698382751402, + "height": 23.186897993244525, + "seed": 1749145259, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 13.997619343986948, + 12.49100239370014 + ], + [ + 13.997619343987013, + 12.571143273483521 + ], + [ + 13.997619343987013, + 12.571143273483521 + ], + [ + 14.99134281582555, + 15.637799332142835 + ], + [ + 12.763478327901545, + 17.961833894827436 + ], + [ + 12.763478327901568, + 18.015250957809787 + ], + [ + -5.198355566925851, + 23.181553229884337 + ], + [ + -5.198355566925851, + 23.181553229884337 + ], + [ + -4.5279316846921045, + 11.154643415238663 + ], + [ + 0.005339668256897138, + -0.005344763360215503 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 754, + "versionNonce": 766356997, + "isDeleted": false, + "id": "rAI8TWMDdXoqyyRbW0MNf", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 641.3357430114843, + "y": 530.919123645792, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 20.21641200934586, + "height": 23.117436449974957, + "seed": 851953995, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + 4.563918399290026, + 11.121963422681532 + ], + [ + 5.353292563132349, + 23.11743644997496 + ], + [ + -12.635254958289572, + 17.935115173129766 + ], + [ + -12.635254958289533, + 17.865658724963506 + ], + [ + -12.635254958289533, + 17.865658724963506 + ], + [ + -14.863119446213531, + 15.541629257382146 + ], + [ + -13.869390879271627, + 12.474978293826098 + ], + [ + 0.021368863234242145, + 0.04273772646850472 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 754, + "versionNonce": 265281739, + "isDeleted": false, + "id": "4WyJXFXc24eKFSIYXAM6L", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 607.1217167532573, + "y": 544.2146158910938, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 12.784847191135828, + "height": 12.474983388929397, + "seed": 691431403, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 5.727273145248169, + -1.3322676295501878e-15 + ], + [ + 9.226671612365799, + 4.445049639334108 + ], + [ + 7.955137633172139, + 10.001351498295145 + ], + [ + 2.8102093194351676, + 12.474983388929395 + ], + [ + -2.3454034259189, + 10.001351498295142 + ], + [ + -3.5581755787700295, + 4.445049639334092 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 755, + "versionNonce": 2117313893, + "isDeleted": false, + "id": "zcT13IENVU0aP9LkOK3OY", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 625.4682156618142, + "y": 559.3769198029996, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 21.984810272010485, + "height": 21.78714064443878, + "seed": 563622539, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + 0.7212526327404134, + 5.10702591327572e-15 + ], + [ + 19.244052305638295, + 3.1254280752084505 + ], + [ + 19.244052305638295, + 3.1254280752084505 + ], + [ + 16.768260091205423, + 8.668625328485522 + ], + [ + 13.409934844218869, + 13.692427751505338 + ], + [ + 9.262235433389911, + 18.09815338358097 + ], + [ + 4.4183309176363466, + 21.787140644438793 + ], + [ + -2.740757966372229, + 4.455739166054528 + ], + [ + -2.740757966372229, + 4.455739166054528 + ], + [ + -2.5152181237572835, + 1.5630757899384222 + ], + [ + -0.021368863234245236, + 0.08014597488666866 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 753, + "versionNonce": 1817153387, + "isDeleted": false, + "id": "8YhZb3uoUhfEw7FRfjPTy", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 594.623294619108, + "y": 559.8203203036147, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 21.8512472341415, + "height": 21.632198553128937, + "seed": 1286056235, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + 2.5858209701657273, + 1.4318157387603865 + ], + [ + 2.8369229460296426, + 4.370243332704308 + ], + [ + 2.8369229460296426, + 4.439699780870577 + ], + [ + -4.279428211510453, + 21.63219855312897 + ], + [ + -4.279428211510453, + 21.63219855312897 + ], + [ + -13.169517299972037, + 13.615624164382494 + ], + [ + -19.01432428811188, + 3.1628159432134098 + ], + [ + -0.6517961845741633, + 0.04807739472538941 + ], + [ + -0.6517961845741633, + 0.04807739472538941 + ], + [ + -0.032058389954645605, + 0.04807739472538499 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "line", + "version": 757, + "versionNonce": 1841541317, + "isDeleted": false, + "id": "P8oul5rTlQ1zLWpnrP9Rk", + "fillStyle": "cross-hatch", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 610.0591683223424, + "y": 566.9438268892583, + "strokeColor": "#326ce5", + "backgroundColor": "#326ce5", + "width": 23.694451803435832, + "height": 19.92456449240321, + "seed": 1607420875, + "groupIds": [ + "iU6cNm0_6Jc7tHERqDYR_" + ], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": null, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 0, + 0 + ], + [ + 1.6676476900384873, + 0.4162597492873822 + ], + [ + 2.868981335984324, + 1.6455247515160254 + ], + [ + 2.938437784150559, + 1.645524751516041 + ], + [ + 11.988808632178861, + 17.977857994701516 + ], + [ + 8.371866131963033, + 19.0463826780665 + ], + [ + 8.371866131963033, + 19.0463826780665 + ], + [ + -1.737012426345295, + 19.92456449240325 + ], + [ + -11.705643171256973, + 18.00991638465619 + ], + [ + -2.6232139332739735, + 1.6775729512641009 + ], + [ + -2.6232139332739735, + 1.6775729512641009 + ], + [ + 0.04808758493198971, + 0.10150464791430291 + ], + [ + 0, + 0 + ] + ] + }, + { + "type": "rectangle", + "version": 256, + "versionNonce": 972771365, + "isDeleted": false, + "id": "Qa-1IAlwB_PPJKn3zJrHW", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 869, + "y": 484, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 291, + "height": 136, + "seed": 396080715, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "type": "text", + "id": "LacPdwjp42kHYUhWkFWmt" + }, + { + "id": "4v25pr07QkyUeTy48CPzk", + "type": "arrow" + }, + { + "id": "jYF9DOMdn-Nr5elnlLzns", + "type": "arrow" + }, + { + "id": "Nd2uv8W9gE-JZ8Puh2Q8A", + "type": "arrow" + }, + { + "id": "UNoDOVrcISvh3tn4USPJz", + "type": "arrow" + } + ], + "updated": 1691670360808, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 235, + "versionNonce": 1916596395, + "isDeleted": false, + "id": "LacPdwjp42kHYUhWkFWmt", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 940.548828125, + "y": 535.9, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 147.90234375, + "height": 32.199999999999996, + "seed": 1200260037, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 2, + "text": "KubeHound", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "Qa-1IAlwB_PPJKn3zJrHW", + "originalText": "KubeHound", + "lineHeight": 1.15, + "baseline": 26 + }, + { + "type": "image", + "version": 105, + "versionNonce": 7441547, + "isDeleted": false, + "id": "CQVVILbzbnmyPkQWzctxx", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1300, + "y": 481.298, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 393.99999999999994, + "height": 131.202, + "seed": 337742213, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "status": "saved", + "fileId": "3f5e12b6a50c2eca8b5b822029a61788d400f053", + "scale": [ + 1, + 1 + ] + }, + { + "type": "arrow", + "version": 152, + "versionNonce": 390830341, + "isDeleted": false, + "id": "4v25pr07QkyUeTy48CPzk", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 700, + "y": 556, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 165, + "height": 1.9157447768529892, + "seed": 1060694891, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": null, + "endBinding": { + "elementId": "Qa-1IAlwB_PPJKn3zJrHW", + "focus": -0.005000396856893404, + "gap": 4 + }, + "lastCommittedPoint": null, + "startArrowhead": "arrow", + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 165, + -1.9157447768529892 + ] + ] + }, + { + "type": "arrow", + "version": 201, + "versionNonce": 127542731, + "isDeleted": false, + "id": "jYF9DOMdn-Nr5elnlLzns", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1162, + "y": 549.3869814054576, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 157, + "height": 1.1638228724989403, + "seed": 2050281003, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Qa-1IAlwB_PPJKn3zJrHW", + "focus": -0.053655117813421256, + "gap": 2 + }, + "endBinding": { + "elementId": "ZInMjpv74vj_jOD2Pr9i6", + "focus": -0.04032631931594341, + "gap": 4 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 157, + 1.1638228724989403 + ] + ] + }, + { + "type": "ellipse", + "version": 232, + "versionNonce": 1931664325, + "isDeleted": false, + "id": "5LyHW7R44FjlVJOBA9ETT", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1196, + "y": 462, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 68, + "height": 64, + "seed": 6460453, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "iHYx5s9EKjtw0YSwnJUVu" + } + ], + "updated": 1691670360808, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 148, + "versionNonce": 243653571, + "isDeleted": false, + "id": "iHYx5s9EKjtw0YSwnJUVu", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1220.4243728576262, + "y": 476.37258300203047, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 19.0679931640625, + "height": 35, + "seed": 1262663115, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691745546739, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "3", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "5LyHW7R44FjlVJOBA9ETT", + "originalText": "3", + "lineHeight": 1.25, + "baseline": 25 + }, + { + "type": "ellipse", + "version": 192, + "versionNonce": 1108432299, + "isDeleted": false, + "id": "Tn0J6q5dcixAc_2KfTgDH", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 746, + "y": 472, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 68, + "height": 64, + "seed": 1559307365, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "YZPTSM29qN6HxblihfN-D" + } + ], + "updated": 1691670360808, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 107, + "versionNonce": 1781069293, + "isDeleted": false, + "id": "YZPTSM29qN6HxblihfN-D", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 776.1643707213957, + "y": 486.37258300203047, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 7.5879974365234375, + "height": 35, + "seed": 913919941, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691745546740, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "1", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "Tn0J6q5dcixAc_2KfTgDH", + "originalText": "1", + "lineHeight": 1.25, + "baseline": 25 + }, + { + "type": "image", + "version": 189, + "versionNonce": 1146054731, + "isDeleted": false, + "id": "qaXykhX7_G0Ur4gfN8l5U", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1446, + "y": 180, + "strokeColor": "transparent", + "backgroundColor": "transparent", + "width": 149, + "height": 149, + "seed": 821435717, + "groupIds": [], + "roundness": null, + "boundElements": [ + { + "id": "BXpY9BWfInuATqONCSUeP", + "type": "arrow" + } + ], + "updated": 1691670360808, + "link": null, + "locked": false, + "status": "saved", + "fileId": "a63456e84b50bbd6ab2da19ff5ab9115a91a260c", + "scale": [ + 1, + 1 + ] + }, + { + "type": "arrow", + "version": 181, + "versionNonce": 2029147621, + "isDeleted": false, + "id": "BXpY9BWfInuATqONCSUeP", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1524, + "y": 338, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 0, + "height": 155.99999999999994, + "seed": 307424037, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "startBinding": { + "elementId": "qaXykhX7_G0Ur4gfN8l5U", + "gap": 9, + "focus": -0.04697986577181208 + }, + "endBinding": { + "elementId": "ZInMjpv74vj_jOD2Pr9i6", + "gap": 8.000000000000057, + "focus": 0.05511811023622048 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 0, + 155.99999999999994 + ] + ] + }, + { + "type": "ellipse", + "version": 330, + "versionNonce": 108144363, + "isDeleted": false, + "id": "wGfwZnnA3zUk8qU5IRmiV", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1548, + "y": 374, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 68, + "height": 64, + "seed": 1082170059, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "3GPfWW4TUmkXjys_kHqid" + } + ], + "updated": 1691670360808, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 248, + "versionNonce": 1635478371, + "isDeleted": false, + "id": "3GPfWW4TUmkXjys_kHqid", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1572.9983703551848, + "y": 388.37258300203047, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 17.919998168945312, + "height": 35, + "seed": 2061044075, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691745546741, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "4", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "wGfwZnnA3zUk8qU5IRmiV", + "originalText": "4", + "lineHeight": 1.25, + "baseline": 25 + }, + { + "type": "text", + "version": 98, + "versionNonce": 1881661605, + "isDeleted": false, + "id": "GmA2C6XJkzXOAGxfNxonK", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 649.9768829345703, + "y": 374, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 269.212890625, + "height": 64.39999999999999, + "seed": 1648806763, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 2, + "text": "Read resources from \nthe K8s API", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Read resources from \nthe K8s API", + "lineHeight": 1.15, + "baseline": 58 + }, + { + "type": "text", + "version": 196, + "versionNonce": 1866442795, + "isDeleted": false, + "id": "xS0Mb_4zwzIE5yaVRjQDm", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1137.697265625, + "y": 364.79999999999995, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 194.60546875, + "height": 64.39999999999999, + "seed": 1261186117, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691670360808, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 2, + "text": "Ingest into \ngraph database", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Ingest into \ngraph database", + "lineHeight": 1.15, + "baseline": 58 + }, + { + "type": "ellipse", + "version": 359, + "versionNonce": 167288069, + "isDeleted": false, + "id": "Lead40U49aI9gLHoKkzMu", + "fillStyle": "hachure", + "strokeWidth": 1, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 974, + "y": 676, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 68, + "height": 64, + "seed": 2135129259, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [ + { + "type": "text", + "id": "zgjaRuFaTCRCGRM3t6Z5z" + }, + { + "id": "Nd2uv8W9gE-JZ8Puh2Q8A", + "type": "arrow" + }, + { + "id": "UNoDOVrcISvh3tn4USPJz", + "type": "arrow" + } + ], + "updated": 1691670383187, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 275, + "versionNonce": 382578765, + "isDeleted": false, + "id": "zgjaRuFaTCRCGRM3t6Z5z", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 997.990374749716, + "y": 690.3725830020305, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 19.935989379882812, + "height": 35, + "seed": 111456075, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691745546743, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 1, + "text": "2", + "textAlign": "center", + "verticalAlign": "middle", + "containerId": "Lead40U49aI9gLHoKkzMu", + "originalText": "2", + "lineHeight": 1.25, + "baseline": 25 + }, + { + "type": "arrow", + "version": 154, + "versionNonce": 1213767461, + "isDeleted": false, + "id": "Nd2uv8W9gE-JZ8Puh2Q8A", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 902, + "y": 626, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 73.30030377912567, + "height": 70.23391850877931, + "seed": 1732962981, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691670384089, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Qa-1IAlwB_PPJKn3zJrHW", + "focus": 0.7864197530864196, + "gap": 6 + }, + "endBinding": { + "elementId": "Lead40U49aI9gLHoKkzMu", + "focus": 0.27172350358387015, + "gap": 1 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": null, + "points": [ + [ + 0, + 0 + ], + [ + 6, + 64 + ], + [ + 73.30030377912567, + 70.23391850877931 + ] + ] + }, + { + "type": "arrow", + "version": 98, + "versionNonce": 2120225195, + "isDeleted": false, + "id": "UNoDOVrcISvh3tn4USPJz", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1042.9065780270603, + "y": 695.760342212372, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 78.09342197293972, + "height": 71.76034221237205, + "seed": 1948650469, + "groupIds": [], + "roundness": { + "type": 2 + }, + "boundElements": [], + "updated": 1691670384089, + "link": null, + "locked": false, + "startBinding": { + "elementId": "Lead40U49aI9gLHoKkzMu", + "focus": -0.2876411508604094, + "gap": 3.2259661543258815 + }, + "endBinding": { + "elementId": "Qa-1IAlwB_PPJKn3zJrHW", + "focus": -0.7555791710945803, + "gap": 4 + }, + "lastCommittedPoint": null, + "startArrowhead": null, + "endArrowhead": "arrow", + "points": [ + [ + 0, + 0 + ], + [ + 67.09342197293972, + -5.760342212372052 + ], + [ + 78.09342197293972, + -71.76034221237205 + ] + ] + }, + { + "type": "text", + "version": 261, + "versionNonce": 645541195, + "isDeleted": false, + "id": "rBCScTBc2va9bhZSefvGL", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 869.814453125, + "y": 754.8, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 272.37109375, + "height": 32.199999999999996, + "seed": 1663532325, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691670360809, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 2, + "text": "Compute attack paths", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Compute attack paths", + "lineHeight": 1.15, + "baseline": 26 + }, + { + "type": "rectangle", + "version": 87, + "versionNonce": 931291365, + "isDeleted": false, + "id": "ZInMjpv74vj_jOD2Pr9i6", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1323, + "y": 502, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 381, + "height": 96, + "seed": 290761579, + "groupIds": [], + "roundness": { + "type": 3 + }, + "boundElements": [ + { + "id": "jYF9DOMdn-Nr5elnlLzns", + "type": "arrow" + }, + { + "id": "BXpY9BWfInuATqONCSUeP", + "type": "arrow" + } + ], + "updated": 1691670360809, + "link": null, + "locked": false + }, + { + "type": "text", + "version": 272, + "versionNonce": 1090276331, + "isDeleted": false, + "id": "Ub4UGl4aMzQLxlEMzXEJg", + "fillStyle": "hachure", + "strokeWidth": 2, + "strokeStyle": "solid", + "roughness": 1, + "opacity": 100, + "angle": 0, + "x": 1622.060546875, + "y": 390.79999999999995, + "strokeColor": "#1e1e1e", + "backgroundColor": "transparent", + "width": 243.87890625, + "height": 32.199999999999996, + "seed": 667055051, + "groupIds": [], + "roundness": null, + "boundElements": [], + "updated": 1691670360809, + "link": null, + "locked": false, + "fontSize": 28, + "fontFamily": 2, + "text": "Visualize and query", + "textAlign": "center", + "verticalAlign": "top", + "containerId": null, + "originalText": "Visualize and query", + "lineHeight": 1.15, + "baseline": 26 + } + ], + "appState": { + "gridSize": null, + "viewBackgroundColor": "#ffffff" + }, + "files": { + "3f5e12b6a50c2eca8b5b822029a61788d400f053": { + "mimeType": "image/png", + "id": "3f5e12b6a50c2eca8b5b822029a61788d400f053", + "dataURL": "", + "created": 1691669750698, + "lastRetrieved": 1691669750698 + }, + "a63456e84b50bbd6ab2da19ff5ab9115a91a260c": { + "mimeType": "image/png", + "id": "a63456e84b50bbd6ab2da19ff5ab9115a91a260c", + "dataURL": "", + "created": 1691669984446, + "lastRetrieved": 1691669984446 + } + } +} \ No newline at end of file diff --git a/docs/images/kubehound-high-level.png b/docs/images/kubehound-high-level.png new file mode 100644 index 000000000..e215527df Binary files /dev/null and b/docs/images/kubehound-high-level.png differ diff --git a/docs/index.md b/docs/index.md new file mode 100644 index 000000000..c79109dda --- /dev/null +++ b/docs/index.md @@ -0,0 +1,21 @@ +# Home + +Welcome to the KubeHound documentation! + +KubeHound creates a graph of attack paths in a Kubernetes cluster, allowing you to identify direct and multi-hop routes an attacker is able to take, visually or through complex graph queries. + +[![](./images/example-graph.png)](./images/example-graph.png) +
*A KubeHound graph showing attack paths between pods, nodes, and identities (click to enlarge)*
+ +After the data is ingested in a graph database, you can ask advanced questions such as: + +- What are all possible container escapes in the cluster? +- What is the shortest exploitable path between an exposed service and a critical asset? +- Is there an attack path from a specific container to a node in the cluster? + +KubeHound was built with efficiency in mind and can consequently handle very large clusters. Ingestion and computation of attack paths typically takes 1 minute for a cluster with 1'000 running pods, 15 minutes for 10'000 pods, and 25 minutes for 25'000 pods. + +Next steps: + +- Learn more about KubeHound [architecture](./architecture.md) and [terminology](./terminology.md) +- [Get started](./user-guide/getting-started.md) using KubeHound \ No newline at end of file diff --git a/docs/js/tablesort.js b/docs/js/tablesort.js new file mode 100644 index 000000000..281c7d892 --- /dev/null +++ b/docs/js/tablesort.js @@ -0,0 +1,12 @@ +/*! + * tablesort v5.2.1 (2020-06-02) + * http://tristen.ca/tablesort/demo/ + * Copyright (c) 2020 ; Licensed MIT +*/ +(function() {function a(b,c){if(!(this instanceof a))return new a(b,c);if(!b||"TABLE"!==b.tagName)throw new Error("Element must be a table");this.init(b,c||{})}var b=[],c=function(a){var b;return window.CustomEvent&&"function"==typeof window.CustomEvent?b=new CustomEvent(a):(b=document.createEvent("CustomEvent"),b.initCustomEvent(a,!1,!1,void 0)),b},d=function(a){return a.getAttribute("data-sort")||a.textContent||a.innerText||""},e=function(a,b){return a=a.trim().toLowerCase(),b=b.trim().toLowerCase(),a===b?0:a0)if(a.tHead&&a.tHead.rows.length>0){for(e=0;e0&&n.push(m),o++;if(!n)return}for(o=0;o + +# CE_MODULE_LOAD | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Load a kernel module from within an overprivileged container to breakout into the node. ## Details @@ -14,7 +25,7 @@ Container isolation mechanisms are restricted to user-space execution. If an att Execution within a container process with the `CAP_SYS_MODULE` capability enabled. -See the [example pod spec](../../test/setup/test-cluster/attacks/CE_MODULE_LOAD.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/CE_MODULE_LOAD.yaml). ## Checks @@ -55,7 +66,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [EscapeModuleLoad](../../pkg/kubehound/graph/edge/escape_module_load.go) ++ [EscapeModuleLoad](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/escape_module_load.go) ## References: diff --git a/docs/edges/CE_NSENTER.md b/docs/reference/attacks/CE_NSENTER.md similarity index 83% rename from docs/edges/CE_NSENTER.md rename to docs/reference/attacks/CE_NSENTER.md index 8e6445c50..d580a5e6b 100644 --- a/docs/edges/CE_NSENTER.md +++ b/docs/reference/attacks/CE_NSENTER.md @@ -1,10 +1,21 @@ -# CE_NSENTER +--- +title: CE_NSENTER +--- -Container escape via the nsenter built-in linux program that allows executing a binary into another namespace. + + +# CE_NSENTER | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Container escape via the nsenter built-in linux program that allows executing a binary into another namespace. ## Details @@ -14,7 +25,7 @@ When both `hostPID: true` and `privileged: true` are set, the pod can see all of Execution within a container process created with `--privileged` AND the `--pid=host` enabled. -See the [example pod spec](../../test/setup/test-cluster/attacks/CE_NSENTER.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/CE_NSENTER.yaml). ## Checks @@ -68,7 +79,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [EscapeNsenter](../../pkg/kubehound/graph/edge/escape_nsenter.go) ++ [EscapeNsenter](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/escape_nsenter.go) ## References: diff --git a/docs/edges/CE_PRIV_MOUNT.md b/docs/reference/attacks/CE_PRIV_MOUNT.md similarity index 80% rename from docs/edges/CE_PRIV_MOUNT.md rename to docs/reference/attacks/CE_PRIV_MOUNT.md index 1768207e4..5fee3b6c8 100644 --- a/docs/edges/CE_PRIV_MOUNT.md +++ b/docs/reference/attacks/CE_PRIV_MOUNT.md @@ -1,10 +1,21 @@ -# CE_PRIV_MOUNT +--- +title: CE_PRIV_MOUNT +--- -Mount the host disk and gain access to the host via arbitrary filesystem write + + +# CE_PRIV_MOUNT | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Mount the host disk and gain access to the host via arbitrary filesystem write ## Details @@ -14,7 +25,7 @@ A container running with `privileged: true` disables almost all the container is Execution within a privileged container process. -See the [example pod spec](../../test/setup/test-cluster/attacks/CE_PRIV_MOUNT.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/CE_PRIV_MOUNT.yaml). ## Checks @@ -67,7 +78,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [EscapePrivMount](../../pkg/kubehound/graph/edge/escape_priv_mount.go) ++ [EscapePrivMount](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/escape_priv_mount.go) ## References: diff --git a/docs/edges/CE_SYS_PTRACE.md b/docs/reference/attacks/CE_SYS_PTRACE.md similarity index 80% rename from docs/edges/CE_SYS_PTRACE.md rename to docs/reference/attacks/CE_SYS_PTRACE.md index 20d6de008..93bf8668c 100644 --- a/docs/edges/CE_SYS_PTRACE.md +++ b/docs/reference/attacks/CE_SYS_PTRACE.md @@ -1,10 +1,21 @@ -# CE_SYS_PTRACE +--- +title: CE_SYS_PTRACE +--- -Given the requisite capabilities, abuse the legitimate OS debugging mechanisms to escape the container via attaching to a node process. + + +# CE_SYS_PTRACE | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Given the requisite capabilities, abuse the legitimate OS debugging mechanisms to escape the container via attaching to a node process. ## Details @@ -14,7 +25,7 @@ The `SYS_PTRACE` capability, which allows the use of `ptrace()`. This system cal To perform this attack, the container must be started with the option `--pid=host`, which enables the sharing of the PID address space between the container and the host operating system, allowing the container process to see every other process running on the host. And the container needs to be granted `SYS_PTRACE` and `SYS_ADMIN` capabilities. -See the [example pod spec](../../test/setup/test-cluster/attacks/CE_SYS_PTRACE.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/CE_SYS_PTRACE.yaml). ## Checks @@ -68,7 +79,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [EscapeSysPtrace](../../pkg/kubehound/graph/edge/escape_sys_ptrace.go) ++ [EscapeSysPtrace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/escape_sys_ptrace.go) ## References: diff --git a/docs/edges/CE_UMH_CORE_PATTERN.md b/docs/reference/attacks/CE_UMH_CORE_PATTERN.md similarity index 87% rename from docs/edges/CE_UMH_CORE_PATTERN.md rename to docs/reference/attacks/CE_UMH_CORE_PATTERN.md index 3735d74ad..9a6b2e921 100644 --- a/docs/edges/CE_UMH_CORE_PATTERN.md +++ b/docs/reference/attacks/CE_UMH_CORE_PATTERN.md @@ -1,10 +1,17 @@ + + # CE_UMH_CORE_PATTERN Container escape via the `core_pattern` `usermode_helper` in the case of an exposed `/proc` mount. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | ## Details @@ -14,7 +21,7 @@ Container escape via the `core_pattern` `usermode_helper` in the case of an expo Execution within a container process with the host `/proc/sys/kernel` (or any parent directory) mounted inside the container. -See the [example pod spec](../../test/setup/test-cluster/attacks/CE_UMH_CORE_PATTERN.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/CE_UMH_CORE_PATTERN.yaml). ## Checks @@ -94,7 +101,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [EscapeUmhCorePattern](../../pkg/kubehound/graph/edge/escape_umh_core_pattern.go) ++ [EscapeUmhCorePattern](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/escape_umh_core_pattern.go) ## References: diff --git a/docs/edges/CONTAINER_ATTACH.md b/docs/reference/attacks/CONTAINER_ATTACH.md similarity index 84% rename from docs/edges/CONTAINER_ATTACH.md rename to docs/reference/attacks/CONTAINER_ATTACH.md index 3bd4c884a..e9436873d 100644 --- a/docs/edges/CONTAINER_ATTACH.md +++ b/docs/reference/attacks/CONTAINER_ATTACH.md @@ -1,10 +1,21 @@ -# CONTAINER_ATTACH +--- +title: CONTAINER_ATTACH +--- -Attach to a container running within a pod given access to the pod. + + +# CONTAINER_ATTACH | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Pod](../vertices/POD.md) | [Container](../vertices/CONTAINER.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | +| [Pod](../entities/pod.md) | [Container](../entities/container.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | + +Attach to a container running within a pod given access to the pod. ## Details @@ -54,7 +65,7 @@ kubectl describe pod ## Calculation -+ [ContainerAttach](../../pkg/kubehound/graph/edge/container_attach.go) ++ [ContainerAttach](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/container_attach.go) ## References: diff --git a/docs/edges/ENDPOINT_EXPLOIT.md b/docs/reference/attacks/ENDPOINT_EXPLOIT.md similarity index 67% rename from docs/edges/ENDPOINT_EXPLOIT.md rename to docs/reference/attacks/ENDPOINT_EXPLOIT.md index 7b2b7021a..9a05e3706 100644 --- a/docs/edges/ENDPOINT_EXPLOIT.md +++ b/docs/reference/attacks/ENDPOINT_EXPLOIT.md @@ -1,10 +1,20 @@ +--- +title: ENDPOINT_EXPLOIT +--- + + # ENDPOINT_EXPLOIT Represents a network endpoint exposed by a container that could be exploited by an attacker (via means known or unknown). This can correspond to a Kubernetes service, node service, node port, or container port. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Endpoint](../vertices/ENDPOINT.md) | [Container](../vertices/CONTAINER.md) | [Network Service Discovery, T1046](https://attack.mitre.org/techniques/T1046/) | +| [Endpoint](../entities/endpoint.md) | [Container](../entities/container.md) | [Exploitation of Remote Services, T1210](https://attack.mitre.org/techniques/T1210/) | ## Details @@ -34,8 +44,8 @@ None ## Calculation -+ [EndpointExploitInternal](../../pkg/kubehound/graph/edge/endpoint_exploit_internal.go) -+ [EndpointExploitExternal](../../pkg/kubehound/graph/edge/endpoint_exploit_external.go) ++ [EndpointExploitInternal](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/endpoint_exploit_internal.go) ++ [EndpointExploitExternal](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/endpoint_exploit_external.go) ## References: diff --git a/docs/edges/EXPLOIT_CONTAINERD_SOCK.md b/docs/reference/attacks/EXPLOIT_CONTAINERD_SOCK.md similarity index 86% rename from docs/edges/EXPLOIT_CONTAINERD_SOCK.md rename to docs/reference/attacks/EXPLOIT_CONTAINERD_SOCK.md index 87896e4d6..c3894e2ec 100644 --- a/docs/edges/EXPLOIT_CONTAINERD_SOCK.md +++ b/docs/reference/attacks/EXPLOIT_CONTAINERD_SOCK.md @@ -1,10 +1,21 @@ -# EXPLOIT_CONTAINERD_SOCK +--- +title: EXPLOIT_CONTAINERD_SOCK +--- -Container escape via the `containerd.sock` file that allows executing a binary into another container. + + +# EXPLOIT_CONTAINERD_SOCK | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Container](../vertices/CONTAINER.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | +| [Container](../entities/container.md) | [Container](../entities/container.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | + +Container escape via the `containerd.sock` file that allows executing a binary into another container. ## Details @@ -31,7 +42,7 @@ docker run -v /var/run/docker.sock:/var/run/docker.sock:ro --rm -it alpine sh docker ps ``` -See the [example pod spec](../../test/setup/test-cluster/attacks/EXPLOIT_CONTAINERD_SOCK.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/EXPLOIT_CONTAINERD_SOCK.yaml). ## Checks @@ -110,7 +121,7 @@ Use a pod security policy or admission controller to prevent or limit the creati ## Calculation -+ [ExploitContainerdSock](../../pkg/kubehound/graph/edge/exploit_containerd_sock.go) ++ [ExploitContainerdSock](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/exploit_containerd_sock.go) ## References: diff --git a/docs/edges/EXPLOIT_HOST_READ.md b/docs/reference/attacks/EXPLOIT_HOST_READ.md similarity index 83% rename from docs/edges/EXPLOIT_HOST_READ.md rename to docs/reference/attacks/EXPLOIT_HOST_READ.md index 51db835be..24d09b477 100644 --- a/docs/edges/EXPLOIT_HOST_READ.md +++ b/docs/reference/attacks/EXPLOIT_HOST_READ.md @@ -1,10 +1,21 @@ -# EXPLOIT_HOST_READ +--- +title: EXPLOIT_HOST_READ +--- -Exploit an arbitrary read from a sensitive host path mounted into the container to gain execution on the host. + + +# EXPLOIT_HOST_READ | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Exploit an arbitrary read from a sensitive host path mounted into the container to gain execution on the host. ## Details @@ -14,7 +25,7 @@ If a sensitive host directory is exposed to a container, even as readonly, it ca Execution within a container with a sensitive host path mounted as readonly. -See the [example pod spec](../../test/setup/test-cluster/attacks/EXPLOIT_HOST_READ_TRAVERSE.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/EXPLOIT_HOST_READ_TRAVERSE.yaml). ## Checks @@ -65,7 +76,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [ExploitHostRead](../../pkg/kubehound/graph/edge/exploit_host_read.go) ++ [ExploitHostRead](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/exploit_host_read.go) ## References: diff --git a/docs/edges/EXPLOIT_HOST_TRAVERSE.md b/docs/reference/attacks/EXPLOIT_HOST_TRAVERSE.md similarity index 92% rename from docs/edges/EXPLOIT_HOST_TRAVERSE.md rename to docs/reference/attacks/EXPLOIT_HOST_TRAVERSE.md index 4519e6377..b29d19ba9 100644 --- a/docs/edges/EXPLOIT_HOST_TRAVERSE.md +++ b/docs/reference/attacks/EXPLOIT_HOST_TRAVERSE.md @@ -1,10 +1,21 @@ -# EXPLOIT_HOST_TRAVERSE +--- +title: EXPLOIT_HOST_TRAVERSE +--- -This attack represents the ability to steal a K8s API token from a container via access to a mounted parent volume of the `/var/lib/kubelet/pods` directory. + + +# EXPLOIT_HOST_TRAVERSE | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Volume](../vertices/VOLUME.md) | [Volume](../vertices/VOLUME.md) | [Unsecured Credentials, T1552](https://attack.mitre.org/techniques/T1552/) | +| [Volume](../entities/volume.md) | [Volume](../entities/volume.md) | [Unsecured Credentials, T1552](https://attack.mitre.org/techniques/T1552/) | + +This attack represents the ability to steal a K8s API token from a container via access to a mounted parent volume of the `/var/lib/kubelet/pods` directory. ## Details @@ -84,7 +95,7 @@ automountServiceAccountToken: false ## Calculation -+ [ExploitHostTraverseToken](../../pkg/kubehound/graph/edge/exploit_host_traverse_token.go) ++ [ExploitHostTraverseToken](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/exploit_host_traverse_token.go) ## References: diff --git a/docs/edges/EXPLOIT_HOST_WRITE.md b/docs/reference/attacks/EXPLOIT_HOST_WRITE.md similarity index 72% rename from docs/edges/EXPLOIT_HOST_WRITE.md rename to docs/reference/attacks/EXPLOIT_HOST_WRITE.md index e5de2833c..fa4cbff04 100644 --- a/docs/edges/EXPLOIT_HOST_WRITE.md +++ b/docs/reference/attacks/EXPLOIT_HOST_WRITE.md @@ -1,20 +1,31 @@ -# EXPLOIT_HOST_WRITE +--- +title: EXPLOIT_HOST_WRITE +--- -Exploit an arbitrary write from a sensitive host path mounted into the container to gain execution on the host. + + +# EXPLOIT_HOST_WRITE | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Exploit an arbitrary write from a sensitive host path mounted into the container to gain execution on the host. ## Details -If a sensitive host directory is mounted in a container with write permissions there are a huge variety of techniques to achieve execution within a container. Given the array of techniques available we choose to assume that any writeable mount in a container is exploitable unless it corresponds to an entry in the ["known-good" list](../../pkg/kubehound/graph/edge/exploit_host_write.go#:~:text=SafeWriteMountList) of mounts. For illustration purposes we will consider an escape to host via creating a cron job to launch a reverse shell as the host's superuser if the host `/etc` directory is mounted with write permissions. +If a sensitive host directory is mounted in a container with write permissions there are a huge variety of techniques to achieve execution within a container. Given the array of techniques available we choose to assume that any writeable mount in a container is exploitable unless it corresponds to an entry in the ["known-good" list](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/exploit_host_write.go#:~:text=SafeWriteMountList) of mounts. For illustration purposes we will consider an escape to host via creating a cron job to launch a reverse shell as the host's superuser if the host `/etc` directory is mounted with write permissions. ## Prerequisites Execution within a container with a sensitive host directory mounted with write permissions. -See the [example pod spec](../../test/setup/test-cluster/attacks/EXPLOIT_HOST_WRITE.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/EXPLOIT_HOST_WRITE.yaml). ## Checks @@ -55,7 +66,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [ExploitHostWrite](../../pkg/kubehound/graph/edge/exploit_host_write.go) ++ [ExploitHostWrite](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/exploit_host_write.go) ## References: diff --git a/docs/edges/IDENTITY_ASSUME.md b/docs/reference/attacks/IDENTITY_ASSUME.md similarity index 82% rename from docs/edges/IDENTITY_ASSUME.md rename to docs/reference/attacks/IDENTITY_ASSUME.md index 59827d578..0e84037df 100644 --- a/docs/edges/IDENTITY_ASSUME.md +++ b/docs/reference/attacks/IDENTITY_ASSUME.md @@ -1,10 +1,21 @@ -# IDENTITY_ASSUME +--- +title: IDENTITY_ASSUME +--- + + -Represents the capacity to act as an [Identity](../vertices/IDENTITY.md) via ownership of a service account token, user PKI certificate, etc. +# IDENTITY_ASSUME | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md), [Node](../vertices/NODE.md) | [Identity](../vertices/IDENTITY.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | +| [Container](../entities/container.md), [Node](../entities/node.md) | [Identity](../entities/identity.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | + +Represents the capacity to act as an [Identity](../entities/identity.md) via ownership of a service account token, user PKI certificate, etc. ## Details @@ -81,8 +92,8 @@ Use a pod security policy or admission controller to prevent or limit the identi ## Calculation -+ [IdentityAssumeContainer](../../pkg/kubehound/graph/edge/identity_assume_container.go) -+ [IdentityAssumeNode](../../pkg/kubehound/graph/edge/identity_assume_node.go) ++ [IdentityAssumeContainer](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/identity_assume_container.go) ++ [IdentityAssumeNode](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/identity_assume_node.go) ## References: diff --git a/docs/edges/IDENTITY_IMPERSONATE.md b/docs/reference/attacks/IDENTITY_IMPERSONATE.md similarity index 73% rename from docs/edges/IDENTITY_IMPERSONATE.md rename to docs/reference/attacks/IDENTITY_IMPERSONATE.md index 2c44777b2..d7934df1e 100644 --- a/docs/edges/IDENTITY_IMPERSONATE.md +++ b/docs/reference/attacks/IDENTITY_IMPERSONATE.md @@ -1,10 +1,17 @@ + + # IDENTITY_IMPERSONATE With a [user impersonation privilege](https://kubernetes.io/docs/reference/access-authn-authz/authentication/#user-impersonation) an attacker can impersonate a more privileged account. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [Identity](../vertices/IDENTITY.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | +| [PermissionSet](../entities/permissionset.md) | [Identity](../entities/identity.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | ## Details @@ -14,7 +21,7 @@ Obtaining the `impersonate users/groups` permission will allow an attacker to ex Ability to interrogate the K8s API with a role allowing impersonate access to users and/or groups. -See the [example pod spec](../../test/setup/test-cluster/attacks/IDENTITY_IMPERSONATE.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/IDENTITY_IMPERSONATE.yaml). ## Checks @@ -45,8 +52,8 @@ Impersonating users is a very powerful privilege and should not be required by t ## Calculation -+ [IdentityImpersonate](../../pkg/kubehound/graph/edge/identity_impersonate.go) -+ [IdentityImpersonateNamespace](../../pkg/kubehound/graph/edge/identity_impersonate_namespace.go) ++ [IdentityImpersonate](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/identity_impersonate.go) ++ [IdentityImpersonateNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/identity_impersonate_namespace.go) ## References: diff --git a/docs/edges/PERMISSION_DISCOVER.md b/docs/reference/attacks/PERMISSION_DISCOVER.md similarity index 74% rename from docs/edges/PERMISSION_DISCOVER.md rename to docs/reference/attacks/PERMISSION_DISCOVER.md index 1541dad42..95c71a760 100644 --- a/docs/edges/PERMISSION_DISCOVER.md +++ b/docs/reference/attacks/PERMISSION_DISCOVER.md @@ -1,10 +1,21 @@ -# PERMISSION_GRANT +--- +title: PERMISSION_DISCOVER +--- + + + +# PERMISSION_DISCOVER Represents the permissions granted to an identity that can be discovered by an attacker. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Identity](../vertices/IDENTITY.md) | [PermissionSet](../vertices/PERMISSIONSET.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | +| [Identity](../entities/identity.md) | [PermissionSet](../entities/permissionset.md) | [Permission Groups Discovery, T1069](https://attack.mitre.org/techniques/T1078/) | ## Details @@ -38,7 +49,7 @@ None ## Calculation -+ [PermissionDiscover](../../pkg/kubehound/graph/edge/permission_discover.go) ++ [PermissionDiscover](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/permission_discover.go) ## References: diff --git a/docs/edges/POD_ATTACH.md b/docs/reference/attacks/POD_ATTACH.md similarity index 96% rename from docs/edges/POD_ATTACH.md rename to docs/reference/attacks/POD_ATTACH.md index 72f5351cf..a7f25b7cf 100644 --- a/docs/edges/POD_ATTACH.md +++ b/docs/reference/attacks/POD_ATTACH.md @@ -1,10 +1,21 @@ -# POD_ATTACH +--- +title: POD_ATTACH +--- -Attach to a running K8s pod from a K8s node. + + +# POD_ATTACH | Source | Destination | MITRE | | --------------------------- | ------------------------------------- |----------------------------------| -| [Node](../vertices/NODE.md) | [Pod](../vertices/POD.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | +| [Node](../entities/node.md) | [Pod](../entities/pod.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | + +Attach to a running K8s pod from a K8s node. ## Details @@ -128,7 +139,7 @@ ctr -n k8s.io task exec -t --exec-id full-control 0f36d12d60d12d041df8941 ## Calculation -+ [PodAttach](../../pkg/kubehound/graph/edge/pod_attach.go) ++ [PodAttach](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/pod_attach.go) ## References: diff --git a/docs/edges/POD_CREATE.md b/docs/reference/attacks/POD_CREATE.md similarity index 85% rename from docs/edges/POD_CREATE.md rename to docs/reference/attacks/POD_CREATE.md index 12dea009b..5087f325e 100644 --- a/docs/edges/POD_CREATE.md +++ b/docs/reference/attacks/POD_CREATE.md @@ -1,10 +1,21 @@ +--- +title: POD_CREATE +--- + + + # POD_CREATE Create a pod with significant privilege (`CAP_SYSADMIN`, `hostPath=/`, etc) and schedule on a target node via setting the `nodeName` selector. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [Node](../vertices/NODE.md) | [Container Orchestration Job, T1053.007](https://attack.mitre.org/techniques/T1053/007/) | +| [PermissionSet](../entities/permissionset.md) | [Node](../entities/node.md) | [Container Orchestration Job, T1053.007](https://attack.mitre.org/techniques/T1053/007/) | ## Details @@ -79,7 +90,7 @@ Use a pod security policy or admission controller to prevent or limit the creati ## Calculation -+ [PodCreate](../../pkg/kubehound/graph/edge/pod_create.go) ++ [PodCreate](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/pod_create.go) ## References: diff --git a/docs/edges/POD_EXEC.md b/docs/reference/attacks/POD_EXEC.md similarity index 69% rename from docs/edges/POD_EXEC.md rename to docs/reference/attacks/POD_EXEC.md index 8de23e206..c7c53694c 100644 --- a/docs/edges/POD_EXEC.md +++ b/docs/reference/attacks/POD_EXEC.md @@ -1,10 +1,21 @@ +--- +title: POD_EXEC +--- + + + # POD_EXEC With the correct privileges an attacker can use the Kubernetes API to obtain a shell on a running pod. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [Pod](../vertices/POD.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | +| [PermissionSet](../entities/permissionset.md) | [Pod](../entities/pod.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | ## Details @@ -14,7 +25,7 @@ An attacker with sufficient permissions can execute arbitrary commands inside th Ability to interrogate the K8s API with a role allowing exec access to pods. -See the [example pod spec](../../test/setup/test-cluster/attacks/POD_EXEC.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/POD_EXEC.yaml). ## Checks @@ -45,8 +56,8 @@ Pod interactive execution is a very powerful privilege and should not be require ## Calculation -+ [PodExec](../../pkg/kubehound/graph/edge/pod_exec.go) -+ [PodExecNamespace](../../pkg/kubehound/graph/edge/pod_exec_namespace.go) ++ [PodExec](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/pod_exec.go) ++ [PodExecNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/pod_exec_namespace.go) ## References: diff --git a/docs/edges/POD_PATCH.md b/docs/reference/attacks/POD_PATCH.md similarity index 83% rename from docs/edges/POD_PATCH.md rename to docs/reference/attacks/POD_PATCH.md index 86505f5c4..3418e37bb 100644 --- a/docs/edges/POD_PATCH.md +++ b/docs/reference/attacks/POD_PATCH.md @@ -1,10 +1,21 @@ +--- +title: POD_PATCH +--- + + + # POD_PATCH With the correct privileges an attacker can use the Kubernetes API to modify certain properties of an existing pod and achieve code execution within the pod | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [Pod](../vertices/POD.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | +| [PermissionSet](../entities/permissionset.md) | [Pod](../entities/pod.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | ## Details @@ -21,7 +32,7 @@ However, this is still just enough to allow an attacker to achieve execution in Ability to interrogate the K8s API with a role allowing pod patch access. -See the [example pod spec](../../test/setup/test-cluster/attacks/POD_PATCH.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/POD_PATCH.yaml). ## Checks @@ -64,8 +75,8 @@ Pod patch is a very powerful privilege and should not be required by the majorit ## Calculation -+ [PodPatch](../../pkg/kubehound/graph/edge/pod_patch.go) -+ [PodPatchNamespace](../../pkg/kubehound/graph/edge/pod_patch_namespace.go) ++ [PodPatch](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/pod_patch.go) ++ [PodPatchNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/pod_patch_namespace.go) ## References: diff --git a/docs/edges/ROLE_BIND.md b/docs/reference/attacks/ROLE_BIND.md similarity index 78% rename from docs/edges/ROLE_BIND.md rename to docs/reference/attacks/ROLE_BIND.md index 113c1af86..deb78a7c0 100644 --- a/docs/edges/ROLE_BIND.md +++ b/docs/reference/attacks/ROLE_BIND.md @@ -1,10 +1,22 @@ +--- +title: ROLE_BIND +--- + + + + # ROLE_BIND A role that grants permission to create or modify `(Cluster)RoleBindings` can allow an attacker to escalate privileges on a compromised user. | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [PermissionSet](../vertices/PERMISSIONSET.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | +| [PermissionSet](../entities/permissionset.md) | [PermissionSet](../entities/permissionset.md) | [Valid Accounts, T1078](https://attack.mitre.org/techniques/T1078/) | ## Details @@ -14,7 +26,7 @@ An attacker with sufficient permission can create a `RoleBinding` with the defau Ability to interact with the K8s API with a role allowing modify or create access to `(Cluster)RoleBindings`. -See the [example pod spec](../../test/setup/test-cluster/attacks/ROLE_BIND.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/ROLE_BIND.yaml). ## Checks @@ -63,8 +75,8 @@ Creating `(Cluster)RoleBinding` is a very powerful privilege and should not be r ## Calculation -+ [RoleBind](../../pkg/kubehound/graph/edge/role_bind.go) -+ [RoleBindNamespace](../../pkg/kubehound/graph/edge/role_bind_namespace.go) ++ [RoleBind](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/role_bind.go) ++ [RoleBindNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/role_bind_namespace.go) ## References: diff --git a/docs/edges/SHARED_PS_NAMESPACE.md b/docs/reference/attacks/SHARED_PS_NAMESPACE.md similarity index 80% rename from docs/edges/SHARED_PS_NAMESPACE.md rename to docs/reference/attacks/SHARED_PS_NAMESPACE.md index 6d758d816..dd9e763a2 100644 --- a/docs/edges/SHARED_PS_NAMESPACE.md +++ b/docs/reference/attacks/SHARED_PS_NAMESPACE.md @@ -1,11 +1,22 @@ -# SHARED_PS_NAMESPACE - -Represents a relationship between containers within the same pod that share a process namespace. +--- +title: SHARED_PS_NAMESPACE +--- + + +# SHARED_PS_NAMESPACE | Source | Destination | MITRE | | --------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Container](../vertices/CONTAINER.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | +| [Container](../entities/container.md) | [Container](../entities/container.md) | [Lateral Movement, TA0008](https://attack.mitre.org/tactics/TA0008/) | + +Represents a relationship between containers within the same pod that share a process namespace. ## Details @@ -15,7 +26,7 @@ Pods represent one or more containers with shared storage and network resources. Access to a container within a pod running other containers with shared process nanespaces -See the [example pod spec](../../test/setup/test-cluster/attacks/SHARED_PS_NAMESPACE.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/SHARED_PS_NAMESPACE.yaml). ## Checks @@ -76,7 +87,7 @@ Prevent the use of shared namespaces in pods, where containers have different ri ## Calculation -+ [SharedPsNamespace](../../pkg/kubehound/graph/edge/shared_ps_namespace.go) ++ [SharedPsNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/shared_ps_namespace.go) ## References: diff --git a/docs/edges/TOKEN_BRUTEFORCE.md b/docs/reference/attacks/TOKEN_BRUTEFORCE.md similarity index 75% rename from docs/edges/TOKEN_BRUTEFORCE.md rename to docs/reference/attacks/TOKEN_BRUTEFORCE.md index 850ee9f3c..c2ddb6549 100644 --- a/docs/edges/TOKEN_BRUTEFORCE.md +++ b/docs/reference/attacks/TOKEN_BRUTEFORCE.md @@ -1,10 +1,21 @@ -# TOKEN_BRUTEFORCE +--- +title: TOKEN_BRUTEFORCE +--- -An identity with a role that allows *get* on secrets (vs list) can potentially view all the serviceaccount tokens in a specific namespace or in the whole cluster (with ClusterRole). + + +# TOKEN_BRUTEFORCE | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [Identity](../vertices/IDENTITY.md) | [Steal Application Access Token, T1528](https://attack.mitre.org/techniques/T1528/) | +| [PermissionSet](../entities/permissionset.md) | [Identity](../entities/identity.md) | [Steal Application Access Token, T1528](https://attack.mitre.org/techniques/T1528/) | + +An identity with a role that allows *get* on secrets (vs list) can potentially view all the serviceaccount tokens in a specific namespace or in the whole cluster (with ClusterRole). ## Details @@ -14,7 +25,7 @@ An attacker in possession of a token with permission to read a secret cannot use Ability to interrogate the K8s API with a role allowing get access to secrets. -See the [example pod spec](../../test/setup/test-cluster/attacks/TOKEN_BRUTEFORCE.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/TOKEN_BRUTEFORCE.yaml). ## Checks @@ -41,8 +52,8 @@ Even *get* on secrets is a very powerful privilege and should not be required by ## Calculation -+ [TokenBruteforce](../../pkg/kubehound/graph/edge/token_bruteforce.go) -+ [TokenBruteforceNamespace](../../pkg/kubehound/graph/edge/token_bruteforce_namespace.go) ++ [TokenBruteforce](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/token_bruteforce.go) ++ [TokenBruteforceNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/token_bruteforce_namespace.go) ## References: diff --git a/docs/edges/TOKEN_LIST.md b/docs/reference/attacks/TOKEN_LIST.md similarity index 73% rename from docs/edges/TOKEN_LIST.md rename to docs/reference/attacks/TOKEN_LIST.md index 631620946..b9d46742f 100644 --- a/docs/edges/TOKEN_LIST.md +++ b/docs/reference/attacks/TOKEN_LIST.md @@ -1,10 +1,21 @@ -# TOKEN_LIST +--- +title: TOKEN_LIST +--- -An identity with a role that allows listing secrets can potentially view all the secrets in a specific namespace or in the whole cluster (with ClusterRole). + + +# TOKEN_LIST | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [PermissionSet](../vertices/PERMISSIONSET.md) | [Identity](../vertices/IDENTITY.md) | [Steal Application Access Token, T1528](https://attack.mitre.org/techniques/T1528/) | +| [PermissionSet](../entities/permissionset.md) | [Identity](../entities/identity.md) | [Steal Application Access Token, T1528](https://attack.mitre.org/techniques/T1528/) | + +An identity with a role that allows listing secrets can potentially view all the secrets in a specific namespace or in the whole cluster (with ClusterRole). ## Details @@ -14,7 +25,7 @@ Obtaining the listing secrets permission will be a significant advantage to an a Ability to interrogate the K8s API with a role allowing list access to secrets. -See the [example pod spec](../../test/setup/test-cluster/attacks/TOKEN_LIST.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setup/test-cluster/attacks/TOKEN_LIST.yaml). ## Checks @@ -44,8 +55,8 @@ Listing secrets is a very powerful privilege and should not be required by the m ## Calculation -+ [TokenList](../../pkg/kubehound/graph/edge/token_list.go) -+ [TokenListNamespace](../../pkg/kubehound/graph/edge/token_list_namespace.go) ++ [TokenList](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/token_list.go) ++ [TokenListNamespace](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/token_list_namespace.go) ## References: diff --git a/docs/edges/TOKEN_STEAL.md b/docs/reference/attacks/TOKEN_STEAL.md similarity index 88% rename from docs/edges/TOKEN_STEAL.md rename to docs/reference/attacks/TOKEN_STEAL.md index 19aefe5ed..576fcc2c6 100644 --- a/docs/edges/TOKEN_STEAL.md +++ b/docs/reference/attacks/TOKEN_STEAL.md @@ -1,10 +1,21 @@ -# TOKEN_STEAL +--- +title: TOKEN_STEAL +--- -This attack represents the ability to steal a K8s API token from an accessible volume. + + +# TOKEN_STEAL | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Volume](../vertices/VOLUME.md) | [Identity](../vertices/IDENTITY.md) | [Unsecured Credentials, T1552](https://attack.mitre.org/techniques/T1552/) | +| [Volume](../entities/volume.md) | [Identity](../entities/identity.md) | [Unsecured Credentials, T1552](https://attack.mitre.org/techniques/T1552/) | + +This attack represents the ability to steal a K8s API token from an accessible volume. ## Details @@ -94,7 +105,7 @@ automountServiceAccountToken: false ## Calculation -+ [TokenSteal](../../pkg/kubehound/graph/edge/token_steal.go) ++ [TokenSteal](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/token_steal.go) ## References: diff --git a/docs/edges/TOKEN_VAR_LOG_SYMLINK.md b/docs/reference/attacks/TOKEN_VAR_LOG_SYMLINK.md similarity index 84% rename from docs/edges/TOKEN_VAR_LOG_SYMLINK.md rename to docs/reference/attacks/TOKEN_VAR_LOG_SYMLINK.md index 21456d0cd..eca64a899 100644 --- a/docs/edges/TOKEN_VAR_LOG_SYMLINK.md +++ b/docs/reference/attacks/TOKEN_VAR_LOG_SYMLINK.md @@ -1,10 +1,21 @@ -# TOKEN_VAR_LOG_SYMLINK +--- +title: TOKEN_VAR_LOG_SYMLINK +--- -Steal all K8s API tokens from a node via an exposed `/var/log` mount. + + +# TOKEN_VAR_LOG_SYMLINK | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Node](../vertices/NODE.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | +| [Container](../entities/container.md) | [Node](../entities/node.md) | [Escape to Host, T1611](https://attack.mitre.org/techniques/T1611/) | + +Steal all K8s API tokens from a node via an exposed `/var/log` mount. ## Details @@ -14,7 +25,7 @@ A pod running as root and with a mount point to the node’s `/var/log` director Execution as root within a container process with the host `/var/log/` (or any parent directory) mounted inside the container. -See the [example pod spec](../../test/setup/test-cluster/attacks/TOKEN_VAR_LOG_SYMLINK.yaml). +See the [example pod spec](https://github.com/DataDog/KubeHound/tree/main/test/setuptest-cluster/attacks/TOKEN_VAR_LOG_SYMLINK.yaml). ## Checks @@ -56,9 +67,12 @@ ln -s / /host/var/log/root_link Call the kubelet API to read the “logs” and extract pod service account tokens: +TODO Christophe: I was unable to call the kubelet on EKS + ```bash $ KUBE_TOKEN=$(cat /var/run/secrets/kubernetes.io/serviceaccount/token) $ NODEIP=$(ip route | awk '/^default/{print $3}') +# On Amazon EKS, if you have access to the IMDS: NODEIP=$(curl http://169.254.169.254/latest/meta-data/local-ipv4) # Find all the pods $ curl -sk -H "Authorization: Bearer $KUBE_TOKEN" https://$NODEIP:10250/logs/root_link/var/lib/kubelet/pods/ @@ -103,7 +117,7 @@ Avoid running containers as the root user. Specify an unprivileged user account ## Calculation -+ [TokenVarLogSymlink](../../pkg/kubehound/graph/edge/token_var_log_symlink.go) ++ [TokenVarLogSymlink](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/token_var_log_symlink.go) ## References: diff --git a/docs/edges/VOLUME_ACCESS.md b/docs/reference/attacks/VOLUME_ACCESS.md similarity index 77% rename from docs/edges/VOLUME_ACCESS.md rename to docs/reference/attacks/VOLUME_ACCESS.md index 7c75c95f1..faa039fb2 100644 --- a/docs/edges/VOLUME_ACCESS.md +++ b/docs/reference/attacks/VOLUME_ACCESS.md @@ -1,10 +1,21 @@ -# VOLUME_ACCESS +--- +title: VOLUME_ACCESS +--- -Represents an attacker with access to a node filesystem gaining access to any volumes mounted inside a container (by definition). + + +# VOLUME_ACCESS | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Node](../vertices/CONTAINER.md) | [Volume](../vertices/VOLUME.md) | [Container and Resource Discovery, T1613](https://attack.mitre.org/techniques/T1613/) | +| [Node](../entities/container.md) | [Volume](../entities/volume.md) | [Container and Resource Discovery, T1613](https://attack.mitre.org/techniques/T1613/) | + +Represents an attacker with access to a node filesystem gaining access to any volumes mounted inside a container (by definition). ## Details @@ -28,7 +39,7 @@ None ## Calculation -+ [VolumeAccess](../../pkg/kubehound/graph/edge/volume_access.go) ++ [VolumeAccess](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/volume_access.go) ## References: diff --git a/docs/edges/VOLUME_DISCOVER.md b/docs/reference/attacks/VOLUME_DISCOVER.md similarity index 91% rename from docs/edges/VOLUME_DISCOVER.md rename to docs/reference/attacks/VOLUME_DISCOVER.md index f8472c191..af0a54e46 100644 --- a/docs/edges/VOLUME_DISCOVER.md +++ b/docs/reference/attacks/VOLUME_DISCOVER.md @@ -1,10 +1,21 @@ -# VOLUME_DISCOVER +--- +title: VOLUME_DISCOVER +--- -Represents an attacker within a container discovering a mounted volume. + + +# VOLUME_DISCOVER | Source | Destination | MITRE | | ----------------------------------------- | ------------------------------------- |----------------------------------| -| [Container](../vertices/CONTAINER.md) | [Volume](../vertices/VOLUME.md) | [Container and Resource Discovery, T1613](https://attack.mitre.org/techniques/T1613/) | +| [Container](../entities/container.md) | [Volume](../entities/volume.md) | [Container and Resource Discovery, T1613](https://attack.mitre.org/techniques/T1613/) | + +Represents an attacker within a container discovering a mounted volume. ## Details @@ -86,7 +97,7 @@ None ## Calculation -+ [VolumeDiscover](../../pkg/kubehound/graph/edge/volume_discover.go) ++ [VolumeDiscover](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/graph/edge/volume_discover.go) ## References: diff --git a/docs/reference/attacks/gen-index.py b/docs/reference/attacks/gen-index.py new file mode 100644 index 000000000..c7fbc73d2 --- /dev/null +++ b/docs/reference/attacks/gen-index.py @@ -0,0 +1,38 @@ +import glob +import yaml + +COMMENT_PREFIX = '' + + +table = """--- +hide: + - toc +--- + +# Attack Reference + +| ID | Name | MITRE ATT&CK Technique | MITRE ATT&CK Tactic | +| :----: | :--: | :-----------------: | :--------------------: | +""" + +for file in sorted(glob.glob('*.md')): + if file == 'index.md': + continue + + with open(file, 'r') as f: + contents = f.read() + startIndex = contents.find(COMMENT_PREFIX) + if startIndex >= 0: + print("Parsing", file) + docsConfig = yaml.safe_load(contents[startIndex+len(COMMENT_PREFIX):contents.find(COMMENT_SUFFIX)]) + attackTacticId, attackTacticName = docsConfig["mitreAttackTactic"].split(' - ') + attackTechniqueId, attackTechniqueName = docsConfig["mitreAttackTechnique"].split(' - ') + table += f'| [{docsConfig["id"]}](./{file}) | {docsConfig["name"]} | {attackTechniqueName} | { attackTacticName} | \n' + else: + print(f"WARNING: {file} does not have a docs config") + + +with open('index.md', 'w') as f: + f.write(table) + \ No newline at end of file diff --git a/docs/reference/attacks/index.md b/docs/reference/attacks/index.md new file mode 100644 index 000000000..9f518ea24 --- /dev/null +++ b/docs/reference/attacks/index.md @@ -0,0 +1,35 @@ +--- +hide: + - toc +--- + +# Attack Reference + +| ID | Name | MITRE ATT&CK Technique | MITRE ATT&CK Tactic | +| :----: | :--: | :-----------------: | :--------------------: | +| [CE_MODULE_LOAD](./CE_MODULE_LOAD.md) | Container escape: Load kernel module | Escape to host | Privilege escalation | +| [CE_NSENTER](./CE_NSENTER.md) | Container escape: nsenter | Escape to host | Privilege escalation | +| [CE_PRIV_MOUNT](./CE_PRIV_MOUNT.md) | Container escape: Mount host filesystem | Escape to host | Privilege escalation | +| [CE_SYS_PTRACE](./CE_SYS_PTRACE.md) | Container escape: Attach to host process via SYS_PTRACE | Escape to host | Privilege escalation | +| [CE_UMH_CORE_PATTERN](./CE_UMH_CORE_PATTERN.md) | Container escape: through core_pattern usermode_helper | Escape to host | Privilege escalation | +| [CONTAINER_ATTACH](./CONTAINER_ATTACH.md) | Attach to running container | N/A | Lateral Movement | +| [ENDPOINT_EXPLOIT](./ENDPOINT_EXPLOIT.md) | Exploit exposed endpoint | Exploitation of Remote Services | Lateral Movement | +| [EXPLOIT_CONTAINERD_SOCK](./EXPLOIT_CONTAINERD_SOCK.md) | Container escape: Through mounted container runtime socket | N/A | Lateral Movement | +| [EXPLOIT_HOST_READ](./EXPLOIT_HOST_READ.md) | Read file from sensitive host mount | Escape to host | Privilege escalation | +| [EXPLOIT_HOST_TRAVERSE](./EXPLOIT_HOST_TRAVERSE.md) | Steal service account token through kubelet host mount | Unsecured Credentials | Credential Access | +| [EXPLOIT_HOST_WRITE](./EXPLOIT_HOST_WRITE.md) | Container escape: Write to sensitive host mount | Escape to host | Privilege escalation | +| [IDENTITY_ASSUME](./IDENTITY_ASSUME.md) | Act as identity | Valid Accounts | Privilege escalation | +| [IDENTITY_IMPERSONATE](./IDENTITY_IMPERSONATE.md) | Impersonate user/group | Valid Accounts | Privilege escalation | +| [PERMISSION_DISCOVER](./PERMISSION_DISCOVER.md) | Enumerate permissions | Permission Groups Discovery | Discovery | +| [POD_ATTACH](./POD_ATTACH.md) | Attach to running pod | N/ATODO | Lateral Movement | +| [POD_CREATE](./POD_CREATE.md) | Create privileged pod | Scheduled Task/Job: Container Orchestration Job | Privilege escalation | +| [POD_EXEC](./POD_EXEC.md) | Exec into running pod | N/A | Lateral Movement | +| [POD_PATCH](./POD_PATCH.md) | Patch running pod | N/A | Lateral Movement | +| [ROLE_BIND](./ROLE_BIND.md) | Create role binding | Valid Accounts | Privilege Escalation | +| [SHARED_PS_NAMESPACE](./SHARED_PS_NAMESPACE.md) | Access container in shared process namespace | N/A | Lateral Movement | +| [TOKEN_BRUTEFORCE](./TOKEN_BRUTEFORCE.md) | Brute-force secret name of service account token | Steal Application Access Token | Credential Access | +| [TOKEN_LIST](./TOKEN_LIST.md) | Access service account token secrets | Steal Application Access Token | Credential Access | +| [TOKEN_STEAL](./TOKEN_STEAL.md) | Steal service account token from volume | Unsecured Credentials | Credential Access | +| [TOKEN_VAR_LOG_SYMLINK](./TOKEN_VAR_LOG_SYMLINK.md) | Steal service account token from volume | Unsecured Credentials | Credential Access | +| [VOLUME_ACCESS](./VOLUME_ACCESS.md) | Access host volume | Container and Resource Discovery | Discovery | +| [VOLUME_DISCOVER](./VOLUME_DISCOVER.md) | Enumerate mounted volumes | Container and Resource Discovery | Discovery | diff --git a/docs/vertices/COMMON.md b/docs/reference/entities/common.md similarity index 100% rename from docs/vertices/COMMON.md rename to docs/reference/entities/common.md diff --git a/docs/vertices/CONTAINER.md b/docs/reference/entities/container.md similarity index 83% rename from docs/vertices/CONTAINER.md rename to docs/reference/entities/container.md index 75145fb79..a40cbd29e 100644 --- a/docs/vertices/CONTAINER.md +++ b/docs/reference/entities/container.md @@ -25,17 +25,17 @@ Properties that are interesting to attackers can be set at a Pod level such as h ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [compromised](./COMMON.md#risk-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [compromised](./common.md#risk-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.Container](../../pkg/kubehound/models/graph/container.go) +[vertex.Container](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/container.go) ## References diff --git a/docs/vertices/ENDPOINT.md b/docs/reference/entities/endpoint.md similarity index 68% rename from docs/vertices/ENDPOINT.md rename to docs/reference/entities/endpoint.md index ea6bf76f2..23d77170e 100644 --- a/docs/vertices/ENDPOINT.md +++ b/docs/reference/entities/endpoint.md @@ -14,22 +14,22 @@ A network endpoint exposed by a container accessible via a Kubernetes service, e | port | `int` | Exposed port of the endpoint | | portName | `string` | Name of the exposed port | | protocol | `string` | Endpoint protocol (TCP, UDP, etc) | -| exposure | `string` | Enum value describing the level of exposure of the endpoint (see [EndpointExposureType](../../pkg/kubehound/models/shared/constants.go)) | +| exposure | `string` | Enum value describing the level of exposure of the endpoint (see [EndpointExposureType](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/shared/constants.go)) | ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [compromised](./COMMON.md#risk-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [compromised](./common.md#risk-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.Endpoint](../../pkg/kubehound/models/graph/endpoint.go) +[vertex.Endpoint](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/endpoint.go) ## References diff --git a/docs/vertices/IDENTITY.md b/docs/reference/entities/identity.md similarity index 61% rename from docs/vertices/IDENTITY.md rename to docs/reference/entities/identity.md index 6a20b9134..4c9a901a3 100644 --- a/docs/vertices/IDENTITY.md +++ b/docs/reference/entities/identity.md @@ -11,17 +11,17 @@ Identity represents a Kubernetes user or service account. ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [critical](./COMMON.md#risk-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [critical](./common.md#risk-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.Identity](../../pkg/kubehound/models/graph/identity.go) +[vertex.Identity](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/identity.go) ## References diff --git a/docs/reference/entities/index.md b/docs/reference/entities/index.md new file mode 100644 index 000000000..e69de29bb diff --git a/docs/vertices/NODE.md b/docs/reference/entities/node.md similarity index 52% rename from docs/vertices/NODE.md rename to docs/reference/entities/node.md index 08db3031b..162c7d034 100644 --- a/docs/vertices/NODE.md +++ b/docs/reference/entities/node.md @@ -10,18 +10,18 @@ A Kubernetes node. Kubernetes runs workloads by placing containers into Pods to ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [compromised](./COMMON.md#risk-information) -+ [critical](./COMMON.md#risk-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [compromised](./common.md#risk-information) ++ [critical](./common.md#risk-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.Node](../../pkg/kubehound/models/graph/node.go) +[vertex.Node](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/node.go) ## References diff --git a/docs/vertices/PERMISSIONSET.md b/docs/reference/entities/permissionset.md similarity index 59% rename from docs/vertices/PERMISSIONSET.md rename to docs/reference/entities/permissionset.md index 8cc3e85ea..8fc2edcc7 100644 --- a/docs/vertices/PERMISSIONSET.md +++ b/docs/reference/entities/permissionset.md @@ -7,21 +7,21 @@ A permission set represents a Kubernetes RBAC `Role` or `ClusterRole`, which con | Property | Type | Description | | ----------------| --------- |----------------------------------------| | name | `string` | Name of the underlying role in Kubernetes | -| rules | `[]string` | List of strings representing the access granted by the role (see generator function [flattenPolicyRules](../../pkg/kubehound/models/converter/graph.go))| +| rules | `[]string` | List of strings representing the access granted by the role (see generator function [flattenPolicyRules](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/converter/graph.go))| ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [critical](./COMMON.md#risk-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [critical](./common.md#risk-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.PermissionSet](../../pkg/kubehound/models/graph/permission_set.go) +[vertex.PermissionSet](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/permission_set.go) ## References diff --git a/docs/vertices/POD.md b/docs/reference/entities/pod.md similarity index 67% rename from docs/vertices/POD.md rename to docs/reference/entities/pod.md index 501de5695..e82d9dac8 100644 --- a/docs/vertices/POD.md +++ b/docs/reference/entities/pod.md @@ -13,18 +13,18 @@ A Kubernetes pod - the smallest deployable units of computing that you can creat ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [compromised](./COMMON.md#risk-information) -+ [critical](./COMMON.md#risk-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [compromised](./common.md#risk-information) ++ [critical](./common.md#risk-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.Pod](../../pkg/kubehound/models/graph/pod.go) +[vertex.Pod](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/pod.go) ## References diff --git a/docs/vertices/VOLUME.md b/docs/reference/entities/volume.md similarity index 70% rename from docs/vertices/VOLUME.md rename to docs/reference/entities/volume.md index e924e0916..aa98d75b2 100644 --- a/docs/vertices/VOLUME.md +++ b/docs/reference/entities/volume.md @@ -14,16 +14,16 @@ Volume represents a volume mounted in a container and exposed by a node. ## Common Properties -+ [storeID](./COMMON.md#store-information) -+ [app](./COMMON.md#ownership-information) -+ [team](./COMMON.md#ownership-information) -+ [service](./COMMON.md#ownership-information) -+ [namespace](./COMMON.md#namespace-information) -+ [isNamespaced](./COMMON.md#namespace-information) ++ [storeID](./common.md#store-information) ++ [app](./common.md#ownership-information) ++ [team](./common.md#ownership-information) ++ [service](./common.md#ownership-information) ++ [namespace](./common.md#namespace-information) ++ [isNamespaced](./common.md#namespace-information) ## Definition -[vertex.Volume](../../pkg/kubehound/models/graph/volume.go) +[vertex.Volume](https://github.com/DataDog/KubeHound/tree/main/pkg/kubehound/models/graph/volume.go) ## References diff --git a/docs/terminology.md b/docs/terminology.md new file mode 100644 index 000000000..c42592d6b --- /dev/null +++ b/docs/terminology.md @@ -0,0 +1,5 @@ +# Terminology + +TODO +- Entity/edge/resource +- Attack/relationship diff --git a/docs/user-guide/common-operations.md b/docs/user-guide/common-operations.md new file mode 100644 index 000000000..30404ce4c --- /dev/null +++ b/docs/user-guide/common-operations.md @@ -0,0 +1 @@ +TODO \ No newline at end of file diff --git a/docs/user-guide/getting-started.md b/docs/user-guide/getting-started.md new file mode 100644 index 000000000..665e6f68c --- /dev/null +++ b/docs/user-guide/getting-started.md @@ -0,0 +1,99 @@ +# Getting started + +## Prerequisites + +To get started with KubeHound, you'll need the following pre-requirements on your system: + +- [Golang](https://go.dev/doc/install) >= 1.20 (`go version`) +- [Docker](https://docs.docker.com/engine/install/) >= 19.03 (`docker version`) +- [Docker Compose](https://docs.docker.com/compose/compose-file/compose-versioning/) >= v2.0 (`docker compose version`) + +## Running KubeHound + +??? info "tl;dr" + + ```bash + make kubehound && bin/kubehound + ``` + +KubeHound ships with a sensible default configuration designed to get new users up and running quickly. First, prepare the application by running: + +```bash +make kubehound +``` + +This will start [backend services](../architecture.md) via docker compose (wiping any existing data), and compile the kubehound binary from source + +Next, make sure your current kubectl context points at the target cluster: + +```bash +# View the current context +kubectl config current-context + +# Set your context +kubectl config set-context + +# alternatively, use https://github.com/ahmetb/kubectx +``` + +Finally, run KubeHound with the default [configuration](TODO): + +``` +bin/kubehound +``` + +Sample output: + +```text +INFO[0000] Starting KubeHound (run_id: aff49337-5e36-46ea-ac1f-ed224bf215ba) component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Initializing launch options component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loading application configuration from default embedded component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Initializing application telemetry component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loading cache provider component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loaded MemCacheProvider cache provider component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loading store database provider component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loaded MongoProvider store provider component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loading graph database provider component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0000] Loaded JanusGraphProvider graph provider component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0001] Starting Kubernetes raw data ingest component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0001] Loading Kubernetes data collector client component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0001] Loaded k8s-api-collector collector client component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +... +INFO[0028] Building edge ExploitHostWrite component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0028] Edge writer 22 ContainerAttach::CONTAINER_ATTACH written component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0028] Building edge IdentityAssumeNode component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0029] Edge writer 8 ExploitHostWrite::EXPLOIT_HOST_WRITE written component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +... +INFO[0039] Completed edge construction component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0039] Completed graph construction component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +INFO[0039] Attack graph generation complete in 39.108174109s component=kubehound run_id=aff49337-5e36-46ea-ac1f-ed224bf215ba service=kubehound +``` + + +## Access the KubeHound data + +At this point, the KubeHound data has been ingested in KubeHound's [graph database](../architecture.md). +You can use any client that supports accessing JanusGraph - we recommend using [gdotv](https://gdotv.com/): + +- [Download and install] gdotv from the [official website](https://gdotv.com/) +- [Create a connection] to the local KubeHound JanusGraph instance + 1. Click on the `New database connection` button + 2. Enter `localhost` as an hostname, and click on the `Test connection` button + 3. Once the connection is successful, click `Submit` - you're good to go! + +## Visualize and query the KubeHound data + +Once the data is loaded in the graph database, it's time to visualize and query it! + +You can explore it interactively in your graph client. Then, refer to KubeHound's [query library](../queries/index.md) to start asking questions to your data. + +## Generating sample data + +If you don't have a cluster at your disposal, install [kind](https://kind.sigs.k8s.io/#installation-and-usage) and run the following command: + +```bash +make sample-graph +``` + +This will spin up a temporary local kind cluster, run KubeHound on it, and destroy the cluster. + diff --git a/docs/user-guide/troubleshooting.md b/docs/user-guide/troubleshooting.md new file mode 100644 index 000000000..30404ce4c --- /dev/null +++ b/docs/user-guide/troubleshooting.md @@ -0,0 +1 @@ +TODO \ No newline at end of file diff --git a/mkdocs.yml b/mkdocs.yml new file mode 100644 index 000000000..52d8dc0b8 --- /dev/null +++ b/mkdocs.yml @@ -0,0 +1,88 @@ +site_name: KubeHound +repo_url: https://github.com/DataDog/KubeHound +repo_name: datadog/KubeHound +plugins: + - search + - awesome-pages +theme: + name: material + palette: + - scheme: default + primary: black + toggle: + icon: material/toggle-switch-off-outline + name: Switch to dark mode + - scheme: slate + primary: black + toggle: + icon: material/toggle-switch + name: Switch to light mode + features: + - navigation.expand + - navigation.indexes + - navigation.top + - navigation.tabs + - navigation.tabs.sticky + - content.code.copy + icon: + admonition: + note: octicons/tag-16 + abstract: octicons/checklist-16 + info: octicons/info-16 + tip: octicons/squirrel-16 + success: octicons/check-16 + question: octicons/question-16 + warning: octicons/alert-16 + failure: octicons/x-circle-16 + danger: octicons/zap-16 + bug: octicons/bug-16 + example: octicons/beaker-16 + quote: octicons/quote-16 + use_directory_urls: true + extra_javascript: + - path: js/tablesort.js #TODO not working + extra: + social: {} + +markdown_extensions: + - pymdownx.highlight: + anchor_linenums: true + - pymdownx.inlinehilite + - pymdownx.snippets + - pymdownx.superfences + - pymdownx.magiclink + - tables + - meta + - abbr + - def_list + - attr_list + - md_in_html + - admonition + - pymdownx.details + - footnotes + - markdown_captions + - pymdownx.tabbed: + alternate_style: true + +nav: + - KubeHound: + - index.md + - Architecture: architecture.md + - Terminology: terminology.md + - Frequently Asked Questions: faq.md + - Comparison with Other Tools: comparison.md + - Contributing: contributing.md + - User Guide: + - Getting Started: user-guide/getting-started.md + - Common Operations: user-guide/common-operations.md + - Troubleshooting: user-guide/troubleshooting.md + - Reference: + - ... |reference/*/*.md + #- Attacks: reference/attacks/index.md + #- Entities: reference/entities/index.md + #- ... | reference/attacks/*.md + #- ... | reference/entities/*.md + - Query Library: + - queries/index.md + +copyright: Copyright 2023-Present Datadog, Inc. \ No newline at end of file