1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 03:05:05 +08:00
gf/contrib/config/kubecm/kubecm.go

180 lines
5.5 KiB
Go

// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
//
// This Source Code Form is subject to the terms of the MIT License.
// If a copy of the MIT was not distributed with this file,
// You can obtain one at https://github.com/gogf/gf.
// Package kubecm implements gcfg.Adapter using kubernetes configmap.
package kubecm
import (
"context"
"fmt"
kubeMetaV1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/watch"
"k8s.io/client-go/kubernetes"
"k8s.io/client-go/rest"
"github.com/gogf/gf/v2/encoding/gjson"
"github.com/gogf/gf/v2/errors/gerror"
"github.com/gogf/gf/v2/frame/g"
"github.com/gogf/gf/v2/os/gcfg"
"github.com/gogf/gf/v2/util/gutil"
)
// Client implements gcfg.Adapter.
type Client struct {
config Config // Config object when created.
client *kubernetes.Clientset // Kubernetes client.
value *g.Var // Configmap content cached. It is `*gjson.Json` value internally.
}
// Config for Client.
type Config struct {
ConfigMap string `v:"required"` // ConfigMap name.
DataItem string `v:"required"` // DataItem is the key item in Configmap data.
Namespace string // Specify the namespace for configmap.
RestConfig *rest.Config // Custom rest config for kube client.
KubeClient *kubernetes.Clientset // Custom kube client.
Watch bool // Watch watches remote configuration updates, which updates local configuration in memory immediately when remote configuration changes.
}
// New creates and returns gcfg.Adapter implementing using kubernetes configmap.
func New(ctx context.Context, config Config) (adapter gcfg.Adapter, err error) {
// Data validation.
err = g.Validator().Data(config).Run(ctx)
if err != nil {
return nil, err
}
// Kubernetes client creating.
if config.KubeClient == nil {
if config.RestConfig == nil {
config.RestConfig, err = NewDefaultKubeConfig(ctx)
if err != nil {
return nil, gerror.Wrapf(err, `create kube config failed`)
}
}
config.KubeClient, err = kubernetes.NewForConfig(config.RestConfig)
if err != nil {
return nil, gerror.Wrapf(err, `create kube client failed`)
}
}
adapter = &Client{
config: config,
client: config.KubeClient,
value: g.NewVar(nil, true),
}
return
}
// Available checks and returns the backend configuration service is available.
// The optional parameter `resource` specifies certain configuration resource.
//
// Note that this function does not return error as it just does simply check for
// backend configuration service.
func (c *Client) Available(ctx context.Context, configMap ...string) (ok bool) {
if len(configMap) == 0 && !c.value.IsNil() {
return true
}
var (
namespace = gutil.GetOrDefaultStr(Namespace(), c.config.Namespace)
configMapName = gutil.GetOrDefaultStr(c.config.ConfigMap, configMap...)
)
_, err := c.config.KubeClient.CoreV1().ConfigMaps(namespace).Get(ctx, configMapName, kubeMetaV1.GetOptions{})
if err != nil {
return false
}
return true
}
// Get retrieves and returns value by specified `pattern` in current resource.
// Pattern like:
// "x.y.z" for map item.
// "x.0.y" for slice item.
func (c *Client) Get(ctx context.Context, pattern string) (value interface{}, err error) {
if c.value.IsNil() {
if err = c.updateLocalValueAndWatch(ctx); err != nil {
return nil, err
}
}
return c.value.Val().(*gjson.Json).Get(pattern).Val(), nil
}
// Data retrieves and returns all configuration data in current resource as map.
// Note that this function may lead lots of memory usage if configuration data is too large,
// you can implement this function if necessary.
func (c *Client) Data(ctx context.Context) (data map[string]interface{}, err error) {
if c.value.IsNil() {
if err = c.updateLocalValueAndWatch(ctx); err != nil {
return nil, err
}
}
return c.value.Val().(*gjson.Json).Map(), nil
}
// init retrieves and caches the configmap content.
func (c *Client) updateLocalValueAndWatch(ctx context.Context) (err error) {
var namespace = gutil.GetOrDefaultStr(Namespace(), c.config.Namespace)
err = c.doUpdate(ctx, namespace)
if err != nil {
return err
}
err = c.doWatch(ctx, namespace)
if err != nil {
return err
}
return nil
}
func (c *Client) doUpdate(ctx context.Context, namespace string) (err error) {
cm, err := c.client.CoreV1().ConfigMaps(namespace).Get(ctx, c.config.ConfigMap, kubeMetaV1.GetOptions{})
if err != nil {
return gerror.Wrapf(
err,
`retrieve configmap "%s" from namespace "%s" failed`,
c.config.ConfigMap, namespace,
)
}
var j *gjson.Json
if j, err = gjson.LoadContent(cm.Data[c.config.DataItem]); err != nil {
return gerror.Wrapf(
err,
`parse config map item from %s[%s] failed`, c.config.ConfigMap, c.config.DataItem,
)
}
c.value.Set(j)
return nil
}
func (c *Client) doWatch(ctx context.Context, namespace string) (err error) {
if !c.config.Watch {
return nil
}
var watchHandler watch.Interface
watchHandler, err = c.client.CoreV1().ConfigMaps(namespace).Watch(ctx, kubeMetaV1.ListOptions{
FieldSelector: fmt.Sprintf(`metadata.name=%s`, c.config.ConfigMap),
Watch: true,
})
if err != nil {
return gerror.Wrapf(
err,
`watch configmap "%s" from namespace "%s" failed`,
c.config.ConfigMap, namespace,
)
}
go c.startAsynchronousWatch(ctx, namespace, watchHandler)
return nil
}
func (c *Client) startAsynchronousWatch(ctx context.Context, namespace string, watchHandler watch.Interface) {
for {
event := <-watchHandler.ResultChan()
switch event.Type {
case watch.Modified:
_ = c.doUpdate(ctx, namespace)
}
}
}