pkg/aggregator/README.md
The data plane aggregator is a reverse proxy similar to kube-aggregator that sits first in the request path. It is responsible for reverse proxying any registered data plane (anything that isn't CRUD + List + Watch) routes to the appropriate app handlers. Anything that does not match the registered data planeroutes will be delegated to the next apiserver in the handler chain (e.g. kube-aggregator). Currently the aggregator uses the Grafana plugin framework to register and proxy requests to Grafana plugins. The benefit of this approach is that it allows apps to be built, packaged, published, and deployed using the existing Grafana plugin framework.
The data plane aggregator is built on top of the Kubernetes API server framework. It introduces a new DataPlaneService Custom Resource Definition (CRD) to dynamically register and proxy requests to Grafana plugins.
The main components are:
pkg/aggregator/apiserver/apiserver.go. It's responsible for:
DataPlaneService resource.DataPlaneServiceRegistrationController.pkg/aggregator/apiserver/dataplaneservice_controller.go, is responsible for:
add, update, and delete events on DataPlaneService resources.DataPlaneService to the PluginHandler to add, update, or remove the corresponding proxy handlers.pkg/aggregator/apiserver/plugin/handler.go, is responsible for:
DataPlaneService resources.pkg/aggregator/apis/aggregation/v0alpha1/types.go, this represents a service provided by a Grafana plugin.The following diagram illustrates the request flow:
sequenceDiagram
autonumber
participant plugin as disk
participant exe as plugin
binaries
participant wrap as apiserver
participant agg as kube-aggregator
participant data as dataplane-aggregator
participant storage as Unified
Storage
Note over plugin,storage: Registration
wrap->>plugin: Load Scope app manifest
wrap->>agg: Create Scope APIService
wrap->>plugin: Load SLO app manifest
wrap->>agg: Create SLO APIService
wrap->>plugin: Load Dashboard app manifest
wrap->>agg: Create Dashboard APIService
wrap->>data: Create Dashboard DataPlaneService
Note over plugin,storage: Storage Access
data->>storage: DataPlaneServices
agg->>storage: APIServices
wrap->>storage: Custom resources
Note over plugin,storage: Create, Update, or Delete Hooks
data->>agg: Local delegation to kube-aggregator
agg->>wrap: HTTP reverse proxy incoming request
wrap->>exe: gRPC admission mutation hook
wrap->>exe: gRPC admission validation hook
Note over plugin,storage: Get, List, or Watch Hooks
data->>agg: Local delegation to kube-aggregator
agg->>wrap: HTTP reverse proxy incoming request
wrap->>exe: gRPC conversion hook
Note over plugin,storage: Subresource & Custom Routes
data->>exe: gRPC plugin resource calls
data->>exe: gRPC plugin query data
data->>exe: gRPC plugin stream
data->>exe: gRPC plugin health
data->>exe: gRPC plugin metrics
Note over plugin,storage: Async (controllers)
exe->>data: Watch
data->>agg: Local delegation to kube-aggregator
agg->>wrap: HTTP reverse proxy incoming request
wrap->>storage: Watch
A DataPlaneService is a non-namespaced resource that represents a service exposed by a Grafana plugin. Here is an example:
apiVersion: aggregation.grafana.app/v0alpha1
kind: DataPlaneService
metadata:
name: <plugin-id>
spec:
pluginID: <plugin-id>
pluginType: <plugin-type> # app or datasource
group: <api-group>
version: <api-version>
services:
- type: <service-type> # admission, conversion, query, stream, route, datasource-proxy
method: <http-method>
path: <url-path>
custom.ini changes:[feature_toggles]
dataplaneAggregator = true
grafanaAPIServerEnsureKubectlAccess = true
make run
export KUBECONFIG=./data/grafana-apiserver/grafana.kubeconfig
kubectl apply -f pkg/aggregator/examples/datasource.yml --validate=false
edit pkg/aggregator/examples/datasource-query.json and update the datasource UID to match the UID of a prometheus data source.
execute query (replace <datasource-uid> with the UID of a prometheus data source):
curl 'http://admin:admin@localhost:3000/apis/prometheus.grafana.app/v0alpha1/namespaces/default/datasources/<datasource-uid>/query' -X POST -d '@pkg/aggregator/examples/datasource-query.json'