1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
|
// Copyright 2023 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Package registryconfig is a transitioning stepping stone used by the
// keyset handle in cases where a configuration is not provided by the user,
// so it needs to resort to using the old global registry methods.
package registryconfig
import (
"fmt"
"reflect"
"sync"
"github.com/tink-crypto/tink-go/v2/core/registry"
"github.com/tink-crypto/tink-go/v2/internal/internalapi"
"github.com/tink-crypto/tink-go/v2/key"
tinkpb "github.com/tink-crypto/tink-go/v2/proto/tink_go_proto"
)
var (
primitiveConstructorsMu sync.RWMutex
primitiveConstructors = make(map[reflect.Type]primitiveConstructor)
)
type primitiveConstructor func(key key.Key) (any, error)
// RegistryConfig is an internal way for the keyset handle to access the
// old global Registry through the new Configuration interface.
type RegistryConfig struct{}
// PrimitiveFromKeyData constructs a primitive from a [key.Key] using the registry.
func (c *RegistryConfig) PrimitiveFromKeyData(keyData *tinkpb.KeyData, _ internalapi.Token) (any, error) {
return registry.PrimitiveFromKeyData(keyData)
}
// PrimitiveFromKey constructs a primitive from a [key.Key] using the registry.
func (c *RegistryConfig) PrimitiveFromKey(key key.Key, _ internalapi.Token) (any, error) {
if key == nil {
return nil, fmt.Errorf("key is nil")
}
constructor, found := primitiveConstructors[reflect.TypeOf(key)]
if !found {
return nil, fmt.Errorf("no constructor found for key %T", key)
}
return constructor(key)
}
// RegisterKeyManager registers a provided [registry.KeyManager] by forwarding
// it directly to the Registry.
func (c *RegistryConfig) RegisterKeyManager(km registry.KeyManager, _ internalapi.Token) error {
return registry.RegisterKeyManager(km)
}
// RegisterPrimitiveConstructor registers a function that constructs primitives
// from a given [key.Key] to the global registry.
func RegisterPrimitiveConstructor[K key.Key](constructor primitiveConstructor) error {
keyType := reflect.TypeFor[K]()
primitiveConstructorsMu.Lock()
defer primitiveConstructorsMu.Unlock()
if existingCreator, found := primitiveConstructors[keyType]; found && reflect.ValueOf(existingCreator).Pointer() != reflect.ValueOf(constructor).Pointer() {
return fmt.Errorf("a different constructor already registered for %v", keyType)
}
primitiveConstructors[keyType] = constructor
return nil
}
// UnregisterPrimitiveConstructor removes the primitive constructor for the
// given key type.
//
// This function is intended to be used in tests only.
func UnregisterPrimitiveConstructor[K key.Key]() {
primitiveConstructorsMu.Lock()
defer primitiveConstructorsMu.Unlock()
delete(primitiveConstructors, reflect.TypeFor[K]())
}
|