From 31ac86c3f044b3fc50d4deb1d86a9606e6cf0967 Mon Sep 17 00:00:00 2001 From: Mateus Molina Date: Mon, 28 Oct 2024 15:53:04 +0100 Subject: [PATCH] Fix default endpoint resolve strategy from client lib not conforming with specification (#504) * test: add bug failing test * fix: change endpoint parsing mechanism of the Aas/Sm DescriptorResolvers to conform with spec * ci: retriggering * fix: address review remarks --- .../resolvers/AasDescriptorResolver.java | 16 +++------- .../resolvers/SubmodelDescriptorResolver.java | 14 +++----- .../aasenvironment/client/TestFixture.java | 25 +++++++++++++-- .../RegistryDescriptorResolverTest.java | 32 +++++++++++++++---- .../mapper/DummyAasDescriptorFactory.java | 29 +++++++++++++---- .../DummySubmodelDescriptorFactory.java | 32 ++++++++++++------- ...bmodelRepositoryRegistryLinkTestSuite.java | 2 +- 7 files changed, 102 insertions(+), 48 deletions(-) diff --git a/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/AasDescriptorResolver.java b/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/AasDescriptorResolver.java index 6e743a8e1..a010f3df7 100644 --- a/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/AasDescriptorResolver.java +++ b/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/AasDescriptorResolver.java @@ -30,17 +30,20 @@ import org.eclipse.digitaltwin.basyx.aasregistry.client.model.AssetAdministrationShellDescriptor; import org.eclipse.digitaltwin.basyx.aasregistry.client.model.Endpoint; +import org.eclipse.digitaltwin.basyx.aasregistry.client.model.ProtocolInformation; import org.eclipse.digitaltwin.basyx.aasservice.client.ConnectedAasService; import org.eclipse.digitaltwin.basyx.client.internal.resolver.DescriptorResolver; /** * Resolves an AasDescriptor into a {@link ConnectedAasService} - * + * * @author mateusmolina, danish * */ public class AasDescriptorResolver implements DescriptorResolver { + static final String SPEC_INTERFACE = "AAS-3.0"; + private final EndpointResolver endpointResolver; /** @@ -66,16 +69,7 @@ public ConnectedAasService resolveDescriptor(AssetAdministrationShellDescriptor public static Optional parseEndpoint(Endpoint endpoint) { try { - if (endpoint == null || endpoint.getProtocolInformation() == null || endpoint.getProtocolInformation() - .getHref() == null) - return Optional.empty(); - - String baseHref = endpoint.getProtocolInformation() - .getHref(); - // TODO not working: String queryString = "?" + endpoint.toUrlQueryString(); - String queryString = ""; - URI uri = new URI(baseHref + queryString); - return Optional.of(uri); + return Optional.ofNullable(endpoint).filter(ep -> ep.getInterface().equals(SPEC_INTERFACE)).map(Endpoint::getProtocolInformation).map(ProtocolInformation::getHref).map(URI::create); } catch (Exception e) { return Optional.empty(); } diff --git a/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/SubmodelDescriptorResolver.java b/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/SubmodelDescriptorResolver.java index 5ddc10d72..4d7014577 100644 --- a/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/SubmodelDescriptorResolver.java +++ b/basyx.aasenvironment/basyx.aasenvironment-client/src/main/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/SubmodelDescriptorResolver.java @@ -25,6 +25,7 @@ package org.eclipse.digitaltwin.basyx.aasenvironment.client.resolvers; +import org.eclipse.digitaltwin.basyx.submodelregistry.client.model.ProtocolInformation; import org.eclipse.digitaltwin.basyx.client.internal.resolver.DescriptorResolver; import java.net.URI; import java.util.Optional; @@ -41,6 +42,8 @@ */ public class SubmodelDescriptorResolver implements DescriptorResolver { + static final String SPEC_INTERFACE = "SUBMODEL-3.0"; + private final EndpointResolver endpointResolver; /** @@ -67,16 +70,7 @@ public ConnectedSubmodelService resolveDescriptor(SubmodelDescriptor smDescripto public static Optional parseEndpoint(Endpoint endpoint) { try { - if (endpoint == null || endpoint.getProtocolInformation() == null || endpoint.getProtocolInformation() - .getHref() == null) - return Optional.empty(); - - String baseHref = endpoint.getProtocolInformation() - .getHref(); - // TODO not working: String queryString = "?" + endpoint.toUrlQueryString(); - String queryString = ""; - URI uri = new URI(baseHref + queryString); - return Optional.of(uri); + return Optional.ofNullable(endpoint).filter(ep -> ep.getInterface().equals(SPEC_INTERFACE)).map(Endpoint::getProtocolInformation).map(ProtocolInformation::getHref).map(URI::create); } catch (Exception e) { return Optional.empty(); } diff --git a/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/TestFixture.java b/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/TestFixture.java index 275f54eb2..5d9468fa1 100644 --- a/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/TestFixture.java +++ b/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/TestFixture.java @@ -25,6 +25,8 @@ package org.eclipse.digitaltwin.basyx.aasenvironment.client; +import java.util.LinkedList; + import org.eclipse.digitaltwin.aas4j.v3.model.AssetAdministrationShell; import org.eclipse.digitaltwin.aas4j.v3.model.AssetInformation; import org.eclipse.digitaltwin.aas4j.v3.model.AssetKind; @@ -38,6 +40,7 @@ import org.eclipse.digitaltwin.aas4j.v3.model.impl.DefaultReference; import org.eclipse.digitaltwin.aas4j.v3.model.impl.DefaultSubmodel; import org.eclipse.digitaltwin.basyx.aasregistry.client.model.AssetAdministrationShellDescriptor; +import org.eclipse.digitaltwin.basyx.aasregistry.client.model.Endpoint; import org.eclipse.digitaltwin.basyx.aasregistry.main.client.mapper.DummyAasDescriptorFactory; import org.eclipse.digitaltwin.basyx.http.Base64UrlEncoder; import org.eclipse.digitaltwin.basyx.submodelregistry.client.mapper.AttributeMapper; @@ -91,6 +94,15 @@ public AssetAdministrationShellDescriptor buildAasPre1Descriptor() { return DummyAasDescriptorFactory.createDummyDescriptor(AAS_PRE1_ID, AAS_PRE1_IDSHORT, AAS_PRE1_GLOBALASSETID, aasRepositoryBasePath); } + public AssetAdministrationShellDescriptor buildAasPre1Descriptor_withMultipleInterfaces() { + LinkedList endpoints = new LinkedList<>(); + + endpoints.add(DummyAasDescriptorFactory.createEndpoint(aasRepositoryBasePath, "AAS-REPOSITORY-3.0")); + endpoints.add(DummyAasDescriptorFactory.createEndpoint(AAS_PRE1_ID, aasRepositoryBasePath, "AAS-3.0")); + + return DummyAasDescriptorFactory.createDummyDescriptor(AAS_PRE1_ID, AAS_PRE1_IDSHORT, AAS_PRE1_GLOBALASSETID, endpoints); + } + public Reference buildSmPre1Ref() { return new DefaultReference.Builder().type(ReferenceTypes.MODEL_REFERENCE).keys(new DefaultKey.Builder().type(KeyTypes.SUBMODEL).value(SM_PRE1_ID).build()).build(); } @@ -100,7 +112,16 @@ public Submodel buildSmPre1() { } public SubmodelDescriptor buildSmPre1Descriptor() { - return DummySubmodelDescriptorFactory.createDummyDescriptor(SM_PRE1_ID, SM_PRE1_IDSHORT, smRepositoryBasePath, null); + return DummySubmodelDescriptorFactory.createDummyDescriptor(SM_PRE1_ID, SM_PRE1_IDSHORT, null, smRepositoryBasePath); + } + + public SubmodelDescriptor buildSmPre1Descriptor_withMultipleInterfaces() { + LinkedList endpoints = new LinkedList<>(); + + endpoints.add(DummySubmodelDescriptorFactory.createEndpoint(smRepositoryBasePath, "SUBMODEL-REPOSITORY-3.0")); + endpoints.add(DummySubmodelDescriptorFactory.createEndpoint(SM_PRE1_ID, smRepositoryBasePath, "SUBMODEL-3.0")); + + return DummySubmodelDescriptorFactory.createDummyDescriptor(SM_PRE1_ID, SM_PRE1_IDSHORT, null, endpoints); } public AssetAdministrationShell buildAasPos1() { @@ -116,7 +137,7 @@ public AssetAdministrationShellDescriptor buildAasPos1Descriptor() { } public SubmodelDescriptor buildSmPos1Descriptor() { - return DummySubmodelDescriptorFactory.createDummyDescriptor(SM_POS1_ID, SM_POS1_IDSHORT, smRepositoryBasePath, new AttributeMapper(ConnectedAasManagerHelper.buildObjectMapper()).mapSemanticId(buildSmPos1SemanticId())); + return DummySubmodelDescriptorFactory.createDummyDescriptor(SM_POS1_ID, SM_POS1_IDSHORT, new AttributeMapper(ConnectedAasManagerHelper.buildObjectMapper()).mapSemanticId(buildSmPos1SemanticId()), smRepositoryBasePath); } public Reference buildSmPos1SemanticId() { diff --git a/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/RegistryDescriptorResolverTest.java b/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/RegistryDescriptorResolverTest.java index e0666db0d..a6ced6534 100644 --- a/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/RegistryDescriptorResolverTest.java +++ b/basyx.aasenvironment/basyx.aasenvironment-client/src/test/java/org/eclipse/digitaltwin/basyx/aasenvironment/client/resolvers/RegistryDescriptorResolverTest.java @@ -31,7 +31,6 @@ import org.eclipse.digitaltwin.aas4j.v3.model.Submodel; import org.eclipse.digitaltwin.basyx.aasenvironment.client.DummyAasEnvironmentComponent; import org.eclipse.digitaltwin.basyx.aasenvironment.client.TestFixture; -import org.eclipse.digitaltwin.basyx.aasregistry.client.ApiException; import org.eclipse.digitaltwin.basyx.aasrepository.AasRepository; import org.eclipse.digitaltwin.basyx.submodelrepository.SubmodelRepository; import org.junit.AfterClass; @@ -52,10 +51,10 @@ public class RegistryDescriptorResolverTest { private static AasRepository aasRepository; private static SubmodelRepository smRepository; - private final static String AAS_REPOSITORY_BASE_PATH = "http://localhost:8081"; - private final static String SM_REPOSITORY_BASE_PATH = "http://localhost:8081"; + private static final String AAS_REPOSITORY_BASE_PATH = "http://localhost:8081"; + private static final String SM_REPOSITORY_BASE_PATH = "http://localhost:8081"; - private final static TestFixture FIXTURE = new TestFixture(AAS_REPOSITORY_BASE_PATH, SM_REPOSITORY_BASE_PATH); + private static final TestFixture FIXTURE = new TestFixture(AAS_REPOSITORY_BASE_PATH, SM_REPOSITORY_BASE_PATH); @BeforeClass public static void initApplication() { @@ -73,7 +72,7 @@ public static void cleanUp() { } @Test - public void resolveAasDescriptor() throws ApiException { + public void resolveAasDescriptor() { AasDescriptorResolver resolver = new AasDescriptorResolver(new EndpointResolver()); AssetAdministrationShell expectedAas = FIXTURE.buildAasPre1(); @@ -84,7 +83,18 @@ public void resolveAasDescriptor() throws ApiException { } @Test - public void resolveSmDescriptor() throws ApiException { + public void resolveAasDescriptor_withMultipleInterfaces() { + AasDescriptorResolver resolver = new AasDescriptorResolver(new EndpointResolver()); + + AssetAdministrationShell expectedAas = FIXTURE.buildAasPre1(); + + AssetAdministrationShell actualAas = resolver.resolveDescriptor(FIXTURE.buildAasPre1Descriptor_withMultipleInterfaces()).getAAS(); + + assertEquals(expectedAas, actualAas); + } + + @Test + public void resolveSmDescriptor() { SubmodelDescriptorResolver resolver = new SubmodelDescriptorResolver(new EndpointResolver()); Submodel expectedSm = FIXTURE.buildSmPre1(); @@ -94,4 +104,14 @@ public void resolveSmDescriptor() throws ApiException { assertEquals(expectedSm, actualSm); } + @Test + public void resolveSmDescriptor_withMultipleInterfaces() { + SubmodelDescriptorResolver resolver = new SubmodelDescriptorResolver(new EndpointResolver()); + + Submodel expectedSm = FIXTURE.buildSmPre1(); + + Submodel actualSm = resolver.resolveDescriptor(FIXTURE.buildSmPre1Descriptor_withMultipleInterfaces()).getSubmodel(); + + assertEquals(expectedSm, actualSm); + } } diff --git a/basyx.aasregistry/basyx.aasregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/aasregistry/main/client/mapper/DummyAasDescriptorFactory.java b/basyx.aasregistry/basyx.aasregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/aasregistry/main/client/mapper/DummyAasDescriptorFactory.java index 7441ce29c..9e2649274 100644 --- a/basyx.aasregistry/basyx.aasregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/aasregistry/main/client/mapper/DummyAasDescriptorFactory.java +++ b/basyx.aasregistry/basyx.aasregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/aasregistry/main/client/mapper/DummyAasDescriptorFactory.java @@ -27,6 +27,8 @@ import java.net.MalformedURLException; import java.net.URL; +import java.util.LinkedList; +import java.util.List; import org.eclipse.digitaltwin.basyx.aasregistry.client.model.AssetAdministrationShellDescriptor; import org.eclipse.digitaltwin.basyx.aasregistry.client.model.AssetKind; @@ -43,7 +45,7 @@ public class DummyAasDescriptorFactory { private static final String AAS_REPOSITORY_PATH = "/shells"; - public static AssetAdministrationShellDescriptor createDummyDescriptor(String aasId, String idShort, String globalAssetId, String... aasRepoBaseUrls) { + public static AssetAdministrationShellDescriptor createDummyDescriptor(String aasId, String idShort, String globalAssetId, List endpoints) { AssetAdministrationShellDescriptor descriptor = new AssetAdministrationShellDescriptor(); @@ -51,23 +53,36 @@ public static AssetAdministrationShellDescriptor createDummyDescriptor(String aa descriptor.setIdShort(idShort); descriptor.setAssetKind(AssetKind.INSTANCE); descriptor.setGlobalAssetId(globalAssetId); + descriptor.setEndpoints(endpoints); + + return descriptor; + } + + public static AssetAdministrationShellDescriptor createDummyDescriptor(String aasId, String idShort, String globalAssetId, String... aasRepoBaseUrls) { + LinkedList endpoints = new LinkedList<>(); + for (String eachUrl : aasRepoBaseUrls) { - descriptor.addEndpointsItem(createEndpointItem(aasId, eachUrl)); + endpoints.add(createEndpoint(aasId, eachUrl, "AAS-3.0")); } - return descriptor; + + return createDummyDescriptor(aasId, idShort, globalAssetId, endpoints); } - private static Endpoint createEndpointItem(String aasId, String aasRepoBaseUrl) { + public static Endpoint createEndpoint(String endpointUrl, String endpointInterface) { Endpoint endpoint = new Endpoint(); - endpoint.setInterface("AAS-3.0"); - endpoint.setProtocolInformation(createProtocolInformation(aasId, aasRepoBaseUrl)); + endpoint.setInterface(endpointInterface); + endpoint.setProtocolInformation(createProtocolInformation(endpointUrl)); return endpoint; } - private static ProtocolInformation createProtocolInformation(String aasId, String aasRepoBaseUrl) { + public static Endpoint createEndpoint(String aasId, String aasRepoBaseUrl, String endpointInterface) { String href = createHref(aasId, aasRepoBaseUrl); + return createEndpoint(href, endpointInterface); + } + + private static ProtocolInformation createProtocolInformation(String href) { ProtocolInformation protocolInformation = new ProtocolInformation(); protocolInformation.setHref(href); protocolInformation.endpointProtocol(getProtocol(href)); diff --git a/basyx.submodelregistry/basyx.submodelregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/submodelregistry/client/mapper/DummySubmodelDescriptorFactory.java b/basyx.submodelregistry/basyx.submodelregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/submodelregistry/client/mapper/DummySubmodelDescriptorFactory.java index d6755f00e..8dfaf8656 100644 --- a/basyx.submodelregistry/basyx.submodelregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/submodelregistry/client/mapper/DummySubmodelDescriptorFactory.java +++ b/basyx.submodelregistry/basyx.submodelregistry-client-native/src/test/java/org/eclipse/digitaltwin/basyx/submodelregistry/client/mapper/DummySubmodelDescriptorFactory.java @@ -28,6 +28,8 @@ import java.net.MalformedURLException; import java.net.URL; import java.util.Arrays; +import java.util.LinkedList; +import java.util.List; import org.eclipse.digitaltwin.basyx.http.Base64UrlEncodedIdentifier; import org.eclipse.digitaltwin.basyx.submodelregistry.client.model.Endpoint; @@ -47,37 +49,45 @@ public class DummySubmodelDescriptorFactory { private static final String SUBMODEL_REPOSITORY_PATH = "/submodels"; - public static SubmodelDescriptor createDummyDescriptor(String smId, String idShort, String smRepoBaseUrl, Reference semanticId) { - return createDummyDescriptor(smId, idShort, new String[] {smRepoBaseUrl}, semanticId); - } - - public static SubmodelDescriptor createDummyDescriptor(String smId, String idShort, String[] smRepoBaseUrls, Reference semanticId) { + public static SubmodelDescriptor createDummyDescriptor(String smId, String idShort, Reference semanticId, List endpoints) { SubmodelDescriptor descriptor = new SubmodelDescriptor(); descriptor.setId(smId); descriptor.setIdShort(idShort); descriptor.setSemanticId(semanticId); + descriptor.setEndpoints(endpoints); + + return descriptor; + } + + public static SubmodelDescriptor createDummyDescriptor(String smId, String idShort, Reference semanticId, String... smRepoBaseUrls) { + LinkedList endpoints = new LinkedList<>(); for (String eachUrl : smRepoBaseUrls) { - descriptor.addEndpointsItem(createEndpointItem(smId, eachUrl)); + endpoints.add(createEndpoint(smId, eachUrl, "SUBMODEL-3.0")); } - return descriptor; + + return createDummyDescriptor(smId, idShort, semanticId, endpoints); } public static Reference createSemanticId() { return new Reference().keys(Arrays.asList(new Key().type(KeyTypes.GLOBALREFERENCE).value("0173-1#01-AFZ615#016"))).type(ReferenceTypes.EXTERNALREFERENCE); } - private static Endpoint createEndpointItem(String smId, String smRepoBaseUrl) { + public static Endpoint createEndpoint(String endpointUrl, String endpointInterface) { Endpoint endpoint = new Endpoint(); - endpoint.setInterface("SUBMODEL-3.0"); - endpoint.setProtocolInformation(createProtocolInformation(smId, smRepoBaseUrl)); + endpoint.setInterface(endpointInterface); + endpoint.setProtocolInformation(createProtocolInformation(endpointUrl)); return endpoint; } - private static ProtocolInformation createProtocolInformation(String smId, String smRepoBaseUrl) { + public static Endpoint createEndpoint(String smId, String smRepoBaseUrl, String endpointInterface) { String href = createHref(smId, smRepoBaseUrl); + return createEndpoint(href, endpointInterface); + } + + private static ProtocolInformation createProtocolInformation(String href) { ProtocolInformation protocolInformation = new ProtocolInformation(); protocolInformation.setHref(href); protocolInformation.endpointProtocol(getProtocol(href)); diff --git a/basyx.submodelrepository/basyx.submodelrepository-feature-registry-integration/src/test/java/org/eclipse/digitaltwin/basyx/submodelrepository/feature/registry/integration/SubmodelRepositoryRegistryLinkTestSuite.java b/basyx.submodelrepository/basyx.submodelrepository-feature-registry-integration/src/test/java/org/eclipse/digitaltwin/basyx/submodelrepository/feature/registry/integration/SubmodelRepositoryRegistryLinkTestSuite.java index 2c02ef48f..0e63d1de7 100644 --- a/basyx.submodelrepository/basyx.submodelrepository-feature-registry-integration/src/test/java/org/eclipse/digitaltwin/basyx/submodelrepository/feature/registry/integration/SubmodelRepositoryRegistryLinkTestSuite.java +++ b/basyx.submodelrepository/basyx.submodelrepository-feature-registry-integration/src/test/java/org/eclipse/digitaltwin/basyx/submodelrepository/feature/registry/integration/SubmodelRepositoryRegistryLinkTestSuite.java @@ -64,7 +64,7 @@ public abstract class SubmodelRepositoryRegistryLinkTestSuite { protected abstract String getSubmodelRegistryUrl(); protected abstract SubmodelRegistryApi getSubmodelRegistryApi(); - private final SubmodelDescriptor DUMMY_DESCRIPTOR = DummySubmodelDescriptorFactory.createDummyDescriptor(DUMMY_SUBMODEL_ID, DUMMY_SUBMODEL_IDSHORT, getSubmodelRepoBaseUrls(), DummySubmodelDescriptorFactory.createSemanticId()); + private final SubmodelDescriptor DUMMY_DESCRIPTOR = DummySubmodelDescriptorFactory.createDummyDescriptor(DUMMY_SUBMODEL_ID, DUMMY_SUBMODEL_IDSHORT, DummySubmodelDescriptorFactory.createSemanticId(), getSubmodelRepoBaseUrls()); @Test public void createSubmodel() throws FileNotFoundException, IOException, ApiException {