001/*- 002 * #%L 003 * HAPI FHIR Storage api 004 * %% 005 * Copyright (C) 2014 - 2025 Smile CDR, Inc. 006 * %% 007 * Licensed under the Apache License, Version 2.0 (the "License"); 008 * you may not use this file except in compliance with the License. 009 * You may obtain a copy of the License at 010 * 011 * http://www.apache.org/licenses/LICENSE-2.0 012 * 013 * Unless required by applicable law or agreed to in writing, software 014 * distributed under the License is distributed on an "AS IS" BASIS, 015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 016 * See the License for the specific language governing permissions and 017 * limitations under the License. 018 * #L% 019 */ 020package ca.uhn.fhir.jpa.subscription.channel.impl; 021 022import ca.uhn.fhir.jpa.subscription.channel.api.IChannelProducer; 023import ca.uhn.fhir.jpa.subscription.channel.api.IChannelReceiver; 024import jakarta.annotation.Nonnull; 025import org.springframework.messaging.MessageHandler; 026import org.springframework.messaging.support.ExecutorSubscribableChannel; 027 028import java.util.ArrayList; 029import java.util.Optional; 030import java.util.concurrent.Executor; 031import java.util.function.Supplier; 032 033import static org.apache.commons.lang3.ObjectUtils.defaultIfNull; 034 035public class LinkedBlockingChannel extends ExecutorSubscribableChannel implements IChannelProducer, IChannelReceiver { 036 037 private final String myName; 038 private final Supplier<Integer> myQueueSizeSupplier; 039 040 public LinkedBlockingChannel(String theName, Executor theExecutor, Supplier<Integer> theQueueSizeSupplier) { 041 super(theExecutor); 042 myName = theName; 043 myQueueSizeSupplier = theQueueSizeSupplier; 044 } 045 046 public int getQueueSizeForUnitTest() { 047 return defaultIfNull(myQueueSizeSupplier.get(), 0); 048 } 049 050 public void clearInterceptorsForUnitTest() { 051 setInterceptors(new ArrayList<>()); 052 } 053 054 @Override 055 public String getName() { 056 return myName; 057 } 058 059 @Override 060 public boolean hasSubscription(@Nonnull MessageHandler handler) { 061 return getSubscribers().stream() 062 .map(t -> (RetryingMessageHandlerWrapper) t) 063 .anyMatch(t -> t.getWrappedHandler() == handler); 064 } 065 066 @Override 067 public boolean subscribe(@Nonnull MessageHandler theHandler) { 068 return super.subscribe(new RetryingMessageHandlerWrapper(theHandler, getName())); 069 } 070 071 @Override 072 public boolean unsubscribe(@Nonnull MessageHandler handler) { 073 Optional<RetryingMessageHandlerWrapper> match = getSubscribers().stream() 074 .map(t -> (RetryingMessageHandlerWrapper) t) 075 .filter(t -> t.getWrappedHandler() == handler) 076 .findFirst(); 077 match.ifPresent(super::unsubscribe); 078 return match.isPresent(); 079 } 080 081 @Override 082 public void destroy() { 083 // nothing 084 } 085 086 /** 087 * Creates a synchronous channel, mostly intended for testing 088 */ 089 public static LinkedBlockingChannel newSynchronous(String theName) { 090 return new LinkedBlockingChannel(theName, null, () -> 0); 091 } 092}