|
21 | 21 | import io.netty.buffer.ByteBuf;
|
22 | 22 | import io.netty.buffer.Unpooled;
|
23 | 23 | import io.netty.channel.embedded.EmbeddedChannel;
|
| 24 | +import io.netty.handler.codec.EncoderException; |
24 | 25 | import org.junit.Rule;
|
25 | 26 | import org.junit.Test;
|
26 | 27 | import org.junit.rules.ExpectedException;
|
27 | 28 |
|
28 |
| -import java.util.Collections; |
| 29 | +import java.io.IOException; |
29 | 30 | import java.util.HashMap;
|
30 | 31 | import java.util.List;
|
31 | 32 |
|
32 |
| -import org.neo4j.driver.internal.InternalNode; |
33 |
| -import org.neo4j.driver.internal.InternalPath; |
34 |
| -import org.neo4j.driver.internal.InternalRelationship; |
35 | 33 | import org.neo4j.driver.internal.async.BoltProtocolUtil;
|
36 | 34 | import org.neo4j.driver.internal.async.ChannelPipelineBuilderImpl;
|
37 | 35 | import org.neo4j.driver.internal.async.inbound.InboundMessageDispatcher;
|
|
41 | 39 | import org.neo4j.driver.v1.exceptions.ClientException;
|
42 | 40 |
|
43 | 41 | import static java.util.Arrays.asList;
|
| 42 | +import static org.hamcrest.CoreMatchers.equalTo; |
| 43 | +import static org.hamcrest.CoreMatchers.instanceOf; |
| 44 | +import static org.hamcrest.MatcherAssert.assertThat; |
44 | 45 | import static org.hamcrest.Matchers.startsWith;
|
45 | 46 | import static org.junit.Assert.assertEquals;
|
46 | 47 | import static org.junit.Assert.assertTrue;
|
| 48 | +import static org.junit.Assert.fail; |
47 | 49 | import static org.neo4j.driver.internal.async.ChannelAttributes.messageDispatcher;
|
48 | 50 | import static org.neo4j.driver.internal.async.ChannelAttributes.setMessageDispatcher;
|
49 | 51 | import static org.neo4j.driver.internal.logging.DevNullLogging.DEV_NULL_LOGGING;
|
50 |
| -import static org.neo4j.driver.v1.Values.EmptyMap; |
| 52 | +import static org.neo4j.driver.internal.util.ValueFactory.emptyNodeValue; |
| 53 | +import static org.neo4j.driver.internal.util.ValueFactory.emptyPathValue; |
| 54 | +import static org.neo4j.driver.internal.util.ValueFactory.emptyRelationshipValue; |
| 55 | +import static org.neo4j.driver.internal.util.ValueFactory.filledNodeValue; |
| 56 | +import static org.neo4j.driver.internal.util.ValueFactory.filledPathValue; |
| 57 | +import static org.neo4j.driver.internal.util.ValueFactory.filledRelationshipValue; |
51 | 58 | import static org.neo4j.driver.v1.Values.ofValue;
|
52 | 59 | import static org.neo4j.driver.v1.Values.parameters;
|
53 | 60 | import static org.neo4j.driver.v1.Values.value;
|
@@ -80,33 +87,51 @@ public void shouldUnpackAllResponses() throws Throwable
|
80 | 87 | }
|
81 | 88 |
|
82 | 89 | @Test
|
83 |
| - public void shouldUnpackAllValues() throws Throwable |
| 90 | + public void shouldPackUnpackValidValues() throws Throwable |
84 | 91 | {
|
85 | 92 | assertSerializesValue( value( parameters( "cat", null, "dog", null ) ) );
|
86 | 93 | assertSerializesValue( value( parameters( "k", 12, "a", "banana" ) ) );
|
87 | 94 | assertSerializesValue( value( asList( "k", 12, "a", "banana" ) ) );
|
88 |
| - assertSerializesValue( value( |
89 |
| - new InternalNode( 1, Collections.singletonList( "User" ), parameters( "name", "Bob", "age", 45 ).asMap( |
90 |
| - ofValue() ) ) |
91 |
| - ) ); |
92 |
| - assertSerializesValue( value( new InternalNode( 1 ) ) ); |
93 |
| - assertSerializesValue( value( |
94 |
| - new InternalRelationship( 1, 1, 1, |
95 |
| - "KNOWS", |
96 |
| - parameters( "name", "Bob", "age", 45 ).asMap( ofValue() ) ) ) ); |
97 |
| - assertSerializesValue( value( |
98 |
| - new InternalPath( |
99 |
| - new InternalNode( 1 ), |
100 |
| - new InternalRelationship( 2, 1, 3, |
101 |
| - "KNOWS", EmptyMap.asMap( ofValue() ) ), |
102 |
| - new InternalNode( 3 ), |
103 |
| - new InternalRelationship( 4, 3, 5, |
104 |
| - "LIKES", EmptyMap.asMap( ofValue() ) ), |
105 |
| - new InternalNode( 5 ) |
106 |
| - ) ) ); |
107 |
| - assertSerializesValue( value( new InternalPath( new InternalNode( 1 ) ) ) ); |
108 | 95 | }
|
109 | 96 |
|
| 97 | + @Test |
| 98 | + public void shouldUnpackNodeRelationshipAndPath() throws Throwable |
| 99 | + { |
| 100 | + // Given |
| 101 | + assertOnlyDeserializesValue( emptyNodeValue() ); |
| 102 | + assertOnlyDeserializesValue( filledNodeValue() ); |
| 103 | + assertOnlyDeserializesValue( emptyRelationshipValue() ); |
| 104 | + assertOnlyDeserializesValue( filledRelationshipValue() ); |
| 105 | + assertOnlyDeserializesValue( emptyPathValue() ); |
| 106 | + assertOnlyDeserializesValue( filledPathValue() ); |
| 107 | + } |
| 108 | + |
| 109 | + |
| 110 | + @Test |
| 111 | + public void shouldErrorPackingNode() throws Throwable |
| 112 | + { |
| 113 | + // Given |
| 114 | + Value value = filledNodeValue(); |
| 115 | + expectIOExceptionWithMessage( value, "Unknown type: NODE" ); |
| 116 | + } |
| 117 | + |
| 118 | + @Test |
| 119 | + public void shouldErrorPackingRelationship() throws Throwable |
| 120 | + { |
| 121 | + // Given |
| 122 | + Value value = filledRelationshipValue(); |
| 123 | + expectIOExceptionWithMessage( value, "Unknown type: RELATIONSHIP" ); |
| 124 | + } |
| 125 | + |
| 126 | + @Test |
| 127 | + public void shouldErrorPackingPath() throws Throwable |
| 128 | + { |
| 129 | + // Given |
| 130 | + Value value = filledPathValue(); |
| 131 | + expectIOExceptionWithMessage( value, "Unknown type: PATH" ); |
| 132 | + } |
| 133 | + |
| 134 | + |
110 | 135 | @Test
|
111 | 136 | public void shouldGiveHelpfulErrorOnMalformedNodeStruct() throws Throwable
|
112 | 137 | {
|
@@ -149,6 +174,11 @@ private void assertSerializes( Message message ) throws Throwable
|
149 | 174 | }
|
150 | 175 |
|
151 | 176 | private EmbeddedChannel newEmbeddedChannel()
|
| 177 | + { |
| 178 | + return newEmbeddedChannel( format ); |
| 179 | + } |
| 180 | + |
| 181 | + private EmbeddedChannel newEmbeddedChannel( MessageFormat format ) |
152 | 182 | {
|
153 | 183 | EmbeddedChannel channel = new EmbeddedChannel();
|
154 | 184 | setMessageDispatcher( channel, new MemorizingInboundMessageDispatcher( channel, DEV_NULL_LOGGING ) );
|
@@ -186,4 +216,51 @@ private Message unpack( ByteBuf packed, EmbeddedChannel channel ) throws Throwab
|
186 | 216 | assertEquals( 1, unpackedMessages.size() );
|
187 | 217 | return unpackedMessages.get( 0 );
|
188 | 218 | }
|
| 219 | + |
| 220 | + private void assertOnlyDeserializesValue( Value value ) throws Throwable |
| 221 | + { |
| 222 | + RecordMessage message = new RecordMessage( new Value[]{value} ); |
| 223 | + ByteBuf packed = knowledgeablePack( message ); |
| 224 | + |
| 225 | + EmbeddedChannel channel = newEmbeddedChannel(); |
| 226 | + Message unpackedMessage = unpack( packed, channel ); |
| 227 | + |
| 228 | + assertEquals( message, unpackedMessage ); |
| 229 | + } |
| 230 | + |
| 231 | + private ByteBuf knowledgeablePack( Message message ) throws IOException |
| 232 | + { |
| 233 | + EmbeddedChannel channel = newEmbeddedChannel( new KnowledgeablePackStreamMessageFormat() ); |
| 234 | + assertTrue( channel.writeOutbound( message ) ); |
| 235 | + |
| 236 | + ByteBuf[] packedMessages = channel.outboundMessages() |
| 237 | + .stream() |
| 238 | + .map( msg -> (ByteBuf) msg ) |
| 239 | + .toArray( ByteBuf[]::new ); |
| 240 | + |
| 241 | + return Unpooled.wrappedBuffer( packedMessages ); |
| 242 | + } |
| 243 | + |
| 244 | + private void expectIOExceptionWithMessage( Value value, String errorMessage ) |
| 245 | + { |
| 246 | + RecordMessage message = new RecordMessage( new Value[]{value} ); |
| 247 | + EmbeddedChannel channel = newEmbeddedChannel(); |
| 248 | + |
| 249 | + try |
| 250 | + { |
| 251 | + pack( message, channel ); |
| 252 | + fail( "Expecting a EncoderException" ); |
| 253 | + } |
| 254 | + catch ( EncoderException e ) |
| 255 | + { |
| 256 | + Throwable cause = e.getCause(); |
| 257 | + assertThat( cause, instanceOf( IOException.class ) ); |
| 258 | + assertThat( cause.getMessage(), equalTo( errorMessage ) ); |
| 259 | + } |
| 260 | + catch ( Exception e ) |
| 261 | + { |
| 262 | + fail( "Expecting a EncoderException but got " + e ); |
| 263 | + } |
| 264 | + } |
| 265 | + |
189 | 266 | }
|
0 commit comments