Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 6 additions & 1 deletion CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,11 @@

Fork the repo, work on an issue

## Updating the generated Kibana client.

If you're work involved the Kibana API, the endpoints may or may not be included in the generated client.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
If you're work involved the Kibana API, the endpoints may or may not be included in the generated client.
If your work involves the Kibana API, the endpoints may or may not be included in the generated client.

Check [generated/kbapi](./generated/kbapi/) for more details.

## Acceptance tests

```bash
Expand Down Expand Up @@ -60,4 +65,4 @@ To release a new provider version:
- updates CHANGELOG.md with the list of changes being released.
[Example](https://github.com/elastic/terraform-provider-elasticstack/commit/be866ebc918184e843dc1dd2f6e2e1b963da386d).

* Once the PR merged, the release CI pipeline can be started by pushing a new release tag to `main` branch.
* Once the PR merged, the release CI pipeline can be started by pushing a new release tag to `main` branch.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
* Once the PR merged, the release CI pipeline can be started by pushing a new release tag to `main` branch.
* Once the PR is merged, the release CI pipeline can be started by pushing a new release tag to the `main` branch.

47 changes: 47 additions & 0 deletions generated/kbapi/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
# Generated Kibana API client

This package includes an API client generated from the Kibana OpenAPI spec.
This client is very much a work in progress, there's opportunity for improvement both within this provider, but also within the OpenAPI spec created during the Kibana build process.

This readme serves to document the status quo, contributions improving this current process are very much welcome.

## Adding new endpoints

There's more detail on the full process below, this section aims to provide a tl;dr to what's required to cover new endpoints in the generated client.

1. `make transform`. This ensures you're starting with the latest API spec. If you encounter issues at this point, at least you know it's not related to any changes you've made :)
1. Add the required API paths to the allow list. See `transformFilterPaths` in `transform_schema.go`. The definition here is straightforward, the path must match exactly what's in the API spec.
1. If endpoints don't use any `oneOf` fields then you're likely done. Run `make transform generate` and start work on the provider resource. Sweeping statements like this tend to be wrong at times, if you encounter issues with the generated client then you'll have to investigate them and may need to define additional transforms.
1. If the endpoints use `oneOf` in the request/response bodies then you'll need to define additional transforms to extract those to reusable components. See `transformKibanaPaths` or `transformFleetPaths` for some examples.
1. `make transform generate` and go test the fresh client.

## Client generation

The actual final Go client generation is relatively straightforward. `make generate` executes [`oapi-codegen`](https://github.com/oapi-codegen/oapi-codegen) which creates `kibana.gen.go` from `oas-filtered.yaml`.

## OpenAPI spec transforms.

This is where the dragons lie.

`make transform` downloads the schema in kibana/main, and passes it through `transform_schema.go`. This transform step aims to patch over:
* Issues with the Kibana schema, any issues we encounter should also be raised in the Kibana repo.
* Issues with oapi-codegen, similarly we should ensure there's a corresponding issue in the oapi-codegen repo.
Copy link
Contributor

@dimuon dimuon Jul 4, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does it make sense to try another tool, e.g. ogen?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep, it's definitely worth investigating. I've tried a bunch of these previously (I can't remember if that includes ogen) and there bugs in every generator I tried. It's worth checking if ogen works better with the current Kibana spec though.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ogen-go/ogen#1480 sounds like we might run into similar issues, still worth a try though.


At a high level, `transform_schema.go`:
* Filters out API paths no in the allow list (see `transformFilterPaths`).
* Removes any `kbnXsrf` parameters. We add this heaer in the client by default. We should look to remove this transform to simplify the overall process.
* Removes the `elastic-api-version` header. We only have a single version at the moment, that said we should remove this transform and properly define this logic within the client.
* Transforms any versioned content types (e.g `application/json; Elastic-Api-Version=2023-10-31`) into plain `application/json`. Similarly to above, the TF provider should properly handle these content types moving forward.
* Ensures Kibana api paths have a `spaceId` parameter (see `transformKibanaPaths`). We should ensure Kibana is correctly including this parameter in any cases where it's required. Not defining this parameter should be seen as a bug in spec generation, not something for clients to work around.
* Extracts nested type definitions to a shared component (see `transformKibanaPaths`). We've encountered issues in oapi-codegen (https://github.com/oapi-codegen/oapi-codegen/issues/1900) where nested, polymorphic types (`oneOf`) create a syntactically invalid API client. To workaround this, `transform_schema` pulls those nested types into re-usable components, and then updated the schema to reference those new types.
* Ultimately the spec is valid here, it would be ideal for oapi-codegen to handle this correctly. That said, this transformation is the most tedious, and complicated in this stack. If we continue to encounter this issue long term we should look to see if it's possible for the Kibana spec generation to create these re-usable types directly.
* Makes several endpoint specific request/response body updates. There are several bugs within some request/response body definitions. These have all been raised as issues in the Kibana repo, in the meantime the transformer is patching the schema to create a working client.

## Possible improvements

These should likely be issues in this repo...

1. Avoid the need for this transformation stuff entirely? It doesn't seem unreasonable to expect the spec provided by Kibana to 'work'. IMO that includes avoiding long standing issues with well used client generators if possible. Anything we can fix within the Kibana repo should be fixed there in preference to adding new transformation code.
1. Migrate the bespoke `transform_schema` to [redocly](https://redocly.com/docs/cli) or oapi-codegen [overlays](https://github.com/oapi-codegen/oapi-codegen?tab=readme-ov-file#modifying-the-input-openapi-specification-with-openapi-overlay). We're reinventing the wheel here. Specifically with the `$ref` issue, if this requires local transformation long term then we should blanket apply this transform to all request/response bodies. That's likely easier done with a well tested tool like redocly than it is with homegrown tooling.
1. Include all paths by default. The barrier here is the high level of transformation required to produce a working client. That's likely reduced substantially since `transform_schema` was introduced and it's likely worth investigation.
1. Remove pointless transforms. Why are we removing examples?
39 changes: 0 additions & 39 deletions generated/kbapi/transform_schema.go
Original file line number Diff line number Diff line change
Expand Up @@ -545,8 +545,6 @@ var transformers = []TransformFunc{
transformAddMisingDescriptions,
transformKibanaPaths,
transformFleetPaths,
// transformRemoveEnums,
// transformAddGoPointersFlag,
transformRemoveExamples,
transformRemoveUnusedComponents,
}
Expand Down Expand Up @@ -1014,22 +1012,6 @@ func transformFleetPaths(schema *Schema) {
schema.Components.Set("schemas.package_policy_request.properties.output_id.x-omitempty", true)
}

// transformRemoveEnums remove all enums.
func transformRemoveEnums(schema *Schema) {
deleteEnumFn := func(key string, node Map) {
if node.Has("enum") {
delete(node, "enum")
}
}

for _, pathInfo := range schema.Paths {
for _, methInfo := range pathInfo.Endpoints {
methInfo.Iterate(deleteEnumFn)
}
}
schema.Components.Iterate(deleteEnumFn)
}

// transformRemoveExamples removes all examples.
func transformRemoveExamples(schema *Schema) {
deleteExampleFn := func(key string, node Map) {
Expand All @@ -1050,27 +1032,6 @@ func transformRemoveExamples(schema *Schema) {
schema.Components.Set("examples", Map{})
}

// transformAddOptionalPointersFlag adds a x-go-type-skip-optional-pointer
// flag to maps and arrays, since they are already nullable types.
func transformAddOptionalPointersFlag(schema *Schema) {
addFlagFn := func(key string, node Map) {
if node["type"] == "array" {
node["x-go-type-skip-optional-pointer"] = true
} else if node["type"] == "object" {
if _, ok := node["properties"]; !ok {
node["x-go-type-skip-optional-pointer"] = true
}
}
}

for _, pathInfo := range schema.Paths {
for _, methInfo := range pathInfo.Endpoints {
methInfo.Iterate(addFlagFn)
}
}
schema.Components.Iterate(addFlagFn)
}

// transformRemoveUnusedComponents removes all unused schema components.
func transformRemoveUnusedComponents(schema *Schema) {
var refs map[string]any
Expand Down
Loading