TestEventBusBase.java

240 lines | 8.159 kB Blame History Raw Download
/*
 * Copyright 2010-2011 Ning, Inc.
 *
 * Ning 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 com.ning.billing.util.bus;

import java.util.UUID;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.testng.Assert;
import org.testng.annotations.AfterClass;
import org.testng.annotations.BeforeClass;

import com.ning.billing.util.UtilTestSuiteWithEmbeddedDB;
import com.ning.billing.util.callcontext.InternalCallContext;
import com.ning.billing.util.events.BusInternalEvent;
import com.ning.billing.util.events.BusInternalEvent.BusInternalEventType;
import com.ning.billing.util.events.DefaultBusInternalEvent;
import com.ning.billing.util.svcsapi.bus.InternalBus;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.google.common.eventbus.Subscribe;
import com.google.inject.Inject;

public class TestEventBusBase {

    protected static final Logger log = LoggerFactory.getLogger(TestEventBusBase.class);

    private final InternalBus eventBus;
    private final InternalCallContext internalCallContext;

    public TestEventBusBase(final InternalBus eventBus, final InternalCallContext internalCallContext) {
        this.eventBus = eventBus;
        this.internalCallContext = internalCallContext;
    }

    public static class MyEvent extends DefaultBusInternalEvent implements BusInternalEvent {
        private final String name;
        private final Long value;
        private final String type;

        @JsonCreator
        public MyEvent(@JsonProperty("name") final String name,
                       @JsonProperty("value") final Long value,
                       @JsonProperty("token") final UUID token,
                       @JsonProperty("type") final String type,
                       @JsonProperty("accountRecordId") final Long accountRecordId,
                       @JsonProperty("tenantRecordId") final Long tenantRecordId) {
            super(token, accountRecordId, tenantRecordId);
            this.name = name;
            this.value = value;
            this.type = type;
        }

        @JsonIgnore
        @Override
        public BusInternalEventType getBusEventType() {
            return BusInternalEventType.valueOf(type);
        }

        public String getName() {
            return name;
        }

        public Long getValue() {
            return value;
        }

        public String getType() {
            return type;
        }
    }

    public static final class MyEventWithException extends MyEvent {

        @JsonCreator
        public MyEventWithException(@JsonProperty("name") final String name,
                                    @JsonProperty("value") final Long value,
                                    @JsonProperty("token") final UUID token,
                                    @JsonProperty("type") final String type,
                                    @JsonProperty("accountRecordId") final Long accountRecordId,
                                    @JsonProperty("tenantRecordId") final Long tenantRecordId) {
            super(name, value, token, type, accountRecordId, tenantRecordId);
        }
    }

    public static final class MyOtherEvent extends DefaultBusInternalEvent implements BusInternalEvent {

        private final String name;
        private final Double value;
        private final String type;

        @JsonCreator
        public MyOtherEvent(@JsonProperty("name") final String name,
                            @JsonProperty("value") final Double value,
                            @JsonProperty("token") final UUID token,
                            @JsonProperty("type") final String type,
                            @JsonProperty("accountRecordId") final Long accountRecordId,
                            @JsonProperty("tenantRecordId") final Long tenantRecordId) {
            super(token, accountRecordId, tenantRecordId);
            this.name = name;
            this.value = value;
            this.type = type;
        }

        @JsonIgnore
        @Override
        public BusInternalEventType getBusEventType() {
            return BusInternalEventType.valueOf(type);
        }


        public String getName() {
            return name;
        }

        public Double getValue() {
            return value;
        }

        public String getType() {
            return type;
        }
    }

    public static class MyEventHandlerException extends RuntimeException {

        private static final long serialVersionUID = 156337823L;

        public MyEventHandlerException(final String msg) {
            super(msg);
        }
    }

    public static class MyEventHandler {

        private final int expectedEvents;

        private volatile int gotEvents;

        public MyEventHandler(final int exp) {
            this.expectedEvents = exp;
            this.gotEvents = 0;
        }

        public synchronized int getEvents() {
            return gotEvents;
        }

        @Subscribe
        public synchronized void processEvent(final MyEvent event) {
            gotEvents++;
            //log.debug("Got event {} {}", event.name, event.value);
        }

        @Subscribe
        public synchronized void processEvent(final MyEventWithException event) {
            throw new MyEventHandlerException("FAIL");
        }

        public synchronized boolean waitForCompletion(final long timeoutMs) {

            final long ini = System.currentTimeMillis();
            long remaining = timeoutMs;
            while (gotEvents < expectedEvents && remaining > 0) {
                try {
                    wait(1000);
                    if (gotEvents == expectedEvents) {
                        break;
                    }
                    remaining = timeoutMs - (System.currentTimeMillis() - ini);
                } catch (InterruptedException ignore) {
                }
            }
            return (gotEvents == expectedEvents);
        }
    }

    public void testSimpleWithException() {
        try {
            final MyEventHandler handler = new MyEventHandler(1);
            eventBus.register(handler);

            eventBus.post(new MyEventWithException("my-event", 1L, UUID.randomUUID(), BusInternalEventType.ACCOUNT_CHANGE.toString(), 1L, 1L), internalCallContext);

            Thread.sleep(50000);
        } catch (Exception ignored) {
        }
    }

    public void testSimple() {
        try {
            final int nbEvents = 5;
            final MyEventHandler handler = new MyEventHandler(nbEvents);
            eventBus.register(handler);

            for (int i = 0; i < nbEvents; i++) {
                eventBus.post(new MyEvent("my-event", (long) i, UUID.randomUUID(), BusInternalEventType.ACCOUNT_CHANGE.toString(), 1L, 1L), internalCallContext);
            }

            final boolean completed = handler.waitForCompletion(10000);
            Assert.assertEquals(completed, true);
        } catch (Exception e) {
            Assert.fail("", e);
        }
    }

    public void testDifferentType() {
        try {
            final MyEventHandler handler = new MyEventHandler(1);
            eventBus.register(handler);

            for (int i = 0; i < 5; i++) {
                eventBus.post(new MyOtherEvent("my-other-event", (double) i, UUID.randomUUID(), BusInternalEventType.BUNDLE_REPAIR.toString(), 1L, 1L), internalCallContext);
            }
            eventBus.post(new MyEvent("my-event", 11l, UUID.randomUUID(), BusInternalEventType.ACCOUNT_CHANGE.toString(), 1L, 1L), internalCallContext);

            final boolean completed = handler.waitForCompletion(10000);
            Assert.assertEquals(completed, true);
        } catch (Exception e) {
            Assert.fail("", e);
        }
    }
}