package au.gov.amsa.risky.format; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.nio.ByteBuffer; import java.util.concurrent.atomic.AtomicBoolean; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import com.google.common.base.Optional; import rx.Observable; import rx.Observable.OnSubscribe; import rx.Subscriber; import rx.Subscription; public class BinaryFixesOnSubscribeFastPath implements OnSubscribe<Fix> { private static final Logger log = LoggerFactory.getLogger(BinaryFixesOnSubscribeFastPath.class); private final File file; private final BinaryFixesFormat format; public BinaryFixesOnSubscribeFastPath(File file, BinaryFixesFormat format) { this.file = file; this.format = format; } public static Observable<Fix> from(File file, BinaryFixesFormat format) { return Observable.create(new BinaryFixesOnSubscribeFastPath(file, format)); } @Override public void call(Subscriber<? super Fix> subscriber) { FileInputStream fis = null; AtomicBoolean closed = new AtomicBoolean(false); try { fis = new FileInputStream(file); subscriber.add(createSubscription(fis, closed)); Optional<Integer> mmsi; if (format == BinaryFixesFormat.WITH_MMSI) mmsi = Optional.absent(); else mmsi = Optional.of(BinaryFixesUtil.getMmsi(file)); reportFixes(mmsi, subscriber, fis, format); if (!subscriber.isUnsubscribed()) { // eagerly close if (closed.compareAndSet(false, true)) fis.close(); subscriber.onCompleted(); } } catch (Exception e) { if (!subscriber.isUnsubscribed()) subscriber.onError(e); } } private Subscription createSubscription(final FileInputStream fis, final AtomicBoolean closed) { return new Subscription() { volatile boolean subscribed = true; @Override public void unsubscribe() { subscribed = false; try { if (closed.compareAndSet(false, true)) fis.close(); } catch (IOException e) { e.printStackTrace(); } } @Override public boolean isUnsubscribed() { return !subscribed; } }; } private static void reportFixes(Optional<Integer> mmsi, Subscriber<? super Fix> subscriber, InputStream fis, BinaryFixesFormat format) throws IOException { int recordSize = BinaryFixes.recordSize(format); byte[] bytes = new byte[4096 * recordSize]; int length = 0; if (subscriber.isUnsubscribed()) return; while ((length = fis.read(bytes)) > 0) { for (int i = 0; i < length; i += recordSize) { if (subscriber.isUnsubscribed()) return; ByteBuffer bb = ByteBuffer.wrap(bytes, i, recordSize); int m; if (mmsi.isPresent()) { m = mmsi.get(); } else { m = bb.getInt(); } Fix fix = null; try { fix = BinaryFixesUtil.toFix(m, bb); } catch (RuntimeException e) { log.warn(e.getMessage()); } if (fix != null) subscriber.onNext(fix); } } } public static void main(String[] args) { System.out.println(Integer.MAX_VALUE); } }