diff --git a/certify-core/src/main/java/io/mosip/certify/core/constants/ErrorConstants.java b/certify-core/src/main/java/io/mosip/certify/core/constants/ErrorConstants.java index 4c5957dd..3b178bf4 100644 --- a/certify-core/src/main/java/io/mosip/certify/core/constants/ErrorConstants.java +++ b/certify-core/src/main/java/io/mosip/certify/core/constants/ErrorConstants.java @@ -26,4 +26,8 @@ public class ErrorConstants { public static final String INVALID_TEMPLATE_ID = "template_with_id_not_found"; public static final String EMPTY_TEMPLATE_CONTENT = "empty_template_content"; public static final String EXPECTED_TEMPLATE_NOT_FOUND = "expected_template_not_found"; + public static final String UNSUPPORTED_IN_CURRENT_PLUGIN_MODE = "unsupported_in_current_plugin_mode"; + public static final String UNSUPPORTED_ALGORITHM = "unsupported_algorithm"; + public static final String INVALID_CERTIFICATE = "invalid_certificate"; + public static final String VERIFICATION_METHOD_GENERATION_FAILED = "verification_method_generation_failed"; } diff --git a/certify-core/src/main/java/io/mosip/certify/core/spi/VCIssuanceService.java b/certify-core/src/main/java/io/mosip/certify/core/spi/VCIssuanceService.java index 63aedaeb..c7fa2830 100644 --- a/certify-core/src/main/java/io/mosip/certify/core/spi/VCIssuanceService.java +++ b/certify-core/src/main/java/io/mosip/certify/core/spi/VCIssuanceService.java @@ -20,4 +20,6 @@ public interface VCIssuanceService { CredentialResponse getCredential(CredentialRequest credentialRequest); Map getCredentialIssuerMetadata(String version); + + Map getDIDDocument(); } diff --git a/certify-service/src/main/java/io/mosip/certify/controller/VCIssuanceController.java b/certify-service/src/main/java/io/mosip/certify/controller/VCIssuanceController.java index b530d87d..490d8f2d 100644 --- a/certify-service/src/main/java/io/mosip/certify/controller/VCIssuanceController.java +++ b/certify-service/src/main/java/io/mosip/certify/controller/VCIssuanceController.java @@ -80,6 +80,11 @@ public Map getMetadata( return vcIssuanceService.getCredentialIssuerMetadata(version); } + @GetMapping(value = "/.well-known/did.json") + public Map getDIDDocument() { + return vcIssuanceService.getDIDDocument(); + } + @ResponseBody @ExceptionHandler(InvalidNonceException.class) diff --git a/certify-service/src/main/java/io/mosip/certify/services/CertifyIssuanceServiceImpl.java b/certify-service/src/main/java/io/mosip/certify/services/CertifyIssuanceServiceImpl.java index 77f899e8..afc2fd25 100644 --- a/certify-service/src/main/java/io/mosip/certify/services/CertifyIssuanceServiceImpl.java +++ b/certify-service/src/main/java/io/mosip/certify/services/CertifyIssuanceServiceImpl.java @@ -34,7 +34,10 @@ import io.mosip.certify.proof.ProofValidator; import io.mosip.certify.proof.ProofValidatorFactory; import io.mosip.certify.utils.CredentialUtils; +import io.mosip.certify.utils.DIDDocumentUtil; import io.mosip.certify.vcsigners.VCSigner; +import io.mosip.kernel.keymanagerservice.dto.KeyPairGenerateResponseDto; +import io.mosip.kernel.keymanagerservice.service.KeymanagerService; import lombok.extern.slf4j.Slf4j; import org.apache.commons.lang3.StringUtils; import org.json.JSONObject; @@ -80,6 +83,9 @@ public class CertifyIssuanceServiceImpl implements VCIssuanceService { @Value("${mosip.certify.data-provider-plugin.issuer-uri}") private String issuerURI; + @Value("${mosip.certify.data-provider-plugin.issuer-public-key-uri}") + private String issuerPublicKeyURI; + @Value("${mosip.certify.data-provider-plugin.rendering-template-id:}") private String renderTemplateId; @@ -95,6 +101,11 @@ public class CertifyIssuanceServiceImpl implements VCIssuanceService { @Autowired private AuditPlugin auditWrapper; + @Autowired + private KeymanagerService keymanagerService; + + private Map didDocument; + @Override public CredentialResponse getCredential(CredentialRequest credentialRequest) { // 1. Credential Request validation @@ -155,6 +166,18 @@ public Map getCredentialIssuerMetadata(String version) { throw new InvalidRequestException(ErrorConstants.UNSUPPORTED_OPENID_VERSION); } + @Override + public Map getDIDDocument() { + if(didDocument != null) + return didDocument; + + KeyPairGenerateResponseDto keyPairGenerateResponseDto = keymanagerService.getCertificate(keyChooser.get(vcSignAlgorithm).getFirst(), Optional.of(keyChooser.get(vcSignAlgorithm).getLast())); + String certificateString = keyPairGenerateResponseDto.getCertificate(); + + didDocument = DIDDocumentUtil.generateDIDDocument(vcSignAlgorithm, certificateString, issuerURI, issuerPublicKeyURI); + return didDocument; + } + private Map convertLatestToVd11(LinkedHashMap vciMetadata) { // Create a list to hold the transformed credentials List> credentialsList = new ArrayList<>(); diff --git a/certify-service/src/main/java/io/mosip/certify/services/VCIssuanceServiceImpl.java b/certify-service/src/main/java/io/mosip/certify/services/VCIssuanceServiceImpl.java index 19b87f29..89c3c4e0 100644 --- a/certify-service/src/main/java/io/mosip/certify/services/VCIssuanceServiceImpl.java +++ b/certify-service/src/main/java/io/mosip/certify/services/VCIssuanceServiceImpl.java @@ -130,6 +130,11 @@ public Map getCredentialIssuerMetadata(String version) { throw new InvalidRequestException(ErrorConstants.UNSUPPORTED_OPENID_VERSION); } + @Override + public Map getDIDDocument() { + throw new InvalidRequestException(ErrorConstants.UNSUPPORTED_IN_CURRENT_PLUGIN_MODE); + } + private Map convertLatestToVd11(LinkedHashMap vciMetadata) { // Create a list to hold the transformed credentials List> credentialsList = new ArrayList<>(); diff --git a/certify-service/src/main/java/io/mosip/certify/utils/DIDDocumentUtil.java b/certify-service/src/main/java/io/mosip/certify/utils/DIDDocumentUtil.java new file mode 100644 index 00000000..2452ce5e --- /dev/null +++ b/certify-service/src/main/java/io/mosip/certify/utils/DIDDocumentUtil.java @@ -0,0 +1,111 @@ +package io.mosip.certify.utils; + +import java.io.ByteArrayInputStream; +import java.security.PublicKey; +import java.security.cert.CertificateFactory; +import java.security.cert.X509Certificate; +import java.security.interfaces.RSAPublicKey; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.HexFormat; +import java.util.Map; + +import org.bouncycastle.jcajce.provider.asymmetric.edec.BCEdDSAPublicKey; +import org.bouncycastle.jce.provider.BouncyCastleProvider; + +import com.nimbusds.jose.jwk.RSAKey; + +import io.ipfs.multibase.Multibase; +import io.mosip.certify.core.constants.ErrorConstants; +import io.mosip.certify.core.constants.SignatureAlg; +import io.mosip.certify.core.exception.CertifyException; +import lombok.extern.slf4j.Slf4j; + +@Slf4j +public class DIDDocumentUtil { + + private static final String MULTICODEC_PREFIX = "ed01"; + + public static Map generateDIDDocument(String vcSignAlgorithm, String certificateString, String issuerURI, String issuerPublicKeyURI) { + + HashMap didDocument = new HashMap(); + didDocument.put("@context", Collections.singletonList("https://www.w3.org/ns/did/v1")); + didDocument.put("alsoKnownAs", new ArrayList<>()); + didDocument.put("service", new ArrayList<>()); + didDocument.put("id", issuerURI); + didDocument.put("authentication", Collections.singletonList(issuerPublicKeyURI)); + didDocument.put("assertionMethod", Collections.singletonList(issuerPublicKeyURI)); + + Map verificationMethod = null; + PublicKey publicKey = loadPublicKeyFromCertificate(certificateString); + try { + switch (vcSignAlgorithm) { + case SignatureAlg.ED25519_SIGNATURE_SUITE_2018: + verificationMethod = generateEd25519VerificationMethod(publicKey, issuerURI, issuerPublicKeyURI); + break; + case SignatureAlg.ED25519_SIGNATURE_SUITE_2020: + verificationMethod = generateEd25519VerificationMethod(publicKey, issuerURI, issuerPublicKeyURI); + break; + case SignatureAlg.RSA_SIGNATURE_SUITE_2018: + verificationMethod = generateRSAVerificationMethod(publicKey, issuerURI, issuerPublicKeyURI); + break; + default: + log.error("Unsupported signature algorithm provided :" + vcSignAlgorithm); + throw new CertifyException(ErrorConstants.UNSUPPORTED_ALGORITHM); + } + } catch(CertifyException e) { + throw e; + } catch (Exception e) { + log.error("Exception occured while generating verification method for given certificate", e.getMessage(), e); + throw new CertifyException(ErrorConstants.VERIFICATION_METHOD_GENERATION_FAILED); + } + + didDocument.put("verificationMethod", Collections.singletonList(verificationMethod)); + return didDocument; + } + + private static PublicKey loadPublicKeyFromCertificate(String certificateString) { + try { + ByteArrayInputStream fis = new ByteArrayInputStream(certificateString.getBytes()); + CertificateFactory certFactory = CertificateFactory.getInstance("X.509", new BouncyCastleProvider()); + X509Certificate certificate = (X509Certificate) certFactory.generateCertificate(fis); + return certificate.getPublicKey(); + } catch (Exception e) { + log.error("Convertion from certificate to public key failed", e.getMessage(), e); + throw new CertifyException(ErrorConstants.INVALID_CERTIFICATE); + } + } + + private static Map generateEd25519VerificationMethod(PublicKey publicKey, String issuerURI, String issuerPublicKeyURI) throws Exception { + BCEdDSAPublicKey edKey = (BCEdDSAPublicKey) publicKey; + byte[] rawBytes = edKey.getPointEncoding(); + byte[] multicodecBytes = HexFormat.of().parseHex(MULTICODEC_PREFIX); + byte[] finalBytes = new byte[multicodecBytes.length + rawBytes.length]; + System.arraycopy(multicodecBytes, 0, finalBytes, 0, multicodecBytes.length); + System.arraycopy(rawBytes, 0, finalBytes, multicodecBytes.length, rawBytes.length); + String publicKeyMultibase = Multibase.encode(Multibase.Base.Base58BTC, finalBytes); + + Map verificationMethod = new HashMap<>(); + verificationMethod.put("id", issuerPublicKeyURI); + verificationMethod.put("type", "Ed25519VerificationKey2020"); + verificationMethod.put("@context", "https://w3id.org/security/suites/ed25519-2020/v1"); + verificationMethod.put("controller", issuerURI); + verificationMethod.put("publicKeyMultibase", publicKeyMultibase); + return verificationMethod; + } + + private static Map generateRSAVerificationMethod(PublicKey publicKey, String issuerURI, String issuerPublicKeyURI) throws Exception { + RSAKey rsaKey = new RSAKey.Builder((RSAPublicKey) publicKey).build(); + Map publicKeyJwk = rsaKey.toJSONObject(); + + Map verificationMethod = new HashMap<>(); + verificationMethod.put("id", issuerPublicKeyURI); + verificationMethod.put("type", "JsonWebKey2020"); + verificationMethod.put("@context", "https://w3id.org/security/suites/jws-2020/v1"); + verificationMethod.put("controller", issuerURI); + verificationMethod.put("publicKeyJwk", publicKeyJwk); + return verificationMethod; + } + +} diff --git a/certify-service/src/test/java/io/mosip/certify/TestVCIssuanceServiceImpl.java b/certify-service/src/test/java/io/mosip/certify/TestVCIssuanceServiceImpl.java index 71e44b0b..db421af6 100644 --- a/certify-service/src/test/java/io/mosip/certify/TestVCIssuanceServiceImpl.java +++ b/certify-service/src/test/java/io/mosip/certify/TestVCIssuanceServiceImpl.java @@ -1,7 +1,9 @@ package io.mosip.certify; +import io.mosip.certify.core.constants.ErrorConstants; import io.mosip.certify.core.dto.CredentialRequest; import io.mosip.certify.core.dto.CredentialResponse; +import io.mosip.certify.core.exception.InvalidRequestException; import io.mosip.certify.core.spi.VCIssuanceService; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; @@ -24,4 +26,9 @@ public CredentialResponse getCredential(CredentialRequest credentialReque public Map getCredentialIssuerMetadata(String version) { return Map.of(); } + + @Override + public Map getDIDDocument() { + throw new InvalidRequestException(ErrorConstants.UNSUPPORTED_IN_CURRENT_PLUGIN_MODE); + } } diff --git a/certify-service/src/test/java/io/mosip/certify/utils/DIDDocumentUtilTest.java b/certify-service/src/test/java/io/mosip/certify/utils/DIDDocumentUtilTest.java new file mode 100644 index 00000000..674869b1 --- /dev/null +++ b/certify-service/src/test/java/io/mosip/certify/utils/DIDDocumentUtilTest.java @@ -0,0 +1,113 @@ +package io.mosip.certify.utils; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertThrows; +import java.util.Collections; +import java.util.List; +import java.util.Map; + +import org.junit.jupiter.api.Test; + +import io.mosip.certify.core.constants.SignatureAlg; +import io.mosip.certify.core.exception.CertifyException; + +class DIDDocumentUtilTest { + + + @Test + @SuppressWarnings("unchecked") + void testGenerateDIDDocumentEd25519Signature2020() throws Exception { + String vcSignAlgorithm = SignatureAlg.ED25519_SIGNATURE_SUITE_2020; + String certificateString = "-----BEGIN CERTIFICATE-----\nMIIC2jCCAcKgAwIBAgIInbzaZeSXQqEwDQYJKoZIhvcNAQELBQAwgYsxCzAJBgNV\nBAYTAklOMQswCQYDVQQIDAJLQTESMBAGA1UEBwwJQkFOR0FMT1JFMQ4wDAYDVQQK\nDAVJSUlUQjEXMBUGA1UECwwORVhBTVBMRS1DRU5URVIxMjAwBgNVBAMMKXd3dy5l\neGFtcGxlLmNvbSAoQ0VSVElGWV9WQ19TSUdOX0VEMjU1MTkpMB4XDTI0MTIyOTA4\nNDY1OFoXDTI3MTIyOTA4NDY1OFowgYYxCzAJBgNVBAYTAklOMQswCQYDVQQIDAJL\nQTESMBAGA1UEBwwJQkFOR0FMT1JFMQ4wDAYDVQQKDAVJSUlUQjEXMBUGA1UECwwO\nRVhBTVBMRS1DRU5URVIxLTArBgNVBAMMJENFUlRJRllfVkNfU0lHTl9FRDI1NTE5\nLUVEMjU1MTlfU0lHTjAqMAUGAytlcAMhAOX8AiOEEHfyJRKJsjshaJps736mS4zS\ncZVcdUpZpEbxoz8wPTAMBgNVHRMBAf8EAjAAMB0GA1UdDgQWBBSVZaEpMbDVgrAy\nZP0ZlwMMXzhS9jAOBgNVHQ8BAf8EBAMCBSAwDQYJKoZIhvcNAQELBQADggEBAAJ4\nPZb+6A5Q5Z2X18B3PLNLs5It2UTu+qL8PhQyoVpEoq44Efl+10qaAiBp7l66sYcf\nsYVhREnJaBACqsEy5cFTZ7j+7Q0GhuepnkYTS9n8DwlOgZgPU0tBBwthbixwFyME\ne2VdtuhyuVnGK8+W6VWMg+lQGyQwPgrzAf6L81bADn+cW6tIVoYd4uuNfoXeM0pL\nTtKMGEyRVdx3Q+wcLEGZXCTYPkUgf+mq8kqf9dCDdDgblPU891msZpg0KGRkLD28\nPF7FPhK0Hq4DzwfhdpiQMe7W19FyH/IXRprJi8LKx4V9Y/rBAvR2loLR0PwVl+VB\nB55c6EluZ6hn9xuwr9w=\n-----END CERTIFICATE-----\n"; + String issuerURI = "did:example:123"; + String issuerPublicKeyURI = "did:example:123#key-0"; + + Map didDocument = DIDDocumentUtil.generateDIDDocument(vcSignAlgorithm, certificateString, issuerURI, issuerPublicKeyURI); + assertEquals(didDocument.get("@context"), Collections.singletonList("https://www.w3.org/ns/did/v1")); + assertEquals(issuerURI, didDocument.get("id")); + assertEquals(Collections.singletonList(issuerPublicKeyURI), didDocument.get("authentication")); + assertEquals(Collections.singletonList(issuerPublicKeyURI), didDocument.get("assertionMethod")); + + Map verificationMethod = ((List>)didDocument.get("verificationMethod")).get(0); + assertEquals(verificationMethod.get("publicKeyMultibase"), "z6Mkuw2HXTbK7fXoVbiuriHdm3NDDcVRYWxRymfzdTE6ZWgQ"); + assertEquals(verificationMethod.get("controller"), issuerURI); + assertEquals(verificationMethod.get("id"), issuerPublicKeyURI); + assertEquals(verificationMethod.get("type"), "Ed25519VerificationKey2020"); + assertEquals(verificationMethod.get("@context"), "https://w3id.org/security/suites/ed25519-2020/v1"); + } + + @Test + @SuppressWarnings("unchecked") + void testGenerateDIDDocumentEd25519Signature2018() throws Exception { + String vcSignAlgorithm = SignatureAlg.ED25519_SIGNATURE_SUITE_2020; + String certificateString = "-----BEGIN CERTIFICATE-----\nMIIC2jCCAcKgAwIBAgIInbzaZeSXQqEwDQYJKoZIhvcNAQELBQAwgYsxCzAJBgNV\nBAYTAklOMQswCQYDVQQIDAJLQTESMBAGA1UEBwwJQkFOR0FMT1JFMQ4wDAYDVQQK\nDAVJSUlUQjEXMBUGA1UECwwORVhBTVBMRS1DRU5URVIxMjAwBgNVBAMMKXd3dy5l\neGFtcGxlLmNvbSAoQ0VSVElGWV9WQ19TSUdOX0VEMjU1MTkpMB4XDTI0MTIyOTA4\nNDY1OFoXDTI3MTIyOTA4NDY1OFowgYYxCzAJBgNVBAYTAklOMQswCQYDVQQIDAJL\nQTESMBAGA1UEBwwJQkFOR0FMT1JFMQ4wDAYDVQQKDAVJSUlUQjEXMBUGA1UECwwO\nRVhBTVBMRS1DRU5URVIxLTArBgNVBAMMJENFUlRJRllfVkNfU0lHTl9FRDI1NTE5\nLUVEMjU1MTlfU0lHTjAqMAUGAytlcAMhAOX8AiOEEHfyJRKJsjshaJps736mS4zS\ncZVcdUpZpEbxoz8wPTAMBgNVHRMBAf8EAjAAMB0GA1UdDgQWBBSVZaEpMbDVgrAy\nZP0ZlwMMXzhS9jAOBgNVHQ8BAf8EBAMCBSAwDQYJKoZIhvcNAQELBQADggEBAAJ4\nPZb+6A5Q5Z2X18B3PLNLs5It2UTu+qL8PhQyoVpEoq44Efl+10qaAiBp7l66sYcf\nsYVhREnJaBACqsEy5cFTZ7j+7Q0GhuepnkYTS9n8DwlOgZgPU0tBBwthbixwFyME\ne2VdtuhyuVnGK8+W6VWMg+lQGyQwPgrzAf6L81bADn+cW6tIVoYd4uuNfoXeM0pL\nTtKMGEyRVdx3Q+wcLEGZXCTYPkUgf+mq8kqf9dCDdDgblPU891msZpg0KGRkLD28\nPF7FPhK0Hq4DzwfhdpiQMe7W19FyH/IXRprJi8LKx4V9Y/rBAvR2loLR0PwVl+VB\nB55c6EluZ6hn9xuwr9w=\n-----END CERTIFICATE-----\n"; + String issuerURI = "did:example:123"; + String issuerPublicKeyURI = "did:example:123#key-0"; + + Map didDocument = DIDDocumentUtil.generateDIDDocument(vcSignAlgorithm, certificateString, issuerURI, issuerPublicKeyURI); + assertEquals(didDocument.get("@context"), Collections.singletonList("https://www.w3.org/ns/did/v1")); + assertEquals(issuerURI, didDocument.get("id")); + assertEquals(Collections.singletonList(issuerPublicKeyURI), didDocument.get("authentication")); + assertEquals(Collections.singletonList(issuerPublicKeyURI), didDocument.get("assertionMethod")); + + Map verificationMethod = ((List>)didDocument.get("verificationMethod")).get(0); + assertEquals(verificationMethod.get("publicKeyMultibase"), "z6Mkuw2HXTbK7fXoVbiuriHdm3NDDcVRYWxRymfzdTE6ZWgQ"); + assertEquals(verificationMethod.get("controller"), issuerURI); + assertEquals(verificationMethod.get("id"), issuerPublicKeyURI); + assertEquals(verificationMethod.get("type"), "Ed25519VerificationKey2020"); + assertEquals(verificationMethod.get("@context"), "https://w3id.org/security/suites/ed25519-2020/v1"); + } + + @Test + @SuppressWarnings("unchecked") + void testGenerateDIDDocumentRSASignature2018() throws Exception { + String vcSignAlgorithm = SignatureAlg.RSA_SIGNATURE_SUITE_2018; + String certificateString = "-----BEGIN CERTIFICATE-----\nMIIDxzCCAq+gAwIBAgIIgusG+rdZJWgwDQYJKoZIhvcNAQELBQAweDELMAkGA1UE\nBhMCSU4xCzAJBgNVBAgMAktBMRIwEAYDVQQHDAlCQU5HQUxPUkUxDjAMBgNVBAoM\nBUlJSVRCMRcwFQYDVQQLDA5FWEFNUExFLUNFTlRFUjEfMB0GA1UEAwwWd3d3LmV4\nYW1wbGUuY29tIChST09UKTAeFw0yNDEyMjkxMDQ4NDRaFw0yNzEyMjkxMDQ4NDRa\nMIGHMQswCQYDVQQGEwJJTjELMAkGA1UECAwCS0ExEjAQBgNVBAcMCUJBTkdBTE9S\nRTEOMAwGA1UECgwFSUlJVEIxFzAVBgNVBAsMDkVYQU1QTEUtQ0VOVEVSMS4wLAYD\nVQQDDCV3d3cuZXhhbXBsZS5jb20gKENFUlRJRllfVkNfU0lHTl9SU0EpMIIBIjAN\nBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAlkO3CPWJ6Jqu9hzm4Eew7EJSbYCX\n7YGBxYAjRHcLuVgsttyRWUZ3DiRYEoN7bG/jCh7E0Gvv4M5ux4VSw3RJlM+9Tfje\nDUkHdZQ0g5A/r69uyy7+zE8MIM2fXcgwEgIZabm/Zb6+T/K6mSsdPQAHnBe1zXoq\ngTuyTT6pVsHbR0+5ULkhN3BuJyhJ7zw8vC1aiFYA2b05nU7H1Rn+axes8+v80mQS\nGR9iJTrGeYtvz8a+gRhvXmK+h8nhUAJaPHJBacCRMErKvgddWkWBtknJZQmnX0RN\n2IC5+egbE8thCVg8BGBcxOoUBHjHYmus0CZNbTMJQIObL62p7caJHnYtHwIDAQAB\no0UwQzASBgNVHRMBAf8ECDAGAQH/AgEBMB0GA1UdDgQWBBSOi5/6I4vvp8eshKNs\nSwr/BtWM/zAOBgNVHQ8BAf8EBAMCAoQwDQYJKoZIhvcNAQELBQADggEBAKHiZu+1\nPjKqvlesbAj4QJkQlpdstz0PgEOnT6+flpcnmyMJj2QvWQbfX8niVWGMIc0HnO+H\ntzc/2oKmO9eQpmdnL4DN7NtuXxbTwTzsGDI934jRZGqHmeCh90j+T7QqSbk+GanC\nOMGFth7aV9j5cDSr7gCIom6N0TEUw/5a3O1+vJCwtQtN29H/+ksro+RYyN4/nbrR\ngix5XRR9VTcsLbM8J8dOxqZxsP+Bgebqp+fqv8QEea4cVYtStEMY6/4M6kKWyL7Q\nsmgwsJ5Vr5w/Y1hOIKaQe9WwWm/T8+byElVgZ/vT5tCYhLxHyBa1vfTgq1FQe5gb\nc6CDSimUO4tcosI=\n-----END CERTIFICATE-----\n"; + String issuerURI = "did:example:123"; + String issuerPublicKeyURI = "did:example:123#key-0"; + + Map didDocument = DIDDocumentUtil.generateDIDDocument(vcSignAlgorithm, certificateString, issuerURI, issuerPublicKeyURI); + assertEquals(didDocument.get("@context"), Collections.singletonList("https://www.w3.org/ns/did/v1")); + assertEquals(issuerURI, didDocument.get("id")); + assertEquals(Collections.singletonList(issuerPublicKeyURI), didDocument.get("authentication")); + assertEquals(Collections.singletonList(issuerPublicKeyURI), didDocument.get("assertionMethod")); + + Map verificationMethod = ((List>)didDocument.get("verificationMethod")).get(0); + assertEquals(((Map)verificationMethod.get("publicKeyJwk")).get("kty"), "RSA"); + assertEquals(((Map)verificationMethod.get("publicKeyJwk")).get("e"), "AQAB"); + assertEquals(((Map)verificationMethod.get("publicKeyJwk")).get("n"), "lkO3CPWJ6Jqu9hzm4Eew7EJSbYCX7YGBxYAjRHcLuVgsttyRWUZ3DiRYEoN7bG_jCh7E0Gvv4M5ux4VSw3RJlM-9TfjeDUkHdZQ0g5A_r69uyy7-zE8MIM2fXcgwEgIZabm_Zb6-T_K6mSsdPQAHnBe1zXoqgTuyTT6pVsHbR0-5ULkhN3BuJyhJ7zw8vC1aiFYA2b05nU7H1Rn-axes8-v80mQSGR9iJTrGeYtvz8a-gRhvXmK-h8nhUAJaPHJBacCRMErKvgddWkWBtknJZQmnX0RN2IC5-egbE8thCVg8BGBcxOoUBHjHYmus0CZNbTMJQIObL62p7caJHnYtHw"); + assertEquals(verificationMethod.get("controller"), issuerURI); + assertEquals(verificationMethod.get("id"), issuerPublicKeyURI); + assertEquals(verificationMethod.get("type"), "JsonWebKey2020"); + assertEquals(verificationMethod.get("@context"), "https://w3id.org/security/suites/jws-2020/v1"); + } + + @Test + void testGenerateDIDDocumentUnsupportedAlgorithm() { + String vcSignAlgorithm = "UnsupportedAlgorithm"; + String certificateString = "-----BEGIN CERTIFICATE-----\nMIIC2jCCAcKgAwIBAgIInbzaZeSXQqEwDQYJKoZIhvcNAQELBQAwgYsxCzAJBgNV\nBAYTAklOMQswCQYDVQQIDAJLQTESMBAGA1UEBwwJQkFOR0FMT1JFMQ4wDAYDVQQK\nDAVJSUlUQjEXMBUGA1UECwwORVhBTVBMRS1DRU5URVIxMjAwBgNVBAMMKXd3dy5l\neGFtcGxlLmNvbSAoQ0VSVElGWV9WQ19TSUdOX0VEMjU1MTkpMB4XDTI0MTIyOTA4\nNDY1OFoXDTI3MTIyOTA4NDY1OFowgYYxCzAJBgNVBAYTAklOMQswCQYDVQQIDAJL\nQTESMBAGA1UEBwwJQkFOR0FMT1JFMQ4wDAYDVQQKDAVJSUlUQjEXMBUGA1UECwwO\nRVhBTVBMRS1DRU5URVIxLTArBgNVBAMMJENFUlRJRllfVkNfU0lHTl9FRDI1NTE5\nLUVEMjU1MTlfU0lHTjAqMAUGAytlcAMhAOX8AiOEEHfyJRKJsjshaJps736mS4zS\ncZVcdUpZpEbxoz8wPTAMBgNVHRMBAf8EAjAAMB0GA1UdDgQWBBSVZaEpMbDVgrAy\nZP0ZlwMMXzhS9jAOBgNVHQ8BAf8EBAMCBSAwDQYJKoZIhvcNAQELBQADggEBAAJ4\nPZb+6A5Q5Z2X18B3PLNLs5It2UTu+qL8PhQyoVpEoq44Efl+10qaAiBp7l66sYcf\nsYVhREnJaBACqsEy5cFTZ7j+7Q0GhuepnkYTS9n8DwlOgZgPU0tBBwthbixwFyME\ne2VdtuhyuVnGK8+W6VWMg+lQGyQwPgrzAf6L81bADn+cW6tIVoYd4uuNfoXeM0pL\nTtKMGEyRVdx3Q+wcLEGZXCTYPkUgf+mq8kqf9dCDdDgblPU891msZpg0KGRkLD28\nPF7FPhK0Hq4DzwfhdpiQMe7W19FyH/IXRprJi8LKx4V9Y/rBAvR2loLR0PwVl+VB\nB55c6EluZ6hn9xuwr9w=\n-----END CERTIFICATE-----\n"; + String issuerURI = "did:example:123"; + String issuerPublicKeyURI = "did:example:123#key-0"; + + CertifyException exception = assertThrows(CertifyException.class, () -> { + DIDDocumentUtil.generateDIDDocument(vcSignAlgorithm, certificateString, issuerURI, issuerPublicKeyURI); + }); + + assertEquals("unsupported_algorithm", exception.getErrorCode()); + } + + @Test + void testGenerateDIDDocumentWithInvalidCertificateString() { + String invalidCertificateString = "INVALID_CERTIFICATE"; + String vcSignAlgorithm = SignatureAlg.RSA_SIGNATURE_SUITE_2018; + String issuerURI = "did:web:example.com"; + String issuerPublicKeyURI = "did:web:example.com#key-0"; + + CertifyException exception = assertThrows(CertifyException.class, () -> { + DIDDocumentUtil.generateDIDDocument(vcSignAlgorithm, invalidCertificateString, issuerURI, issuerPublicKeyURI); + }); + + assertEquals("invalid_certificate", exception.getErrorCode()); + } + +} diff --git a/certify-service/src/test/java/io/mosip/certify/vcformatters/VelocityTemplatingEngineImplTest.java b/certify-service/src/test/java/io/mosip/certify/vcformatters/VelocityTemplatingEngineImplTest.java index cb17f57a..05cb6ed1 100644 --- a/certify-service/src/test/java/io/mosip/certify/vcformatters/VelocityTemplatingEngineImplTest.java +++ b/certify-service/src/test/java/io/mosip/certify/vcformatters/VelocityTemplatingEngineImplTest.java @@ -177,7 +177,6 @@ public void testTemplating_localOnly() { Map templateMap = Map.of("templateName", "MockVerifiableCredential,VerifiableCredential:https://vharsh.github.io/DID/mock-context.json,https://www.w3.org/2018/credentials/v1", "issuerURI", "https://example.com/fake-issuer"); String actualJSON = formatter.format(ret, templateMap); - System.out.println(actualJSON); try { JSONObject j = new JSONObject(actualJSON); } catch (JSONException e) { diff --git a/docker-compose/docker-compose-injistack/config/certify-csvdp-farmer.properties b/docker-compose/docker-compose-injistack/config/certify-csvdp-farmer.properties index 7fc539db..60e4fe7d 100644 --- a/docker-compose/docker-compose-injistack/config/certify-csvdp-farmer.properties +++ b/docker-compose/docker-compose-injistack/config/certify-csvdp-farmer.properties @@ -24,42 +24,6 @@ mosip.certify.mock.data-provider.csv.data-columns=id,fullName,mobileNumber,dateO mosip.certify.mock.data-provider.csv-registry-uri=/home/mosip/config/farmer_identity_data.csv mosip.certify.data-provider-plugin.rendering-template-id= mosip.certify.key-values={\ - 'vd12' : {\ - 'credential_issuer': '${mosip.certify.identifier}',\ - 'authorization_servers': {'${mosip.certify.authorization.url}'},\ - 'credential_endpoint': '${mosipbox.public.url}${server.servlet.path}/issuance/vd12/credential',\ - 'display': {{'name': 'Insurance', 'locale': 'en'}},\ - 'credentials_supported' : {\ - 'FarmerProfileCredential' : {\ - 'format': 'ldp_vc',\ - 'scope' : 'farmer_vc_ldp',\ - 'cryptographic_binding_methods_supported': {'did:jwk'},\ - 'cryptographic_suites_supported': {'Ed25519Signature2020'},\ - 'proof_types_supported': {'jwt'},\ - 'credential_definition': {\ - 'type': {'VerifiableCredential','FarmerCredential'},\ - 'credentialSubject': {\ - 'fullName': {'display': {{'name': 'Full Name','locale': 'en'}}}, \ - 'mobileNumber': {'display': {{'name': 'Phone Number','locale': 'en'}}},\ - 'dateOfBirth': {'display': {{'name': 'Date of Birth','locale': 'en'}}},\ - 'gender': {'display': {{'name': 'Gender','locale': 'en'}}},\ - 'state': {'display': {{'name': 'State','locale': 'en'}}},\ - 'district': {'display': {{'name': 'District','locale': 'en'}}},\ - 'villageOrTown': {'display': {{'name': 'Village or Town','locale': 'en'}}},\ - 'postalCode': {'display': {{'name': 'Postal Code','locale': 'en'}}},\ - 'landArea': {'display': {{'name': 'Land Area','locale': 'en'}}},\ - 'landOwnershipType': {'display': {{'name': 'Land Ownership Type','locale': 'en'}}},\ - 'primaryCropType': {'display': {{'name': 'Primary Crop Name','locale': 'en'}}},\ - 'secondaryCropType': {'display': {{'name': 'Secondary Crop type','locale': 'en'}}},\ - 'farmerID': {'display': {{'name': 'Farmer ID','locale': 'en'}}}\ - }},\ - 'display': {{'name': 'Farmer Profile Verifiable Credential', \ - 'locale': 'en', \ - 'logo': {'url': 'https://sunbird.org/images/sunbird-logo-new.png','alt_text': 'a square logo of a Sunbird'},\ - 'background_color': '#FDFAF9',\ - 'text_color': '#7C4616'}},\ - 'order' : {'farmerID','fullName','mobileNumber','dateOfBirth','gender','state','district','villageOrTown','postalCode','landArea','landOwnershipType','primaryCropType','secondaryCropType'}\ - }}},\ 'latest' : {\ 'credential_issuer': '${mosip.certify.identifier}', \ 'authorization_servers': {'${mosip.certify.authorization.url}'}, \ diff --git a/docker-compose/docker-compose-injistack/config/farmer_identity_data.csv b/docker-compose/docker-compose-injistack/config/farmer_identity_data.csv index 9a3f2010..8909596c 100644 --- a/docker-compose/docker-compose-injistack/config/farmer_identity_data.csv +++ b/docker-compose/docker-compose-injistack/config/farmer_identity_data.csv @@ -1,6 +1,6 @@ id,fullName,mobileNumber,dateOfBirth,gender,state,district,villageOrTown,postalCode,landArea,landOwnershipType,primaryCropType,secondaryCropType,face,farmerID -1234,John Doe,9876543210,25-05-1990,Male,Karnataka,Bangalore,Koramangala,453000,3 hectares,Owner,Maize,Rice,"",987654321 -876543210,Mary Smith,8765432109,12-11-1985,Female,Uttar Pradesh,Lucknow,Gomti Nagar,226010,2.5 acres,Tenant,Rice,Wheat,"a",123456789 -7550166913,Raj Patel,7550166914,24-01-1998,Male,Karnataka,Bangalore,Koramangala,560068,5 acres,Self-owned,Cotton,Barley,"",4567538771 -654321098,Sarah Johnson,6543210987,15-02-1988,Male,Tamil Nadu,Madurai,Melur,625106,1 hectare,Owner,Banana,Mango,"a",4567538772 -543210987,Arun Kumar,5432109876,22-09-1995,Male,Karnataka,Bangalore,Koramangala,560034,2 acres,Tenant,Tomato,Peppers,"a",345678910 \ No newline at end of file +1234,John Doe,9876543210,25-05-1990,Male,Karnataka,Bangalore,Koramangala,453000,3 hectares,Owner,Maize,Rice,"",987654321 +7550166913,Mary Smith,8765432109,12-11-1985,Female,Uttar Pradesh,Lucknow,Gomti Nagar,226010,2.5 acres,Tenant,Rice,Wheat,"",123456789 +876543210,Raj Patel,7550166914,24-01-1998,Male,Karnataka,Bangalore,Koramangala,560068,5 acres,Self-owned,Cotton,Barley,"",4567538771 +654321098,Sarah Johnson,6543210987,15-02-1988,Male,Tamil Nadu,Madurai,Melur,625106,1 hectare,Owner,Banana,Mango,"",4567538772 +543210987,Arun Kumar,5432109876,22-09-1995,Male,Karnataka,Bangalore,Koramangala,560034,2 acres,Tenant,Tomato,Peppers,"",345678910 \ No newline at end of file diff --git a/docker-compose/docker-compose-injistack/config/mimoto-default.properties b/docker-compose/docker-compose-injistack/config/mimoto-default.properties index deffd6d6..d1f0c7a9 100644 --- a/docker-compose/docker-compose-injistack/config/mimoto-default.properties +++ b/docker-compose/docker-compose-injistack/config/mimoto-default.properties @@ -41,12 +41,12 @@ mosip.inji.minStorageRequired=2 # START bootstrap.properties -spring.cloud.config.uri=http://nginx/ +spring.cloud.config.uri=http://inji-web:3004/ spring.cloud.config.name=mimoto,inji spring.application.name=mimoto #config.server.file.storage.uri=https://raw.githubusercontent.com/mosip/mosip-config/collab1/ -config.server.file.storage.uri=http://nginx/ +config.server.file.storage.uri=http://inji-web:3004/ management.endpoint.health.show-details=always management.endpoints.web.exposure.include=info,health,refresh diff --git a/docker-compose/docker-compose-injistack/config/mimoto-issuers-config.json b/docker-compose/docker-compose-injistack/config/mimoto-issuers-config.json index 2619ab97..28e6a89a 100644 --- a/docker-compose/docker-compose-injistack/config/mimoto-issuers-config.json +++ b/docker-compose/docker-compose-injistack/config/mimoto-issuers-config.json @@ -1,7 +1,8 @@ { "issuers": [ { - "credential_issuer": "Mock", + "credential_issuer": "Farmer", + "issuer_id": "Farmer", "protocol": "OpenId4VCI", "display": [ { @@ -10,22 +11,23 @@ "url": "https://api.collab.mosip.net/inji/mosip-logo.png", "alt_text": "mosip-logo" }, - "title": "Mock Identity", - "description": "Download Mock Identity Credential", + "title": "Farmers Credentials", + "description": "Download Farmers Credentials", "language": "en" } ], - "client_id": "mpartner-mock-testing", + "client_id": "wallet-demo", "redirect_uri": "io.mosip.residentapp.inji://oauthredirect", - "token_endpoint": "http://localhost:8099/v1/mimoto/get-token/Mock", + "token_endpoint": "http://localhost:8099/v1/mimoto/get-token/Farmer", "authorization_audience": "https://esignet-mock.dev1.mosip.net/v1/esignet/oauth/v2/token", "proxy_token_endpoint": "https://esignet-mock.dev1.mosip.net/v1/esignet/oauth/v2/token", - "client_alias": "mpartner-mock-testing", + "client_alias": "wallet-demo-client", "qr_code_type": "EmbeddedVC", "enabled": "true", "wellknown_endpoint": "http://certify:8090/v1/certify/issuance/.well-known/openid-credential-issuer" }, { + "issuer_id": "MockMdl", "credential_issuer": "MockMdl", "protocol": "OpenId4VCI", "display": [ @@ -49,6 +51,6 @@ "qr_code_type": "EmbeddedVC", "enabled": "true", "wellknown_endpoint": "http://certify:8090/v1/certify/issuance/.well-known/openid-credential-issuer" - } + } ] } diff --git a/docker-compose/docker-compose-injistack/docker-compose.yaml b/docker-compose/docker-compose-injistack/docker-compose.yaml index c14ee76c..53c348e3 100644 --- a/docker-compose/docker-compose-injistack/docker-compose.yaml +++ b/docker-compose/docker-compose-injistack/docker-compose.yaml @@ -25,10 +25,12 @@ services: - SPRING_CONFIG_LOCATION=/home/mosip/config/ - enable_certify_artifactory=false - download_hsm_client=false + - mosipbox_public_url=http://certify:8090 volumes: - ./config/certify-default.properties:/home/mosip/config/certify-default.properties - ./config/certify-csvdp-farmer.properties:/home/mosip/config/certify-csvdp-farmer.properties - ./config/certify-mock-mdl.properties:/home/mosip/config/certify-mock-mdl.properties + - ./config/farmer_identity_data.csv:/home/mosip/config/farmer_identity_data.csv - ./data/CERTIFY_PKCS12:/home/mosip/CERTIFY_PKCS12 - ./loader_path/certify/:/home/mosip/additional_jars/ # modify the below file to change the identity fields in the VC @@ -38,24 +40,9 @@ services: depends_on: - database - nginx: - container_name: nginx - image: nginx:alpine - ports: - - '80:80' - volumes: - - ./config/mimoto-issuers-config.json:/config/server/mimoto-issuers-config.json - - ./config/mimoto-trusted-verifiers.json:/config/server/mimoto-trusted-verifiers.json - - ./config/credential-template.html:/config/server/credential-template.html - - ./nginx.conf:/etc/nginx/nginx.conf - depends_on: - - mimoto-service - networks: - - network - mimoto-service: container_name: 'Mimoto-Service' - image: mosipid/mimoto:0.14.0 + image: mosipqa/mimoto:release-0.15.x user: root ports: - '8099:8099' @@ -74,14 +61,20 @@ services: inji-web: container_name: 'inji-web' - image: mosipid/inji-web:0.10.0 + image: mosipqa/inji-web:release-0.11.x ports: - '3001:3004' environment: - - MIMOTO_HOST=http://localhost:8080/v1/mimoto # Pointing to the Nginx service - DEFAULT_LANG=en + - MIMOTO_HOST=http://localhost:3001/v1/mimoto + volumes: + - ./config/mimoto-default.properties:/home/mosip/mimoto-default.properties + - ./config/mimoto-issuers-config.json:/home/mosip/mimoto-issuers-config.json + - ./config/mimoto-trusted-verifiers.json:/home/mosip/mimoto-trusted-verifiers.json + - ./config/credential-template.html:/home/mosip/credential-template.html + - ./nginx.conf:/etc/nginx/conf.d/default.conf depends_on: - - nginx + - mimoto-service networks: - network diff --git a/docker-compose/docker-compose-injistack/nginx.conf b/docker-compose/docker-compose-injistack/nginx.conf index a11042b8..fd5d0e04 100644 --- a/docker-compose/docker-compose-injistack/nginx.conf +++ b/docker-compose/docker-compose-injistack/nginx.conf @@ -1,34 +1,48 @@ -events { } +server { + listen 3004; + # Default location for normal static files + location / { + root /usr/share/nginx/html; + index index.html index.htm; + try_files $uri $uri/ /index.html; + } -http { - server { - listen 80; + # Serve files from /home/mosip with autoindex enabled + location ~* \.json$ { + root /home/mosip; + autoindex on; + } - # Serve static files, if any - location / { - root /config/server; - autoindex on; - } + # Serve files from /home/mosip with autoindex enabled + location /credential-template.html { + root /home/mosip; + autoindex on; + } - # Proxy API requests to mimoto-service - location /v1/mimoto/ { - proxy_pass http://mimoto-service:8099/v1/mimoto/; - proxy_set_header Host $host; - proxy_set_header X-Real-IP $remote_addr; - proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; - proxy_set_header X-Forwarded-Proto $scheme; - proxy_http_version 1.1; - proxy_set_header Connection ""; + # Proxy API requests to mimoto-service + location /v1/mimoto/ { + proxy_pass http://mimoto-service:8099/v1/mimoto/; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + proxy_http_version 1.1; + proxy_set_header Connection ""; - # Add CORS headers - add_header 'Access-Control-Allow-Origin' '*' always; - add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS' always; - add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Cache-Control' always; + # Add CORS headers + add_header 'Access-Control-Allow-Origin' '*' always; + add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS' always; + add_header 'Access-Control-Allow-Headers' 'Content-Type, Authorization, Cache-Control' always; - # Handle OPTIONS requests (for pre-flight checks) - if ($request_method = 'OPTIONS') { - return 204; - } + # Handle OPTIONS requests (for pre-flight checks) + if ($request_method = 'OPTIONS') { + return 204; } } -} + + error_page 500 502 503 504 /50x.html; + + location = /50x.html { + root /usr/share/nginx/html; + } +} \ No newline at end of file