cartography-openapi is a tool for generating modules for Cartography from specification files in OpenAPI format.
Developing modules for Cartography can be confusing, especially regarding the data model and various requirements. Importing multiple resources can also be tedious.
With this in mind, cartography-openapi helps generate the skeleton of a module, allowing developers to focus on added value (drift detection, custom links, etc.).
For more details, refer to the Cartography developer documentation.
cartography-openapi supports importing files in OpenAPI v3 format.
For more details, refer to the OpenAPI Specification
See cartography_openapi.parser.Parser._parse
- Parse the server block to try to determine the API
base_url. - Parse the components:
- List properties that will become fields (with types such as string, int, etc.).
- List properties that will become relationships to other nodes (fields that return another object).
- Parse the paths:
- Keep only paths with the
GETmethod that return known components. - Extract path parameters, query parameters, and body parameters.
- Keep only paths with the
See cartography_openapi.parser.Parser.build_module
- Entities will be built in the order they were declared in the command invocation.
- For each entity, the tool will:
- Find the corresponding component in the API.
- Identify the best path (1) to enumerate the list of these components.
- Identify the best path (1) to access a single component.
- From the identified paths, determine if a parent entity can be defined.
See cartography_openapi.component.Component.set_enumeration_path and cartography_openapi.component.Component.set_direct_path.
This part is the core of the tool’s functionality. It empirically determines:
- The paths used to list all entities.
- The paths used to access a specific component.
The logic is as follows:
If the path to access a Realm is /admin/realm/{realmId} and the path to access a Group is /admin/realm/{realmId}/group/{groupId}, then:
- Group is considered a sub-resource of Realm.
- An edge must exist between the two nodes.
- The enumeration must be handled recursively.
Path evaluation is based on the following criteria:
- No previous path
- Linkable vs non-linkable (the path is a sub-path of the direct path of another component)
- The new path is better because it has less parameters
- The new path is better because it is shorter (allow to prefer x/groups over x/groups-default)
See cartography_openapi.module.Module.export
- The model (nodes & relationships) of each entity is exported to a specific file using a Jinja template.
- Each entity generates an "intel" file containing the logic to sync, get, transform, and load the entity into Cartography.
- The module generates the
__init__.pyfile for the module's intel, which calls each entity-specific file via thesyncfunction. - The
syncfunction, which will be called by Cartography, is built recursively to handle sub-resources:- Example: Synchronizing tenants
- For each tenant, synchronize environments
- For each environment, synchronize resources
- Example: Synchronizing tenants
pip3 install cartography-openapi
uv sync --frozen
uv run python3 cartography_openapi
wget https://www.keycloak.org/docs-api/latest/rest-api/openapi.json
uv run python3 main.py -v -n Keycloak -f ./openapi.json RealmRepresentation=Realm ClientRepresentation=Client GroupRepresentation=Group UserRepresentation=User
For more examples, refer to the examples folder.
You must provide the list of API components to include in the module. You can specify a different name for these components (e.g., RealmRepresentation=Realm will import the RealmRepresentation component of the API under the Realm node).
| Option | Required | Description |
|---|---|---|
| --name (-n) | YES | Name of the intel module |
| --url (-u) | (*) | URL of the OpenAPI specifications |
| --file (-f) | (*) | Path of the OpenAPI specifications |
| --verbose (-v) | Display debug level messsages | |
| --output (-o) | Output directory (default .) |
|
| --ignore (-i) | Ignore specific paths (e.g. /path/to/ignore or /path/to/*) |
(*) You must provide one (and only one source) for the OpenAPI specification
cartography-openapi will generate a <OUTPUT>/<NAME>_module folder containing the necessary files to add the module to Cartography. This folder includes:
- The
modelfolder, which must be moved tocartography/model/<NAME> - The
intelfolder, which must be moved tocartography/intel/<NAME> - The
tests_datafolder, which must be moved totests/data/<NAME> - The
tests_integrationfolder, wich must be moved totests/integration/cartography/intel/<NAME> - The
docsfolder, wich must be moved todocs/root/modules/<NAME>
The following operations must be performed manually:
- Add the necessary configuration keys in
cartography/config.py - Add the required parameters in
cartography/cli.py - Import your module in
cartography/sync.py - Add a reference to the module in
docs/root/usage/schema.md - Update the cartography
README.md
cartography-openapi is a Proof of Concept, and many features are still missing (we are actively working on them):
- The generated code does not handle authentication for API calls
This project is licensed under the Apache 2.0 License.
Here are the topics we are working on for upcoming releases:
- handle authentication
- generate Cartography config