mirror of
https://github.com/zitadel/zitadel.git
synced 2025-08-12 03:57:32 +00:00
chore: move the go code into a subfolder
This commit is contained in:
46
apps/api/internal/id/config.go
Normal file
46
apps/api/internal/id/config.go
Normal file
@@ -0,0 +1,46 @@
|
||||
package id
|
||||
|
||||
const (
|
||||
DefaultWebhookPath = "http://metadata.google.internal/computeMetadata/v1/instance/id"
|
||||
)
|
||||
|
||||
type Config struct {
|
||||
// Configuration for the identification of machines.
|
||||
Identification Identification
|
||||
}
|
||||
|
||||
type Identification struct {
|
||||
// Configuration for using private IP to identify a machine.
|
||||
PrivateIp PrivateIp
|
||||
// Configuration for using hostname to identify a machine.
|
||||
Hostname Hostname
|
||||
// Configuration for using a webhook to identify a machine.
|
||||
Webhook Webhook
|
||||
}
|
||||
|
||||
type PrivateIp struct {
|
||||
// Try to use private IP when identifying the machine uniquely
|
||||
Enabled bool
|
||||
}
|
||||
|
||||
type Hostname struct {
|
||||
// Try to use hostname when identifying the machine uniquely
|
||||
Enabled bool
|
||||
}
|
||||
|
||||
type Webhook struct {
|
||||
// Try to use webhook when identifying the machine uniquely
|
||||
Enabled bool
|
||||
// The URL of the metadata endpoint to query
|
||||
Url string
|
||||
// (Optional) A JSONPath expression for the data to extract from the response from the metadata endpoint
|
||||
JPath *string
|
||||
// (Optional) Headers to pass in the metadata request
|
||||
Headers *map[string]string
|
||||
}
|
||||
|
||||
func Configure(config *Config) {
|
||||
if config != nil {
|
||||
GeneratorConfig = config
|
||||
}
|
||||
}
|
3
apps/api/internal/id/gen_mock.go
Normal file
3
apps/api/internal/id/gen_mock.go
Normal file
@@ -0,0 +1,3 @@
|
||||
package id
|
||||
|
||||
//go:generate mockgen -package mock -destination ./mock/generator.mock.go github.com/zitadel/zitadel/internal/id Generator
|
5
apps/api/internal/id/id_generator.go
Normal file
5
apps/api/internal/id/id_generator.go
Normal file
@@ -0,0 +1,5 @@
|
||||
package id
|
||||
|
||||
type Generator interface {
|
||||
Next() (string, error)
|
||||
}
|
54
apps/api/internal/id/mock/generator.mock.go
Normal file
54
apps/api/internal/id/mock/generator.mock.go
Normal file
@@ -0,0 +1,54 @@
|
||||
// Code generated by MockGen. DO NOT EDIT.
|
||||
// Source: github.com/zitadel/zitadel/internal/id (interfaces: Generator)
|
||||
//
|
||||
// Generated by this command:
|
||||
//
|
||||
// mockgen -package mock -destination ./mock/generator.mock.go github.com/zitadel/zitadel/internal/id Generator
|
||||
//
|
||||
|
||||
// Package mock is a generated GoMock package.
|
||||
package mock
|
||||
|
||||
import (
|
||||
reflect "reflect"
|
||||
|
||||
gomock "go.uber.org/mock/gomock"
|
||||
)
|
||||
|
||||
// MockGenerator is a mock of Generator interface.
|
||||
type MockGenerator struct {
|
||||
ctrl *gomock.Controller
|
||||
recorder *MockGeneratorMockRecorder
|
||||
}
|
||||
|
||||
// MockGeneratorMockRecorder is the mock recorder for MockGenerator.
|
||||
type MockGeneratorMockRecorder struct {
|
||||
mock *MockGenerator
|
||||
}
|
||||
|
||||
// NewMockGenerator creates a new mock instance.
|
||||
func NewMockGenerator(ctrl *gomock.Controller) *MockGenerator {
|
||||
mock := &MockGenerator{ctrl: ctrl}
|
||||
mock.recorder = &MockGeneratorMockRecorder{mock}
|
||||
return mock
|
||||
}
|
||||
|
||||
// EXPECT returns an object that allows the caller to indicate expected use.
|
||||
func (m *MockGenerator) EXPECT() *MockGeneratorMockRecorder {
|
||||
return m.recorder
|
||||
}
|
||||
|
||||
// Next mocks base method.
|
||||
func (m *MockGenerator) Next() (string, error) {
|
||||
m.ctrl.T.Helper()
|
||||
ret := m.ctrl.Call(m, "Next")
|
||||
ret0, _ := ret[0].(string)
|
||||
ret1, _ := ret[1].(error)
|
||||
return ret0, ret1
|
||||
}
|
||||
|
||||
// Next indicates an expected call of Next.
|
||||
func (mr *MockGeneratorMockRecorder) Next() *gomock.Call {
|
||||
mr.mock.ctrl.T.Helper()
|
||||
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Next", reflect.TypeOf((*MockGenerator)(nil).Next))
|
||||
}
|
33
apps/api/internal/id/mock/generator.mock.impl.go
Normal file
33
apps/api/internal/id/mock/generator.mock.impl.go
Normal file
@@ -0,0 +1,33 @@
|
||||
package mock
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"go.uber.org/mock/gomock"
|
||||
)
|
||||
|
||||
func NewIDGenerator(t *testing.T) *MockGenerator {
|
||||
m := NewMockGenerator(gomock.NewController(t))
|
||||
m.EXPECT().Next().Return("1", nil)
|
||||
return m
|
||||
}
|
||||
|
||||
func NewIDGeneratorExpectIDs(t *testing.T, ids ...string) *MockGenerator {
|
||||
m := NewMockGenerator(gomock.NewController(t))
|
||||
for _, id := range ids {
|
||||
m.EXPECT().Next().Return(id, nil)
|
||||
}
|
||||
return m
|
||||
}
|
||||
|
||||
func ExpectID(t *testing.T, id string) *MockGenerator {
|
||||
m := NewMockGenerator(gomock.NewController(t))
|
||||
m.EXPECT().Next().Return(id, nil)
|
||||
return m
|
||||
}
|
||||
|
||||
func NewIDGeneratorExpectError(t *testing.T, err error) *MockGenerator {
|
||||
m := NewMockGenerator(gomock.NewController(t))
|
||||
m.EXPECT().Next().Return("", err)
|
||||
return m
|
||||
}
|
236
apps/api/internal/id/sonyflake.go
Normal file
236
apps/api/internal/id/sonyflake.go
Normal file
@@ -0,0 +1,236 @@
|
||||
package id
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"hash/fnv"
|
||||
"io"
|
||||
"net"
|
||||
"net/http"
|
||||
"os"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/drone/envsubst"
|
||||
"github.com/jarcoal/jpath"
|
||||
"github.com/sony/sonyflake"
|
||||
"github.com/zitadel/logging"
|
||||
)
|
||||
|
||||
type sonyflakeGenerator struct {
|
||||
*sonyflake.Sonyflake
|
||||
}
|
||||
|
||||
func (s *sonyflakeGenerator) Next() (string, error) {
|
||||
id, err := s.NextID()
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
return strconv.FormatUint(id, 10), nil
|
||||
}
|
||||
|
||||
var (
|
||||
GeneratorConfig *Config = nil
|
||||
sonyFlakeGenerator Generator = nil
|
||||
)
|
||||
|
||||
// SonyFlakeGenerator creates a new id generator
|
||||
// the function panics if the generator cannot be created
|
||||
func SonyFlakeGenerator() Generator {
|
||||
if sonyFlakeGenerator == nil {
|
||||
sfg := Generator(&sonyflakeGenerator{
|
||||
sonyflake.NewSonyflake(sonyflake.Settings{
|
||||
MachineID: machineID,
|
||||
StartTime: time.Date(2019, 4, 29, 0, 0, 0, 0, time.UTC),
|
||||
}),
|
||||
})
|
||||
|
||||
sonyFlakeGenerator = sfg
|
||||
}
|
||||
|
||||
return sonyFlakeGenerator
|
||||
}
|
||||
|
||||
// the following is a copy of sonyflake (https://github.com/sony/sonyflake/blob/master/sonyflake.go)
|
||||
// with the change of using the "POD-IP" if no private ip is found
|
||||
func privateIPv4() (net.IP, error) {
|
||||
as, err := net.InterfaceAddrs()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
for _, a := range as {
|
||||
ipnet, ok := a.(*net.IPNet)
|
||||
if !ok || ipnet.IP.IsLoopback() {
|
||||
continue
|
||||
}
|
||||
|
||||
ip := ipnet.IP.To4()
|
||||
if isPrivateIPv4(ip) {
|
||||
return ip, nil
|
||||
}
|
||||
}
|
||||
|
||||
// change: use "POD_IP"
|
||||
ip := net.ParseIP(os.Getenv("POD_IP"))
|
||||
if ip == nil {
|
||||
return nil, errors.New("no private ip address")
|
||||
}
|
||||
if ipV4 := ip.To4(); ipV4 != nil {
|
||||
return ipV4, nil
|
||||
}
|
||||
return nil, errors.New("no pod ipv4 address")
|
||||
}
|
||||
|
||||
func isPrivateIPv4(ip net.IP) bool {
|
||||
return ip != nil &&
|
||||
(ip[0] == 10 || ip[0] == 172 && (ip[1] >= 16 && ip[1] < 32) || ip[0] == 192 && ip[1] == 168)
|
||||
}
|
||||
|
||||
func MachineIdentificationMethod() string {
|
||||
if GeneratorConfig.Identification.PrivateIp.Enabled {
|
||||
return "Private Ip"
|
||||
}
|
||||
|
||||
if GeneratorConfig.Identification.Hostname.Enabled {
|
||||
return "Hostname"
|
||||
}
|
||||
|
||||
if GeneratorConfig.Identification.Webhook.Enabled {
|
||||
return "Webhook"
|
||||
}
|
||||
|
||||
return "No machine identification method has been enabled"
|
||||
}
|
||||
|
||||
func machineID() (uint16, error) {
|
||||
if GeneratorConfig == nil {
|
||||
logging.Panic("cannot create a unique id for the machine, generator has not been configured")
|
||||
}
|
||||
|
||||
errors := []string{}
|
||||
if GeneratorConfig.Identification.PrivateIp.Enabled {
|
||||
ip, err := lower16BitPrivateIP()
|
||||
if err == nil {
|
||||
return ip, nil
|
||||
}
|
||||
errors = append(errors, fmt.Sprintf("failed to get Private IP address %s", err))
|
||||
}
|
||||
|
||||
if GeneratorConfig.Identification.Hostname.Enabled {
|
||||
hn, err := hostname()
|
||||
if err == nil {
|
||||
return hn, nil
|
||||
}
|
||||
errors = append(errors, fmt.Sprintf("failed to get Hostname %s", err))
|
||||
}
|
||||
|
||||
if GeneratorConfig.Identification.Webhook.Enabled {
|
||||
cid, err := metadataWebhookID()
|
||||
if err == nil {
|
||||
return cid, nil
|
||||
}
|
||||
errors = append(errors, fmt.Sprintf("failed to query metadata webhook %s", err))
|
||||
}
|
||||
|
||||
if len(errors) == 0 {
|
||||
errors = append(errors, "No machine identification method enabled.")
|
||||
}
|
||||
|
||||
logging.WithFields("errors", strings.Join(errors, ", ")).Panic("none of the enabled methods for identifying the machine succeeded")
|
||||
// this return will never happen because of panic one line before
|
||||
return 0, nil
|
||||
}
|
||||
|
||||
func lower16BitPrivateIP() (uint16, error) {
|
||||
ip, err := privateIPv4()
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
return uint16(ip[2])<<8 + uint16(ip[3]), nil
|
||||
}
|
||||
|
||||
func hostname() (uint16, error) {
|
||||
host, err := os.Hostname()
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
h := fnv.New32()
|
||||
_, hashErr := h.Write([]byte(host))
|
||||
if hashErr != nil {
|
||||
return 0, hashErr
|
||||
}
|
||||
|
||||
return uint16(h.Sum32()), nil
|
||||
}
|
||||
|
||||
func metadataWebhookID() (uint16, error) {
|
||||
webhook := GeneratorConfig.Identification.Webhook
|
||||
url, err := envsubst.EvalEnv(webhook.Url)
|
||||
if err != nil {
|
||||
url = webhook.Url
|
||||
}
|
||||
|
||||
req, err := http.NewRequest(
|
||||
http.MethodGet,
|
||||
url,
|
||||
nil,
|
||||
)
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
if webhook.Headers != nil {
|
||||
for key, value := range *webhook.Headers {
|
||||
req.Header.Set(key, value)
|
||||
}
|
||||
}
|
||||
|
||||
resp, err := (&http.Client{}).Do(req)
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
defer resp.Body.Close()
|
||||
|
||||
if resp.StatusCode >= 400 && resp.StatusCode < 600 {
|
||||
return 0, fmt.Errorf("metadata endpoint returned an unsuccessful status code %d", resp.StatusCode)
|
||||
}
|
||||
body, err := io.ReadAll(resp.Body)
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
data, err := extractMetadataResponse(webhook.JPath, body)
|
||||
if err != nil {
|
||||
return 0, err
|
||||
}
|
||||
|
||||
h := fnv.New32()
|
||||
if _, err = h.Write(data); err != nil {
|
||||
return 0, err
|
||||
}
|
||||
return uint16(h.Sum32()), nil
|
||||
}
|
||||
|
||||
func extractMetadataResponse(path *string, data []byte) ([]byte, error) {
|
||||
if path != nil {
|
||||
jp, err := jpath.NewFromBytes(data)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
results := jp.Query(*path)
|
||||
if len(results) == 0 {
|
||||
return nil, fmt.Errorf("metadata endpoint response was successful, but JSONPath provided didn't match anything in the response: %s", string(data[:]))
|
||||
}
|
||||
|
||||
return json.Marshal(results)
|
||||
}
|
||||
|
||||
return data, nil
|
||||
}
|
Reference in New Issue
Block a user