if (stmt instanceof RpcEffectiveStatement rpc) {
stack.enterSchemaTree(rpc.output().argument());
writeOperationOutput(stack, context.writerParameters(), (ContainerNode) context.data(), output);
+ return;
} else if (stmt instanceof ActionEffectiveStatement action) {
stack.enterSchemaTree(action.output().argument());
writeOperationOutput(stack, context.writerParameters(), (ContainerNode) context.data(), output);
+ return;
}
}
writeData(stack, context.writerParameters(), context.data(), output);
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.nio.charset.StandardCharsets;
import java.text.ParseException;
import java.util.List;
import java.util.function.Consumer;
import javax.ws.rs.container.AsyncResponse;
+import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.UriInfo;
+import javax.ws.rs.ext.MessageBodyWriter;
import org.eclipse.jdt.annotation.NonNull;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.extension.ExtendWith;
import org.opendaylight.mdsal.dom.api.DOMRpcService;
import org.opendaylight.mdsal.dom.spi.FixedDOMSchemaService;
import org.opendaylight.restconf.api.ApiPath;
+import org.opendaylight.restconf.api.MediaTypes;
import org.opendaylight.restconf.common.errors.RestconfDocumentedException;
import org.opendaylight.restconf.common.errors.RestconfError;
import org.opendaylight.restconf.nb.rfc8040.AbstractJukeboxTest;
+import org.opendaylight.restconf.nb.rfc8040.jersey.providers.JsonNormalizedNodeBodyWriter;
+import org.opendaylight.restconf.nb.rfc8040.jersey.providers.XmlNormalizedNodeBodyWriter;
import org.opendaylight.restconf.nb.rfc8040.legacy.NormalizedNodePayload;
import org.opendaylight.restconf.server.mdsal.MdsalRestconfServer;
import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
return JUKEBOX_SCHEMA;
}
+ static final void assertJson(final String expectedJson, final NormalizedNodePayload payload) {
+ assertPayload(expectedJson, payload, new JsonNormalizedNodeBodyWriter(),
+ MediaTypes.APPLICATION_YANG_DATA_JSON);
+ }
+
+ static final void assertXml(final String expectedXml, final NormalizedNodePayload payload) {
+ assertPayload(expectedXml, payload, new XmlNormalizedNodeBodyWriter(), MediaTypes.APPLICATION_YANG_DATA_XML);
+ }
+
+ private static void assertPayload(final String expected, final NormalizedNodePayload payload,
+ final MessageBodyWriter<NormalizedNodePayload> writer, final String mediaType) {
+ final var baos = new ByteArrayOutputStream();
+ try {
+ writer.writeTo(payload, null, null, null, MediaType.valueOf(mediaType), null, baos);
+ } catch (IOException e) {
+ throw new AssertionError(e);
+ }
+ assertEquals(expected, baos.toString(StandardCharsets.UTF_8));
+ }
+
static final NormalizedNode assertNormalizedNode(final int status, final Consumer<AsyncResponse> invocation) {
- return assertEntity(NormalizedNodePayload.class, status, invocation).data();
+ return assertNormalizedNodePayload(status, invocation).data();
+ }
+
+ static final NormalizedNodePayload assertNormalizedNodePayload(final int status,
+ final Consumer<AsyncResponse> invocation) {
+ return assertEntity(NormalizedNodePayload.class, status, invocation);
}
static final <T> T assertEntity(final Class<T> expectedType, final int expectedStatus,
package org.opendaylight.restconf.nb.jaxrs;
import static org.junit.jupiter.api.Assertions.assertEquals;
+import static org.junit.jupiter.api.Assertions.assertInstanceOf;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.eq;
import org.opendaylight.mdsal.dom.spi.SimpleDOMActionResult;
import org.opendaylight.restconf.api.ApiPath;
import org.opendaylight.restconf.nb.rfc8040.AbstractInstanceIdentifierTest;
+import org.opendaylight.restconf.nb.rfc8040.legacy.NormalizedNodePayload;
import org.opendaylight.restconf.server.mdsal.MdsalRestconfServer;
import org.opendaylight.yangtools.yang.common.QName;
import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
@ExtendWith(MockitoExtension.class)
class Netconf799Test extends AbstractInstanceIdentifierTest {
private static final QName OUTPUT_QNAME = QName.create(CONT_QNAME, "output");
+ private static final Absolute RESET_PATH = Absolute.of(CONT_QNAME, CONT1_QNAME, RESET_QNAME);
@Mock
private UriInfo uriInfo;
@Test
void testInvokeAction() throws Exception {
- doReturn(Futures.immediateFuture(new SimpleDOMActionResult(
- ImmutableNodes.newContainerBuilder().withNodeIdentifier(NodeIdentifier.create(OUTPUT_QNAME)).build())))
- .when(actionService).invokeAction(eq(Absolute.of(CONT_QNAME, CONT1_QNAME, RESET_QNAME)), any(), any());
+ doReturn(Futures.immediateFuture(new SimpleDOMActionResult(ImmutableNodes.newContainerBuilder()
+ .withNodeIdentifier(NodeIdentifier.create(OUTPUT_QNAME))
+ .build())))
+ .when(actionService).invokeAction(eq(RESET_PATH), any(), any());
final var restconf = new JaxRsRestconf(new MdsalRestconfServer(new FixedDOMSchemaService(IID_SCHEMA),
dataBroker, rpcService, actionService, mountPointService));
assertEquals(204, response.getStatus());
assertNull(response.getEntity());
}
+
+ @Test
+ void testInvokeActionOutput() throws Exception {
+ doReturn(Futures.immediateFuture(new SimpleDOMActionResult(ImmutableNodes.newContainerBuilder()
+ .withNodeIdentifier(NodeIdentifier.create(OUTPUT_QNAME))
+ .withChild(ImmutableNodes.leafNode(QName.create(OUTPUT_QNAME, "timestamp"), "somevalue"))
+ .build())))
+ .when(actionService).invokeAction(eq(RESET_PATH), any(), any());
+
+ final var restconf = new JaxRsRestconf(new MdsalRestconfServer(new FixedDOMSchemaService(IID_SCHEMA),
+ dataBroker, rpcService, actionService, mountPointService));
+ doReturn(new MultivaluedHashMap<>()).when(uriInfo).getQueryParameters();
+ doReturn(true).when(asyncResponse).resume(captor.capture());
+ restconf.postDataJSON(ApiPath.parse("instance-identifier-module:cont/cont1/reset"),
+ stringInputStream("""
+ {
+ "instance-identifier-module:input": {
+ "delay": 600
+ }
+ }"""), uriInfo, asyncResponse);
+ final var response = captor.getValue();
+ assertEquals(200, response.getStatus());
+
+ final var payload = assertInstanceOf(NormalizedNodePayload.class, response.getEntity());
+ AbstractRestconfTest.assertJson("""
+ {"instance-identifier-module:output":{"timestamp":"somevalue"}}""", payload);
+ AbstractRestconfTest.assertXml("""
+ <output xmlns="instance:identifier:module"><timestamp>somevalue</timestamp></output>""", payload);
+ }
}
.build();
private static final ContainerNode OUTPUT = ImmutableNodes.newContainerBuilder()
.withNodeIdentifier(new NodeIdentifier(QName.create(RPC, "output")))
- .withChild(ImmutableNodes.leafNode(QName.create(RPC, "content"), "operation result"))
+ .withChild(ImmutableNodes.newContainerBuilder()
+ .withNodeIdentifier(new NodeIdentifier(QName.create(RPC, "cont-out")))
+ .withChild(ImmutableNodes.leafNode(QName.create(RPC, "lf-out"), "operation result"))
+ .build())
.build();
private static final EffectiveModelContext MODEL_CONTEXT =
YangParserTestUtils.parseYangResourceDirectory("/invoke-rpc");
doReturn(Futures.immediateFuture(new DefaultDOMRpcResult(OUTPUT, List.of())))
.when(rpcService).invokeRpc(RPC, INPUT);
- assertEquals(OUTPUT, assertNormalizedNode(200, ar -> restconf.operationsJsonPOST(
+ final var payload = assertNormalizedNodePayload(200, ar -> restconf.operationsJsonPOST(
apiPath("invoke-rpc-module:rpc-test"),
stringInputStream("""
{
"lf" : "test"
}
}
- }"""), uriInfo, ar)));
+ }"""), uriInfo, ar));
+ assertEquals(OUTPUT, payload.data());
+ assertJson("""
+ {"invoke-rpc-module:output":{"cont-out":{"lf-out":"operation result"}}}""", payload);
+ assertXml("""
+ <output xmlns="invoke:rpc:module"><cont-out><lf-out>operation result</lf-out></cont-out></output>""",
+ payload);
}
private void prepNNC(final ContainerNode result) {
default 0;
}
}
+ output {
+ leaf timestamp {
+ type string;
+ }
+ }
}
}
}