| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package org.apache.qpid.protonj2.test.driver.expectations; |
| |
| import static org.hamcrest.CoreMatchers.equalTo; |
| import static org.hamcrest.CoreMatchers.notNullValue; |
| |
| import java.util.Map; |
| |
| import org.apache.qpid.protonj2.test.driver.AMQPTestDriver; |
| import org.apache.qpid.protonj2.test.driver.LinkTracker; |
| import org.apache.qpid.protonj2.test.driver.SessionTracker; |
| import org.apache.qpid.protonj2.test.driver.actions.BeginInjectAction; |
| import org.apache.qpid.protonj2.test.driver.actions.DetachInjectAction; |
| import org.apache.qpid.protonj2.test.driver.codec.ListDescribedType; |
| import org.apache.qpid.protonj2.test.driver.codec.primitives.Symbol; |
| import org.apache.qpid.protonj2.test.driver.codec.primitives.UnsignedInteger; |
| import org.apache.qpid.protonj2.test.driver.codec.primitives.UnsignedShort; |
| import org.apache.qpid.protonj2.test.driver.codec.transport.Detach; |
| import org.apache.qpid.protonj2.test.driver.codec.transport.ErrorCondition; |
| import org.apache.qpid.protonj2.test.driver.codec.util.TypeMapper; |
| import org.apache.qpid.protonj2.test.driver.matchers.transport.DetachMatcher; |
| import org.hamcrest.Matcher; |
| |
| import io.netty.buffer.ByteBuf; |
| |
| /** |
| * Scripted expectation for the AMQP Detach performative |
| */ |
| public class DetachExpectation extends AbstractExpectation<Detach> { |
| |
| private final DetachMatcher matcher = new DetachMatcher(); |
| |
| private DetachInjectAction response; |
| |
| public DetachExpectation(AMQPTestDriver driver) { |
| super(driver); |
| |
| // Default validation of mandatory fields |
| withHandle(notNullValue()); |
| } |
| |
| @Override |
| public DetachExpectation onChannel(int channel) { |
| super.onChannel(channel); |
| return this; |
| } |
| |
| public DetachInjectAction respond() { |
| response = new DetachInjectAction(driver); |
| driver.addScriptedElement(response); |
| return response; |
| } |
| |
| //----- Handle the performative and configure response is told to respond |
| |
| @Override |
| public void handleDetach(int frameSize, Detach detach, ByteBuf payload, int channel, AMQPTestDriver context) { |
| super.handleDetach(frameSize, detach, payload, channel, context); |
| |
| final UnsignedShort remoteChannel = UnsignedShort.valueOf(channel); |
| final SessionTracker session = driver.sessions().getSessionFromRemoteChannel(remoteChannel); |
| |
| if (session == null) { |
| throw new AssertionError(String.format( |
| "Received Detach on channel [%s] that has no matching Session for that remote channel. ", remoteChannel)); |
| } |
| |
| final LinkTracker link = session.handleRemoteDetach(detach); |
| |
| if (link == null) { |
| throw new AssertionError(String.format( |
| "Received Detach on channel [%s] that has no matching Attached link for that remote handle. ", detach.getHandle())); |
| } |
| |
| if (response != null) { |
| // Input was validated now populate response with auto values where not configured |
| // to say otherwise by the test. |
| if (response.onChannel() == BeginInjectAction.CHANNEL_UNSET) { |
| response.onChannel(link.getSession().getLocalChannel()); |
| } |
| |
| if (response.getPerformative().getHandle() == null) { |
| response.withHandle(link.getHandle()); |
| } |
| |
| if (response.getPerformative().getClosed() == null) { |
| response.withClosed(detach.getClosed()); |
| } |
| } |
| } |
| |
| //----- Type specific with methods that perform simple equals checks |
| |
| public DetachExpectation withHandle(int handle) { |
| return withHandle(equalTo(UnsignedInteger.valueOf(handle))); |
| } |
| |
| public DetachExpectation withHandle(long handle) { |
| return withHandle(equalTo(UnsignedInteger.valueOf(handle))); |
| } |
| |
| public DetachExpectation withHandle(UnsignedInteger handle) { |
| return withHandle(equalTo(handle)); |
| } |
| |
| public DetachExpectation withClosed(boolean closed) { |
| return withClosed(equalTo(closed)); |
| } |
| |
| public DetachExpectation withError(ErrorCondition error) { |
| return withError(equalTo(error)); |
| } |
| |
| public DetachExpectation withError(String condition, String description) { |
| return withError(equalTo(new ErrorCondition(Symbol.valueOf(condition), description))); |
| } |
| |
| public DetachExpectation withError(String condition, String description, Map<String, Object> info) { |
| return withError(equalTo(new ErrorCondition(Symbol.valueOf(condition), description, TypeMapper.toSymbolKeyedMap(info)))); |
| } |
| |
| public DetachExpectation withError(Symbol condition, String description) { |
| return withError(equalTo(new ErrorCondition(condition, description))); |
| } |
| |
| public DetachExpectation withError(Symbol condition, String description, Map<Symbol, Object> info) { |
| return withError(equalTo(new ErrorCondition(condition, description, info))); |
| } |
| |
| //----- Matcher based with methods for more complex validation |
| |
| public DetachExpectation withHandle(Matcher<?> m) { |
| matcher.addFieldMatcher(Detach.Field.HANDLE, m); |
| return this; |
| } |
| |
| public DetachExpectation withClosed(Matcher<?> m) { |
| matcher.addFieldMatcher(Detach.Field.CLOSED, m); |
| return this; |
| } |
| |
| public DetachExpectation withError(Matcher<?> m) { |
| matcher.addFieldMatcher(Detach.Field.ERROR, m); |
| return this; |
| } |
| |
| @Override |
| protected Matcher<ListDescribedType> getExpectationMatcher() { |
| return matcher; |
| } |
| |
| @Override |
| protected Class<Detach> getExpectedTypeClass() { |
| return Detach.class; |
| } |
| } |