added @Throws to CryptoService and changed the CustomConfigParser to be JAVA annotation (#6767)

This commit is contained in:
Alexey Kadyrov 2020-10-14 14:08:03 +01:00 committed by GitHub
parent 64dc73c7b1
commit 9d87c5dfb9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 44 additions and 19 deletions

View File

@ -0,0 +1,16 @@
package net.corda.nodeapi.internal.config;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
* This annotation can be used to provide ConfigParser for the class,
* the [parseAs] method will use the provided parser instead of data class constructs to parse the object.
*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface CustomConfigParser {
Class parser();
}

View File

@ -28,7 +28,6 @@ import kotlin.reflect.KClass
import kotlin.reflect.KProperty import kotlin.reflect.KProperty
import kotlin.reflect.KType import kotlin.reflect.KType
import kotlin.reflect.full.createInstance import kotlin.reflect.full.createInstance
import kotlin.reflect.full.findAnnotation
import kotlin.reflect.full.memberProperties import kotlin.reflect.full.memberProperties
import kotlin.reflect.full.primaryConstructor import kotlin.reflect.full.primaryConstructor
import kotlin.reflect.jvm.jvmErasure import kotlin.reflect.jvm.jvmErasure
@ -36,13 +35,6 @@ import kotlin.reflect.jvm.jvmErasure
@Target(AnnotationTarget.PROPERTY) @Target(AnnotationTarget.PROPERTY)
annotation class OldConfig(val value: String) annotation class OldConfig(val value: String)
/**
* This annotation can be used to provide ConfigParser for the class,
* the [parseAs] method will use the provided parser instead of data class constructs to parse the object.
*/
@Target(AnnotationTarget.CLASS)
annotation class CustomConfigParser(val parser: KClass<out ConfigParser<*>>)
interface ConfigParser<T> { interface ConfigParser<T> {
fun parse(config: Config): T fun parse(config: Config): T
} }
@ -77,7 +69,9 @@ fun <T : Any> Config.parseAs(
baseDirectory: Path? = null baseDirectory: Path? = null
): T { ): T {
// Use custom parser if provided, instead of treating the object as data class. // Use custom parser if provided, instead of treating the object as data class.
clazz.findAnnotation<CustomConfigParser>()?.let { return uncheckedCast(it.parser.createInstance().parse(this)) } @Suppress("UNCHECKED_CAST")
clazz.java.getAnnotation(CustomConfigParser::class.java)?.let {
return ((it.parser.createInstance() as ConfigParser<T>).parse(this)) }
require(clazz.isData) { require(clazz.isData) {
"Only Kotlin data classes or class annotated with CustomConfigParser can be parsed. Offending: ${clazz.qualifiedName}" "Only Kotlin data classes or class annotated with CustomConfigParser can be parsed. Offending: ${clazz.qualifiedName}"

View File

@ -1,6 +1,5 @@
package net.corda.nodeapi.internal.cryptoservice package net.corda.nodeapi.internal.cryptoservice
import net.corda.core.DoNotImplement
import net.corda.core.crypto.Crypto import net.corda.core.crypto.Crypto
import net.corda.core.crypto.SignatureScheme import net.corda.core.crypto.SignatureScheme
import net.corda.nodeapi.internal.crypto.X509Utilities import net.corda.nodeapi.internal.crypto.X509Utilities
@ -11,14 +10,15 @@ import java.security.PublicKey
/** /**
* Unlike [CryptoService] can only perform "read-only" operations but never create new key pairs. * Unlike [CryptoService] can only perform "read-only" operations but never create new key pairs.
*/ */
@DoNotImplement
interface SignOnlyCryptoService { interface SignOnlyCryptoService {
/** Check if this [CryptoService] has a private key entry for the input alias. */ /** Check if this [CryptoService] has a private key entry for the input alias. */
fun containsKey(alias: String): Boolean fun containsKey(alias: String): Boolean
/** /**
* Returns the [PublicKey] of the input alias or null if it doesn't exist. * Returns the [PublicKey] of the input alias or null if it doesn't exist.
* @throws CryptoServiceException for general cryptographic exceptions).
*/ */
@Throws(CryptoServiceException::class)
fun getPublicKey(alias: String): PublicKey? fun getPublicKey(alias: String): PublicKey?
/** /**
@ -27,6 +27,7 @@ interface SignOnlyCryptoService {
* The signAlgorithm if specified determines the signature scheme used for signing, if * The signAlgorithm if specified determines the signature scheme used for signing, if
* not specified then the signature scheme is based on the private key scheme. * not specified then the signature scheme is based on the private key scheme.
*/ */
@Throws(CryptoServiceException::class)
fun sign(alias: String, data: ByteArray, signAlgorithm: String? = null): ByteArray fun sign(alias: String, data: ByteArray, signAlgorithm: String? = null): ByteArray
/** /**
@ -48,7 +49,6 @@ interface SignOnlyCryptoService {
/** /**
* Fully-powered crypto service which can sign as well as create new key pairs. * Fully-powered crypto service which can sign as well as create new key pairs.
*/ */
@DoNotImplement
interface CryptoService : SignOnlyCryptoService { interface CryptoService : SignOnlyCryptoService {
/** /**
@ -58,10 +58,11 @@ interface CryptoService : SignOnlyCryptoService {
* because some schemes might not be standardised and thus an official OID might for this scheme not exist yet. * because some schemes might not be standardised and thus an official OID might for this scheme not exist yet.
* *
* Returns the [PublicKey] of the generated [KeyPair]. * Returns the [PublicKey] of the generated [KeyPair].
* @throws CryptoServiceException for general cryptographic exceptions).
*/ */
@Throws(CryptoServiceException::class)
fun generateKeyPair(alias: String, scheme: SignatureScheme): PublicKey fun generateKeyPair(alias: String, scheme: SignatureScheme): PublicKey
// ****************************************************** // ******************************************************
// ENTERPRISE ONLY CODE FOR WRAPPING KEYS API STARTS HERE // ENTERPRISE ONLY CODE FOR WRAPPING KEYS API STARTS HERE
// ****************************************************** // ******************************************************
@ -74,7 +75,9 @@ interface CryptoService : SignOnlyCryptoService {
* The default value is true. * The default value is true.
* *
* @throws IllegalArgumentException if a key already exists under this alias (and [failIfExists] is set to true]. * @throws IllegalArgumentException if a key already exists under this alias (and [failIfExists] is set to true].
* @throws CryptoServiceException for general cryptographic exceptions.
*/ */
@Throws(CryptoServiceException::class)
fun createWrappingKey(alias: String, failIfExists: Boolean = true) fun createWrappingKey(alias: String, failIfExists: Boolean = true)
/** /**
@ -90,7 +93,9 @@ interface CryptoService : SignOnlyCryptoService {
* @param childKeyScheme the parameters of the key pair to be generated. * @param childKeyScheme the parameters of the key pair to be generated.
* *
* @throws IllegalStateException if there is no master key existing under the alias specified ([masterKeyAlias]). * @throws IllegalStateException if there is no master key existing under the alias specified ([masterKeyAlias]).
* @throws CryptoServiceException for general cryptographic exceptions).
*/ */
@Throws(CryptoServiceException::class)
fun generateWrappedKeyPair(masterKeyAlias: String, childKeyScheme: SignatureScheme = defaultWrappingSignatureScheme()): Pair<PublicKey, WrappedPrivateKey> fun generateWrappedKeyPair(masterKeyAlias: String, childKeyScheme: SignatureScheme = defaultWrappingSignatureScheme()): Pair<PublicKey, WrappedPrivateKey>
/** /**
@ -101,7 +106,9 @@ interface CryptoService : SignOnlyCryptoService {
* @param payloadToSign the payload to be signed. * @param payloadToSign the payload to be signed.
* *
* @throws IllegalStateException if there is no master key existing under the alias specified ([masterKeyAlias]). * @throws IllegalStateException if there is no master key existing under the alias specified ([masterKeyAlias]).
* @throws CryptoServiceException for general cryptographic exceptions).
*/ */
@Throws(CryptoServiceException::class)
fun sign(masterKeyAlias: String, wrappedPrivateKey: WrappedPrivateKey, payloadToSign: ByteArray): ByteArray fun sign(masterKeyAlias: String, wrappedPrivateKey: WrappedPrivateKey, payloadToSign: ByteArray): ByteArray
/** /**
@ -115,7 +122,7 @@ interface CryptoService : SignOnlyCryptoService {
fun getWrappingMode(): WrappingMode? fun getWrappingMode(): WrappingMode?
/** /**
* Returns the [SignatureScheme] that should be used with this [CryptoService] when generating wrapped keys * Returns the [SignatureScheme] that should be used with this [CryptoService] when generating wrapped keys.
*/ */
fun defaultWrappingSignatureScheme(): SignatureScheme = Crypto.ECDSA_SECP256R1_SHA256 fun defaultWrappingSignatureScheme(): SignatureScheme = Crypto.ECDSA_SECP256R1_SHA256
@ -125,11 +132,11 @@ interface CryptoService : SignOnlyCryptoService {
} }
/** /**
* If an exception is deemed unrecoverable then it must be set with the flag ``isRecoverable=false`` * If an exception is deemed unrecoverable then it must be set with the flag ``isRecoverable=false``.
* *
* [ManagedCryptoService] will assume any [Throwable] which isn't a [CryptoServiceException] is recoverable and * [ManagedCryptoService] will assume any [Throwable] which isn't a [CryptoServiceException] is recoverable and
* will wrap it in a [CryptoServiceException] with ``isRecoverable=true``. * will wrap it in a [CryptoServiceException] with ``isRecoverable=true``.
*/ */
open class CryptoServiceException(message: String?, cause: Throwable? = null, val isRecoverable: Boolean = true) : Exception(message, cause) open class CryptoServiceException(message: String?, cause: Throwable? = null, val isRecoverable: Boolean = true) : Exception(message, cause)
enum class WrappingMode { enum class WrappingMode {
@ -140,7 +147,15 @@ enum class WrappingMode {
/** /**
* In normal mode, wrapped keys' material is encrypted, never exposed to the application and only accessible from inside the HSM. * In normal mode, wrapped keys' material is encrypted, never exposed to the application and only accessible from inside the HSM.
*/ */
WRAPPED WRAPPED,
/**
* Marker ENUM for when the HSM to be used supports a native key storage mode that supports LARGE (in the 100,000s at least) numbers of keys.
*/
NATIVE
} }
/**
* For the WRAPPED mode the keyMaterial will contain encrypted private key nad for the NATIVE that would be the alias
* which the CryptoService implementation assigned to the private key.
*/
class WrappedPrivateKey(val keyMaterial: ByteArray, val signatureScheme: SignatureScheme, val encodingVersion: Int? = null) class WrappedPrivateKey(val keyMaterial: ByteArray, val signatureScheme: SignatureScheme, val encodingVersion: Int? = null)

View File

@ -7,7 +7,7 @@ import net.corda.nodeapi.internal.config.CustomConfigParser
/** /**
* Data structure for controlling the way how Certificate Revocation Lists are handled. * Data structure for controlling the way how Certificate Revocation Lists are handled.
*/ */
@CustomConfigParser(RevocationConfigParser::class) @CustomConfigParser(parser = RevocationConfigParser::class)
interface RevocationConfig { interface RevocationConfig {
enum class Mode { enum class Mode {

View File

@ -266,7 +266,7 @@ class ConfigParsingTest {
data class TestObjects(val values: List<TestObject>) data class TestObjects(val values: List<TestObject>)
@CustomConfigParser(TestParser::class) @CustomConfigParser(parser = TestParser::class)
sealed class TestObject { sealed class TestObject {
data class Type1(val value: String) : TestObject() data class Type1(val value: String) : TestObject()
data class Type2(val value: String) : TestObject() data class Type2(val value: String) : TestObject()