/*
* 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.notificationq;
import java.io.IOException;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import org.joda.time.DateTime;
import org.skife.jdbi.v2.Handle;
import org.skife.jdbi.v2.IDBI;
import org.skife.jdbi.v2.tweak.HandleCallback;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.ning.billing.util.Hostname;
import com.ning.billing.util.callcontext.InternalCallContext;
import com.ning.billing.util.entity.dao.EntitySqlDao;
import com.ning.billing.util.entity.dao.EntitySqlDaoWrapperFactory;
import com.ning.billing.util.notificationq.NotificationQueueService.NotificationQueueHandler;
import com.ning.billing.util.notificationq.dao.NotificationSqlDao;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.google.common.collect.ImmutableList;
public class DefaultNotificationQueue implements NotificationQueue {
private static final Logger log = LoggerFactory.getLogger(DefaultNotificationQueue.class);
private final IDBI dbi;
private final NotificationSqlDao dao;
private final String hostname;
private final String svcName;
private final String queueName;
private final ObjectMapper objectMapper;
private final NotificationQueueHandler handler;
private final NotificationQueueService notificationQueueService;
private volatile boolean isStarted;
public DefaultNotificationQueue(final String svcName, final String queueName, final NotificationQueueHandler handler,
final IDBI dbi, final NotificationQueueService notificationQueueService) {
this.svcName = svcName;
this.queueName = queueName;
this.handler = handler;
this.dbi = dbi;
this.dao = dbi.onDemand(NotificationSqlDao.class);
this.hostname = Hostname.get();
this.notificationQueueService = notificationQueueService;
this.objectMapper = new ObjectMapper();
}
@Override
public void recordFutureNotification(final DateTime futureNotificationTime,
final NotificationKey notificationKey,
final InternalCallContext context) throws IOException {
recordFutureNotificationInternal(futureNotificationTime, notificationKey, dao, context);
}
@Override
public void recordFutureNotificationFromTransaction(final EntitySqlDaoWrapperFactory<EntitySqlDao> transactionalDao,
final DateTime futureNotificationTime,
final NotificationKey notificationKey,
final InternalCallContext context) throws IOException {
final NotificationSqlDao transactionalNotificationDao = transactionalDao.transmogrify(NotificationSqlDao.class);
recordFutureNotificationInternal(futureNotificationTime, notificationKey, transactionalNotificationDao, context);
}
private void recordFutureNotificationInternal(final DateTime futureNotificationTime,
final NotificationKey notificationKey,
final NotificationSqlDao thisDao,
final InternalCallContext context) throws IOException {
final String json = objectMapper.writeValueAsString(notificationKey);
final UUID futureUserToken = UUID.randomUUID();
final Notification notification = new DefaultNotification(getFullQName(), hostname, notificationKey.getClass().getName(), json,
context.getUserToken(), futureUserToken, futureNotificationTime, context.getAccountRecordId(), context.getTenantRecordId());
thisDao.insertNotification(notification, context);
}
@Override
public void removeNotificationsByKey(final NotificationKey notificationKey, final InternalCallContext context) {
dao.removeNotificationsByKey(notificationKey.toString(), context);
}
@Override
public List<Notification> getNotificationForAccountAndDate(final UUID accountId, final DateTime effectiveDate, final InternalCallContext context) {
// TODO we have the same use case in InternalCallContextFactory, do we need some sort of helper class?
final Long accountRecordId = dbi.withHandle(new HandleCallback<Long>() {
@Override
public Long withHandle(final Handle handle) throws Exception {
final List<Map<String, Object>> values = handle.select("select record_id from accounts where id = " + accountId.toString());
if (values.size() == 0) {
return null;
} else {
return (Long) values.get(0).get("record_id");
}
}
});
if (accountId == null) {
return ImmutableList.<Notification>of();
} else {
return dao.getNotificationForAccountAndDate(accountRecordId, effectiveDate.toDate(), context);
}
}
@Override
public void removeNotification(final UUID notificationId, final InternalCallContext context) {
dao.removeNotification(notificationId.toString(), context);
}
@Override
public String getFullQName() {
return NotificationQueueServiceBase.getCompositeName(svcName, queueName);
}
@Override
public String getServiceName() {
return svcName;
}
@Override
public String getQueueName() {
return queueName;
}
@Override
public NotificationQueueHandler getHandler() {
return handler;
}
@Override
public void startQueue() {
notificationQueueService.startQueue();
isStarted = true;
}
@Override
public void stopQueue() {
// Order matters...
isStarted = false;
notificationQueueService.stopQueue();
}
@Override
public boolean isStarted() {
return isStarted;
}
}