From 886e6e71e79d5111395636481cd30c2c4f95fbf8 Mon Sep 17 00:00:00 2001 From: blavenie <benoit.lavenier@e-is.pro> Date: Mon, 10 Apr 2017 22:23:51 +0200 Subject: [PATCH] ES: emai subscription: now remember last execution time, to filter on latest events --- .../elasticsearch/client/Duniter4jClient.java | 3 + .../client/Duniter4jClientImpl.java | 30 ++- .../elasticsearch/dao/AbstractDao.java | 3 +- .../dao/AbstractIndexTypeDao.java | 19 ++ .../elasticsearch/dao/impl/BlockDaoImpl.java | 36 +--- .../service/BlockchainService.java | 22 +- .../service/CurrencyService.java | 2 + .../subscription/PluginSettings.java | 14 +- .../subscription/dao/DaoModule.java | 3 + .../dao/SubscriptionIndexTypeDao.java | 4 + .../execution/SubscriptionExecutionDao.java | 23 +++ .../SubscriptionExecutionDaoImpl.java | 168 +++++++++++++++ .../dao/record/SubscriptionRecordDao.java | 4 +- .../dao/record/SubscriptionRecordDaoImpl.java | 13 +- .../model/SubscriptionExecution.java | 78 +++++++ ...scription.java => SubscriptionRecord.java} | 2 +- .../model/email/EmailSubscription.java | 19 +- .../service/SubscriptionService.java | 192 +++++++++++------- .../subscription/util/DateUtils.java | 49 +++++ .../util/stringtemplate/I18nRenderer.java | 24 --- .../util/stringtemplate/StringRenderer.java | 32 +++ ...duniter4j-es-subscription_en_GB.properties | 13 +- ...duniter4j-es-subscription_fr_FR.properties | 17 +- .../main/resources/templates/cesium_logo.st | 6 + .../resources/templates/html_email_content.st | 103 +++++++--- .../resources/templates/html_event_item.st | 10 + .../main/resources/templates/text_email.st | 10 +- .../resources/templates/text_event_item.st | 2 +- .../service/SubscriptionServiceTest.java | 2 +- .../service/SubscriptionTemplateTest.java | 54 ++--- .../user/service/UserEventService.java | 2 + 31 files changed, 719 insertions(+), 240 deletions(-) create mode 100644 duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java create mode 100644 duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java create mode 100644 duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java rename duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/{Subscription.java => SubscriptionRecord.java} (98%) create mode 100644 duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/DateUtils.java delete mode 100644 duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java create mode 100644 duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java create mode 100644 duniter4j-es-subscription/src/main/resources/templates/cesium_logo.st create mode 100644 duniter4j-es-subscription/src/main/resources/templates/html_event_item.st diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java index cdaea5b3..bb2a7705 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java @@ -4,6 +4,7 @@ import org.duniter.core.beans.Bean; import org.duniter.core.client.model.local.LocalEntity; import org.duniter.elasticsearch.dao.handler.StringReaderHandler; import org.elasticsearch.ElasticsearchException; +import org.elasticsearch.action.ActionRequestBuilder; import org.elasticsearch.action.bulk.BulkRequestBuilder; import org.elasticsearch.client.Client; import org.elasticsearch.search.SearchHit; @@ -73,4 +74,6 @@ public interface Duniter4jClient extends Bean, Client { void bulkFromStream(InputStream is, String indexName, String indexType, StringReaderHandler handler); void flushDeleteBulk(final String index, final String type, BulkRequestBuilder bulkRequest); + + void safeExecuteRequest(ActionRequestBuilder<?, ?, ?> request, boolean wait); } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java index 54942a82..1a157ea1 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java @@ -94,6 +94,7 @@ import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.logging.ESLogger; import org.elasticsearch.common.logging.Loggers; import org.elasticsearch.common.settings.Settings; +import org.elasticsearch.common.util.concurrent.EsRejectedExecutionException; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.search.SearchHit; import org.elasticsearch.search.SearchHitField; @@ -152,10 +153,9 @@ public class Duniter4jClientImpl implements Duniter4jClient { @Override public void updateDocumentFromJson(String index, String type, String id, String json) { // Execute indexBlocksFromNode - client.prepareUpdate(index, type, id) + safeExecuteRequest(client.prepareUpdate(index, type, id) .setRefresh(true) - .setDoc(json) - .execute().actionGet(); + .setDoc(json), true); } @Override @@ -963,4 +963,28 @@ public class Duniter4jClientImpl implements Duniter4jClient { public void close() { client.close(); } + + public void safeExecuteRequest(ActionRequestBuilder<?, ?, ?> request, boolean wait) { + // Execute in a pool + if (!wait) { + boolean acceptedInPool = false; + while(!acceptedInPool) + try { + request.execute(); + acceptedInPool = true; + } + catch(EsRejectedExecutionException e) { + // not accepted, so wait + try { + Thread.sleep(1000); // 1s + } + catch(InterruptedException e2) { + // silent + } + } + + } else { + request.execute().actionGet(); + } + } } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java index 0fe773e9..08e21ed6 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java @@ -39,6 +39,7 @@ import org.duniter.elasticsearch.dao.handler.StringReaderHandler; import org.duniter.elasticsearch.exception.AccessDeniedException; import org.duniter.elasticsearch.exception.NotFoundException; import org.elasticsearch.ElasticsearchException; +import org.elasticsearch.action.ActionRequestBuilder; import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequestBuilder; import org.elasticsearch.action.admin.indices.exists.indices.IndicesExistsRequestBuilder; import org.elasticsearch.action.admin.indices.exists.indices.IndicesExistsResponse; @@ -58,6 +59,7 @@ import org.elasticsearch.common.bytes.BytesArray; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.logging.ESLogger; import org.elasticsearch.common.logging.Loggers; +import org.elasticsearch.common.util.concurrent.EsRejectedExecutionException; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.search.SearchHit; import org.elasticsearch.search.SearchHitField; @@ -101,5 +103,4 @@ public abstract class AbstractDao implements Bean { /* -- protected methods -- */ - } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java index 4f1bf5f6..684950e2 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java @@ -25,6 +25,7 @@ package org.duniter.elasticsearch.dao; import com.fasterxml.jackson.core.JsonProcessingException; import org.duniter.core.exception.TechnicalException; +import org.duniter.core.util.Preconditions; import org.duniter.elasticsearch.dao.handler.StringReaderHandler; import org.elasticsearch.action.bulk.BulkRequestBuilder; @@ -175,4 +176,22 @@ public abstract class AbstractIndexTypeDao<T extends IndexTypeDao> extends Abstr public boolean existsIndex() { return client.existsIndex(index); } + + public void create(String json, boolean wait) { + Preconditions.checkNotNull(json); + + // Execute + client.safeExecuteRequest(client.prepareIndex(getIndex(), getType()) + .setRefresh(false) // let's see if this works + .setSource(json), wait); + } + + public void update(String id, String json, boolean wait) { + Preconditions.checkNotNull(json); + + // Execute + client.safeExecuteRequest(client.prepareUpdate(getIndex(), getType(), id) + .setRefresh(false) // let's see if this works + .setDoc(json), wait); + } } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java index 2964b8da..0520d7b5 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java @@ -30,18 +30,14 @@ import org.duniter.core.exception.TechnicalException; import org.duniter.core.util.Preconditions; import org.duniter.core.util.StringUtils; import org.duniter.core.util.json.JsonSyntaxException; -import org.duniter.elasticsearch.PluginSettings; import org.duniter.elasticsearch.dao.AbstractDao; import org.duniter.elasticsearch.dao.BlockDao; -import org.elasticsearch.action.ActionRequestBuilder; import org.elasticsearch.action.bulk.BulkRequestBuilder; import org.elasticsearch.action.index.IndexRequestBuilder; import org.elasticsearch.action.search.SearchRequestBuilder; import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.search.SearchType; import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.util.concurrent.EsRejectedExecutionException; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.index.query.QueryBuilders; @@ -88,7 +84,7 @@ public class BlockDaoImpl extends AbstractDao implements BlockDao { .setSource(json); // Execute - safeExecute(request, wait); + client.safeExecuteRequest(request, wait); } catch(JsonProcessingException e) { throw new TechnicalException(e); @@ -114,7 +110,7 @@ public class BlockDaoImpl extends AbstractDao implements BlockDao { .setSource(json); // Execute - safeExecute(request, wait); + client.safeExecuteRequest(request, wait); } public boolean isExists(String currencyName, String id) { @@ -138,7 +134,7 @@ public class BlockDaoImpl extends AbstractDao implements BlockDao { .setDoc(json); // Execute - safeExecute(request, wait); + client.safeExecuteRequest(request, wait); } catch(JsonProcessingException e) { throw new TechnicalException(e); @@ -162,7 +158,7 @@ public class BlockDaoImpl extends AbstractDao implements BlockDao { .setDoc(json); // Execute - safeExecute(request, wait); + client.safeExecuteRequest(request, wait); } public List<BlockchainBlock> findBlocksByHash(String currencyName, String query) { @@ -361,28 +357,4 @@ public class BlockDaoImpl extends AbstractDao implements BlockDao { return result; } - - protected void safeExecute(ActionRequestBuilder<?, ?, ?> request, boolean wait) { - // Execute in a pool - if (!wait) { - boolean acceptedInPool = false; - while(!acceptedInPool) - try { - request.execute(); - acceptedInPool = true; - } - catch(EsRejectedExecutionException e) { - // not accepted, so wait - try { - Thread.sleep(1000); // 1s - } - catch(InterruptedException e2) { - // silent - } - } - - } else { - request.execute().actionGet(); - } - } } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java index b1934f16..25b2134c 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java @@ -48,13 +48,12 @@ import org.duniter.core.util.websocket.WebsocketClientEndpoint; import org.duniter.elasticsearch.PluginSettings; import org.duniter.elasticsearch.client.Duniter4jClient; import org.duniter.elasticsearch.dao.BlockDao; -import org.duniter.elasticsearch.dao.impl.BlockDaoImpl; import org.duniter.elasticsearch.exception.DuplicateIndexIdException; +import org.duniter.elasticsearch.exception.NotFoundException; import org.duniter.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.action.bulk.BulkItemResponse; import org.elasticsearch.action.bulk.BulkRequestBuilder; import org.elasticsearch.action.bulk.BulkResponse; -import org.elasticsearch.client.Client; import org.elasticsearch.common.inject.Inject; import org.nuiton.i18n.I18n; @@ -85,7 +84,6 @@ public class BlockchainService extends AbstractService { private final JsonAttributeParser blockHashParser = new JsonAttributeParser("hash"); private final JsonAttributeParser blockPreviousHashParser = new JsonAttributeParser("previousHash"); - private Client client; private BlockDao blockDao; @Inject @@ -401,8 +399,26 @@ public class BlockchainService extends AbstractService { return blockDao.getBlockById(currencyName, CURRENT_BLOCK_ID); } + public Map<String, Object> getBlockFieldsById(String currencyName, int number, String... fields) { + return getBlockFieldsById(currencyName, String.valueOf(number), fields); + } + + public Map<String, Object> getCurrentBlockFields(String currencyName, String... fields) { + return getBlockFieldsById(currencyName, CURRENT_BLOCK_ID, fields); + } + /* -- Internal methods -- */ + + protected Map<String, Object> getBlockFieldsById(String currency, String blockId, String... fields) { + try { + return client.getMandatoryFieldsById(currency, BLOCK_TYPE, blockId, fields); + } + catch(NotFoundException e) { + throw new BlockNotFoundException(e); + } + } + protected Collection<String> indexBlocksNoBulk(Peer peer, String currencyName, int firstNumber, int lastNumber, ProgressionModel progressionModel) { Set<String> missingBlockNumbers = new LinkedHashSet<>(); diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java index f0cdcd62..1594e65f 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java @@ -230,6 +230,8 @@ public class CurrencyService extends AbstractService { saveCurrency(currency, pubkey); } + + /* -- Internal methods -- */ diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java index 4151475a..f6e36663 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java @@ -84,11 +84,19 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { } /** - * Time interval (millisecond) to send email ? (default: 3600000 = 1h) + * Day of the week to trigger weekly email subscription (default: 1) * @return */ - public long getExecuteEmailSubscriptionsInterval() { - return settings.getAsLong("duniter.subscription.email.interval", 36000000l) /*1hour*/; + public int getEmailSubscriptionsExecuteDayOfWeek() { + return settings.getAsInt("duniter.subscription.email.dayOfWeek", 1); + } + + /** + * Hour in day to trigger daily email subscription (default: 4 AM) + * @return + */ + public int getEmailSubscriptionsExecuteHour() { + return settings.getAsInt("duniter.subscription.email.hourOfDay", 4) /*4 hour in the morning*/; } /* -- delegate methods -- */ diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java index 7851afbe..bbf87958 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java @@ -22,6 +22,8 @@ package org.duniter.elasticsearch.subscription.dao; * #L% */ +import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDao; +import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDaoImpl; import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDaoImpl; import org.elasticsearch.common.inject.AbstractModule; @@ -36,6 +38,7 @@ public class DaoModule extends AbstractModule implements Module { // Subscription types bind(SubscriptionRecordDao.class).to(SubscriptionRecordDaoImpl.class).asEagerSingleton(); + bind(SubscriptionExecutionDao.class).to(SubscriptionExecutionDaoImpl.class).asEagerSingleton(); } } \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java index 1b3637b6..d538eb37 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java @@ -32,8 +32,12 @@ public interface SubscriptionIndexTypeDao<T extends SubscriptionIndexTypeDao> ex String create(final String json); + void create(final String json, boolean wait); + void update(final String id, final String json); + void update(final String id, final String json, boolean wait); + void checkSameDocumentIssuer(String id, String expectedIssuer); } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java new file mode 100644 index 00000000..f957532c --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java @@ -0,0 +1,23 @@ +package org.duniter.elasticsearch.subscription.dao.execution; + +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexTypeDao; +import org.duniter.elasticsearch.subscription.model.SubscriptionExecution; +import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; + +import java.util.List; + +/** + * Created by blavenie on 03/04/17. + */ +public interface SubscriptionExecutionDao<T extends SubscriptionIndexTypeDao> extends SubscriptionIndexTypeDao<T> { + + String TYPE = "execution"; + + SubscriptionExecution getLastExecution(SubscriptionRecord record); + + SubscriptionExecution getLastExecution(String recipient, String subscriptionType, String recordId); + + Long getLastExecutionTime(String recipient, String subscriptionType, String recordId); + + Long getLastExecutionTime(SubscriptionRecord record); +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java new file mode 100644 index 00000000..f38d6d28 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java @@ -0,0 +1,168 @@ +package org.duniter.elasticsearch.subscription.dao.execution; + +import com.fasterxml.jackson.core.JsonProcessingException; +import org.duniter.core.client.model.bma.BlockchainBlock; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.util.CollectionUtils; +import org.duniter.core.util.Preconditions; +import org.duniter.core.util.StringUtils; +import org.duniter.elasticsearch.subscription.PluginSettings; +import org.duniter.elasticsearch.subscription.dao.AbstractSubscriptionIndexTypeDao; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; +import org.duniter.elasticsearch.subscription.model.SubscriptionExecution; +import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; +import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; +import org.elasticsearch.action.index.IndexRequestBuilder; +import org.elasticsearch.action.search.SearchResponse; +import org.elasticsearch.action.search.SearchType; +import org.elasticsearch.action.update.UpdateRequestBuilder; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.xcontent.XContentBuilder; +import org.elasticsearch.common.xcontent.XContentFactory; +import org.elasticsearch.index.query.BoolQueryBuilder; +import org.elasticsearch.index.query.QueryBuilders; +import org.elasticsearch.search.SearchHit; +import org.elasticsearch.search.sort.SortOrder; + +import java.io.IOException; +import java.util.Arrays; +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +/** + * Created by blavenie on 03/04/17. + */ +public class SubscriptionExecutionDaoImpl extends AbstractSubscriptionIndexTypeDao<SubscriptionExecutionDaoImpl> implements SubscriptionExecutionDao<SubscriptionExecutionDaoImpl> { + + @Inject + public SubscriptionExecutionDaoImpl(PluginSettings pluginSettings, SubscriptionIndexDao indexDao) { + super(SubscriptionIndexDao.INDEX, TYPE, pluginSettings); + + indexDao.register(this); + } + + @Override + public SubscriptionExecution getLastExecution(SubscriptionRecord record) { + Preconditions.checkNotNull(record); + Preconditions.checkNotNull(record.getIssuer()); + Preconditions.checkNotNull(record.getType()); + Preconditions.checkNotNull(record.getId()); + + return getLastExecution(record.getIssuer(), record.getType(), record.getId()); + } + + @Override + public SubscriptionExecution getLastExecution(String recipient, String recordType, String recordId) { + + BoolQueryBuilder query = QueryBuilders.boolQuery() + .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECIPIENT, recipient)) + .must(QueryBuilders.termsQuery(SubscriptionExecution.PROPERTY_RECORD_TYPE, recordType)) + .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECORD_ID, recordId)); + + SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) + .setTypes(SubscriptionExecutionDao.TYPE) + .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) + .setQuery(query) + .setFetchSource(true) + .setFrom(0).setSize(1) + .addSort(SubscriptionExecution.PROPERTY_TIME, SortOrder.DESC) + .get(); + + if (response.getHits().getTotalHits() == 0) return null; + + SearchHit hit = response.getHits().getHits()[0]; + return client.readSourceOrNull(hit, SubscriptionExecution.class); + } + + @Override + public Long getLastExecutionTime(SubscriptionRecord record) { + Preconditions.checkNotNull(record); + Preconditions.checkNotNull(record.getIssuer()); + Preconditions.checkNotNull(record.getType()); + Preconditions.checkNotNull(record.getId()); + + return getLastExecutionTime(record.getIssuer(), record.getType(), record.getId()); + } + + @Override + public Long getLastExecutionTime(String recipient, String recordType, String recordId) { + + BoolQueryBuilder query = QueryBuilders.boolQuery() + .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECIPIENT, recipient)) + .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECORD_ID, recordId)) + .must(QueryBuilders.termsQuery(SubscriptionExecution.PROPERTY_RECORD_ID, recordType)); + + SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) + .setTypes(SubscriptionExecutionDao.TYPE) + .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) + .setQuery(query) + .addField(SubscriptionExecution.PROPERTY_TIME) + .setFrom(0).setSize(1) + .addSort(SubscriptionExecution.PROPERTY_TIME, SortOrder.DESC) + .get(); + + if (response.getHits().getTotalHits() == 0) return null; + SearchHit hit = response.getHits().getHits()[0]; + return hit.field(SubscriptionExecution.PROPERTY_TIME).getValue(); + } + + @Override + public XContentBuilder createTypeMapping() { + try { + XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() + .startObject(getType()) + .startObject("properties") + + // issuer + .startObject("issuer") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // recipient + .startObject("recipient") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // record type + .startObject("recordType") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // record id + .startObject("recordId") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // time + .startObject("time") + .field("type", "integer") + .endObject() + + // hash + .startObject("hash") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // signature + .startObject("signature") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + .endObject() + .endObject().endObject(); + + return mapping; + } + catch(IOException ioe) { + throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); + } + } + +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java index b8b2703c..d1fa0b80 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java @@ -1,7 +1,7 @@ package org.duniter.elasticsearch.subscription.dao.record; import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexTypeDao; -import org.duniter.elasticsearch.subscription.model.Subscription; +import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; import java.util.List; @@ -12,5 +12,5 @@ public interface SubscriptionRecordDao<T extends SubscriptionIndexTypeDao> exten String TYPE = "record"; - List<Subscription> getSubscriptions(int from, int size, String recipient, String... types); + List<SubscriptionRecord> getSubscriptions(int from, int size, String recipient, String... types); } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java index 983ae405..f3682afc 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java @@ -5,7 +5,7 @@ import org.duniter.core.util.CollectionUtils; import org.duniter.elasticsearch.subscription.PluginSettings; import org.duniter.elasticsearch.subscription.dao.AbstractSubscriptionIndexTypeDao; import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.model.Subscription; +import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.search.SearchType; @@ -13,7 +13,6 @@ import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilder; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.search.SearchHit; @@ -36,12 +35,12 @@ public class SubscriptionRecordDaoImpl extends AbstractSubscriptionIndexTypeDao< } @Override - public List<Subscription> getSubscriptions(int from, int size, String recipient, String... types) { + public List<SubscriptionRecord> getSubscriptions(int from, int size, String recipient, String... types) { BoolQueryBuilder query = QueryBuilders.boolQuery() - .must(QueryBuilders.termQuery(Subscription.PROPERTY_RECIPIENT, recipient)); + .must(QueryBuilders.termQuery(SubscriptionRecord.PROPERTY_RECIPIENT, recipient)); if (CollectionUtils.isNotEmpty(types)) { - query.must(QueryBuilders.termsQuery(Subscription.PROPERTY_TYPE, types)); + query.must(QueryBuilders.termsQuery(SubscriptionRecord.PROPERTY_TYPE, types)); } SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) @@ -123,9 +122,9 @@ public class SubscriptionRecordDaoImpl extends AbstractSubscriptionIndexTypeDao< } } - protected Subscription toSubscription(SearchHit searchHit) { + protected SubscriptionRecord toSubscription(SearchHit searchHit) { - Subscription record = null; + SubscriptionRecord record = null; if (SubscriptionRecordDao.TYPE.equals(searchHit.getType())) { record = client.readSourceOrNull(searchHit, EmailSubscription.class); diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java new file mode 100644 index 00000000..63cd57f4 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java @@ -0,0 +1,78 @@ +package org.duniter.elasticsearch.subscription.model; + +/* + * #%L + * Duniter4j :: ElasticSearch GChange plugin + * %% + * Copyright (C) 2014 - 2017 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + +import com.fasterxml.jackson.annotation.JsonIgnore; +import org.duniter.core.client.model.elasticsearch.Record; + +/** + * Created by blavenie on 01/12/16. + */ +public class SubscriptionExecution extends Record { + + public static final String PROPERTY_RECIPIENT = "recipient"; + + public static final String PROPERTY_RECORD_TYPE = "recordType"; + + public static final String PROPERTY_RECORD_ID = "recordId"; + + private String recipient; + private String recordType; + private String recordId; + + private SubscriptionRecord record; + + public String getRecipient() { + return recipient; + } + + public void setRecipient(String recipient) { + this.recipient = recipient; + } + + public String getRecordType() { + return recordType; + } + + public void setRecordType(String recordType) { + this.recordType = recordType; + } + + public String getRecordId() { + return recordId; + } + + public void setRecordId(String recordId) { + this.recordId = recordId; + } + + @JsonIgnore + public SubscriptionRecord getRecord() { + return record; + } + + @JsonIgnore + public void setRecord(SubscriptionRecord record) { + this.record = record; + } +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Subscription.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionRecord.java similarity index 98% rename from duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Subscription.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionRecord.java index 2ffc884e..6a5081cc 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Subscription.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionRecord.java @@ -28,7 +28,7 @@ import org.duniter.core.client.model.elasticsearch.Record; /** * Created by blavenie on 01/12/16. */ -public class Subscription<T> extends Record{ +public class SubscriptionRecord<T> extends Record{ public static final String PROPERTY_RECIPIENT = "recipient"; diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java index 663297df..781c7778 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java @@ -22,12 +22,12 @@ package org.duniter.elasticsearch.subscription.model.email; * #L% */ -import org.duniter.elasticsearch.subscription.model.Subscription; +import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; /** * Created by blavenie on 01/12/16. */ -public class EmailSubscription extends Subscription<EmailSubscription.Content> { +public class EmailSubscription extends SubscriptionRecord<EmailSubscription.Content> { public static final String TYPE = "email"; @@ -35,9 +35,15 @@ public class EmailSubscription extends Subscription<EmailSubscription.Content> { return new EmailSubscription.Content(); } + public enum Frequency { + daily, + weekly + } + public static class Content { public static final String PROPERTY_EMAIL = "email"; + public static final String PROPERTY_FREQUENCY = "frequency"; public static final String PROPERTY_LOCALE = "locale"; public static final String PROPERTY_INCLUDES = "includes"; public static final String PROPERTY_EXCLUDES = "excludes"; @@ -46,6 +52,7 @@ public class EmailSubscription extends Subscription<EmailSubscription.Content> { private String[] includes; private String[] excludes; private String locale; + private Frequency frequency; public String getEmail() { return email; @@ -78,6 +85,14 @@ public class EmailSubscription extends Subscription<EmailSubscription.Content> { public void setLocale(String locale) { this.locale = locale; } + + public Frequency getFrequency() { + return frequency; + } + + public void setFrequency(Frequency frequency) { + this.frequency = frequency; + } } } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java index 7571a7f6..18c31276 100644 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java @@ -37,11 +37,14 @@ import org.duniter.core.util.StringUtils; import org.duniter.core.util.crypto.CryptoUtils; import org.duniter.elasticsearch.client.Duniter4jClient; import org.duniter.elasticsearch.subscription.PluginSettings; +import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDao; import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; -import org.duniter.elasticsearch.subscription.model.Subscription; +import org.duniter.elasticsearch.subscription.model.SubscriptionExecution; +import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; +import org.duniter.elasticsearch.subscription.util.DateUtils; import org.duniter.elasticsearch.subscription.util.stringtemplate.DateRenderer; -import org.duniter.elasticsearch.subscription.util.stringtemplate.I18nRenderer; +import org.duniter.elasticsearch.subscription.util.stringtemplate.StringRenderer; import org.duniter.elasticsearch.threadpool.ThreadPool; import org.duniter.elasticsearch.user.model.UserEvent; import org.duniter.elasticsearch.user.service.AdminService; @@ -49,11 +52,13 @@ import org.duniter.elasticsearch.user.service.MailService; import org.duniter.elasticsearch.user.service.UserEventService; import org.duniter.elasticsearch.user.service.UserService; import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.unit.TimeValue; import org.nuiton.i18n.I18n; -import org.stringtemplate.v4.*; +import org.stringtemplate.v4.ST; +import org.stringtemplate.v4.STGroup; +import org.stringtemplate.v4.STGroupDir; import java.util.*; +import java.util.concurrent.TimeUnit; import java.util.stream.Collectors; /** @@ -62,6 +67,7 @@ import java.util.stream.Collectors; public class SubscriptionService extends AbstractService { private SubscriptionRecordDao subscriptionRecordDao; + private SubscriptionExecutionDao subscriptionExecutionDao; private ThreadPool threadPool; private MailService mailService; private AdminService adminService; @@ -74,6 +80,7 @@ public class SubscriptionService extends AbstractService { PluginSettings settings, CryptoService cryptoService, SubscriptionRecordDao subscriptionRecordDao, + SubscriptionExecutionDao subscriptionExecutionDao, ThreadPool threadPool, MailService mailService, AdminService adminService, @@ -81,6 +88,7 @@ public class SubscriptionService extends AbstractService { UserEventService userEventService) { super("subscription.service", client, settings, cryptoService); this.subscriptionRecordDao = subscriptionRecordDao; + this.subscriptionExecutionDao = subscriptionExecutionDao; this.threadPool = threadPool; this.mailService = mailService; this.adminService = adminService; @@ -123,14 +131,26 @@ public class SubscriptionService extends AbstractService { return this; } - threadPool.scheduleWithFixedDelay( - this::executeEmailSubscriptions, - new TimeValue(pluginSettings.getExecuteEmailSubscriptionsInterval())); + + // Startup Start + threadPool.schedule(() -> executeEmailSubscriptions(EmailSubscription.Frequency.daily)); + + // Daily execution + threadPool.scheduler().scheduleAtFixedRate( + () -> executeEmailSubscriptions(EmailSubscription.Frequency.daily), + DateUtils.delayBeforeHour(pluginSettings.getEmailSubscriptionsExecuteHour()), + 1, TimeUnit.DAYS); + + // Weekly execution + threadPool.scheduler().scheduleAtFixedRate( + () -> executeEmailSubscriptions(EmailSubscription.Frequency.weekly), + DateUtils.delayBeforeDayAndHour(pluginSettings.getEmailSubscriptionsExecuteDayOfWeek(), pluginSettings.getEmailSubscriptionsExecuteHour()), + 7, TimeUnit.DAYS); return this; } - public void executeEmailSubscriptions() { + public void executeEmailSubscriptions(final EmailSubscription.Frequency frequency) { final String senderPubkey = pluginSettings.getNodePubkey(); @@ -139,11 +159,11 @@ public class SubscriptionService extends AbstractService { boolean hasMore = true; while (hasMore) { - List<Subscription> subscriptions = subscriptionRecordDao.getSubscriptions(from, size, senderPubkey, EmailSubscription.TYPE); + List<SubscriptionRecord> subscriptions = subscriptionRecordDao.getSubscriptions(from, size, senderPubkey, EmailSubscription.TYPE); // Get profiles titles, for issuers and the sender Set<String> issuers = subscriptions.stream() - .map(Subscription::getIssuer) + .map(SubscriptionRecord::getIssuer) .distinct() .collect(Collectors.toSet()); final Map<String, String> profileTitles = userService.getProfileTitles( @@ -151,12 +171,12 @@ public class SubscriptionService extends AbstractService { final String senderName = (profileTitles != null && profileTitles.containsKey(senderPubkey)) ? profileTitles.get(senderPubkey) : ModelUtils.minifyPubkey(senderPubkey); - subscriptions.stream() + subscriptions.parallelStream() .map(record -> decryptEmailSubscription((EmailSubscription)record)) - .filter(Objects::nonNull) + .filter(record -> (record != null && record.getContent().getFrequency() == frequency)) .map(record -> processEmailSubscription(record, senderPubkey, senderName, profileTitles)) .filter(Objects::nonNull) - .forEach(this::saveSubscription); + .forEach(this::saveExecution); hasMore = CollectionUtils.size(subscriptions) >= size; from += size; @@ -199,7 +219,7 @@ public class SubscriptionService extends AbstractService { return subscription; } - protected EmailSubscription processEmailSubscription(final EmailSubscription subscription, + protected SubscriptionExecution processEmailSubscription(final EmailSubscription subscription, final String senderPubkey, final String senderName, final Map<String, String> profileTitles) { @@ -207,26 +227,44 @@ public class SubscriptionService extends AbstractService { logger.info(String.format("Processing email subscription [%s]", subscription.getId())); - Long lastTime = 0l; // TODO get it from subscription ? + SubscriptionExecution lastExecution = subscriptionExecutionDao.getLastExecution(subscription); + Long lastExecutionTime; + + if (lastExecution != null) { + lastExecutionTime = lastExecution.getTime(); + } + // If first email execution: only send event from the last 7 days. + else { + Calendar defaultDateLimit = new GregorianCalendar(); + defaultDateLimit.setTimeInMillis(System.currentTimeMillis()); + defaultDateLimit.add(Calendar.DAY_OF_YEAR, - 7); + defaultDateLimit.set(Calendar.HOUR_OF_DAY, 0); + defaultDateLimit.set(Calendar.MINUTE, 0); + defaultDateLimit.set(Calendar.SECOND, 0); + defaultDateLimit.set(Calendar.MILLISECOND, 0); + lastExecutionTime = defaultDateLimit.getTimeInMillis() / 1000; + } // Get last user events String[] includes = subscription.getContent() == null ? null : subscription.getContent().getIncludes(); String[] excludes = subscription.getContent() == null ? null : subscription.getContent().getExcludes(); - List<UserEvent> userEvents = userEventService.getUserEvents(subscription.getIssuer(), lastTime, includes, excludes); + List<UserEvent> userEvents = userEventService.getUserEvents(subscription.getIssuer(), lastExecutionTime, includes, excludes); + if (CollectionUtils.isEmpty(userEvents)) return null; // no events: stop here - STGroup templates = new STGroupDir("templates", '$', '$'); - templates.registerRenderer(Date.class, new DateRenderer()); - //templates.registerRenderer(String.class, new StringRenderer()); - //templates.registerRenderer(Number.class, new NumberRenderer()); - templates.registerRenderer(String.class, new I18nRenderer()); + // Get user locale String[] localParts = subscription.getContent() != null && subscription.getContent().getLocale() != null ? subscription.getContent().getLocale().split("-") : new String[]{"en", "GB"}; - Locale issuerLocale = localParts.length >= 2 ? new Locale(localParts[0].toLowerCase(), localParts[1].toUpperCase()) : new Locale(localParts[0].toLowerCase()); - // Compute text - String text = fillTemplate( + // Configure templates engine + STGroup templates = new STGroupDir("templates", '$', '$'); + templates.registerRenderer(Date.class, new DateRenderer()); + templates.registerRenderer(String.class, new StringRenderer()); + //templates.registerRenderer(Number.class, new NumberRenderer()); + + // Compute text content + final String text = fillTemplate( templates.getInstanceOf("text_email"), subscription, senderPubkey, @@ -237,7 +275,7 @@ public class SubscriptionService extends AbstractService { .render(issuerLocale); // Compute HTML content - String html = fillTemplate( + final String html = fillTemplate( templates.getInstanceOf("html_email_content"), subscription, senderPubkey, @@ -247,22 +285,38 @@ public class SubscriptionService extends AbstractService { pluginSettings.getCesiumUrl()) .render(issuerLocale); - mailService.sendHtmlEmailWithText( + + + // Schedule email sending + threadPool.schedule(() -> mailService.sendHtmlEmailWithText( emailSubjectPrefix + I18n.t("duniter4j.es.subscription.email.subject", userEvents.size()), text, "<body>" + html + "</body>", - subscription.getContent().getEmail()); - return subscription; + subscription.getContent().getEmail())); + + + // Compute last time (should be the first one, as events are sorted in DESC order) + Long lastEventTime = userEvents.get(0).getTime(); + if (lastExecution == null) { + lastExecution = new SubscriptionExecution(); + lastExecution.setRecipient(subscription.getIssuer()); + lastExecution.setRecordType(subscription.getType()); + lastExecution.setRecordId(subscription.getId()); + } + lastExecution.setTime(lastEventTime); + + + return lastExecution; } public static ST fillTemplate(ST template, - EmailSubscription subscription, - String senderPubkey, - String senderName, - Map<String, String> issuerProfilNames, - List<UserEvent> userEvents, - String cesiumSiteUrl) { + EmailSubscription subscription, + String senderPubkey, + String senderName, + Map<String, String> issuerProfilNames, + List<UserEvent> userEvents, + String cesiumSiteUrl) { String issuerName = issuerProfilNames != null && issuerProfilNames.containsKey(subscription.getIssuer()) ? issuerProfilNames.get(subscription.getIssuer()) : ModelUtils.minifyPubkey(subscription.getIssuer()); @@ -271,7 +325,8 @@ public class SubscriptionService extends AbstractService { try { // Compute body template.add("url", cesiumSiteUrl); - template.add("issuer", issuerName); + template.add("issuerPubkey", subscription.getIssuer()); + template.add("issuerName", issuerName); template.add("senderPubkey", senderPubkey); template.add("senderName", senderName); userEvents.forEach(userEvent -> { @@ -282,7 +337,6 @@ public class SubscriptionService extends AbstractService { description, new Date(userEvent.getTime() * 1000) }); - }); return template; @@ -293,66 +347,46 @@ public class SubscriptionService extends AbstractService { } } + protected void saveExecution(SubscriptionExecution execution) { + Preconditions.checkNotNull(execution); + Preconditions.checkNotNull(execution.getRecipient()); + Preconditions.checkNotNull(execution.getRecordType()); + Preconditions.checkNotNull(execution.getRecordId()); - public static String computeTextEmail(STGroup templates, - Locale issuerLocale, - EmailSubscription subscription, - String senderPubkey, - String senderName, - Map<String, String> issuerProfilNames, - List<UserEvent> userEvents, - String cesiumSiteUrl) { - String issuerName = issuerProfilNames != null && issuerProfilNames.containsKey(subscription.getIssuer()) ? - issuerProfilNames.get(subscription.getIssuer()) : - ModelUtils.minifyPubkey(subscription.getIssuer()); + // Update issuer + execution.setIssuer(pluginSettings.getNodePubkey()); - try { - // Compute text content - ST tpl = templates.getInstanceOf("text_email"); - tpl.add("url", cesiumSiteUrl); - tpl.add("issuer", issuerName); - tpl.add("url", cesiumSiteUrl); - tpl.add("senderPubkey", senderPubkey); - tpl.add("senderName", senderName); - userEvents.forEach(userEvent -> { - String description = userEvent.getParams() != null ? - I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase(), userEvent.getParams()) : - I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase()); - tpl.addAggr("events.{description, time}", new Object[]{ - description, - new Date(userEvent.getTime() * 1000) - }); + // Fill hash + signature + String json = toJson(execution, true/*skip hash and signature*/); + execution.setHash(cryptoService.hash(json)); + execution.setSignature(cryptoService.sign(json, pluginSettings.getNodeKeypair().getSecKey())); - }); + if (execution.getId() == null) { - return tpl.render(); + subscriptionExecutionDao.create(json, false/*not wait*/); } - catch (Exception e) { - throw new TechnicalException(e); + else { + subscriptionExecutionDao.update(execution.getId(), json, false/*not wait*/); } } - protected EmailSubscription saveSubscription(EmailSubscription subscription) { - Preconditions.checkNotNull(subscription); - - //mailService.sendEmail(); - return subscription; - } - - private String toJson(EmailSubscription subscription) { - return toJson(subscription, false); + private String toJson(Record record) { + return toJson(record, false); } - private String toJson(EmailSubscription subscription, boolean cleanHashAndSignature) { + private String toJson(Record record, boolean cleanHashAndSignature) { + Preconditions.checkNotNull(record); try { - String json = objectMapper.writeValueAsString(subscription); + String json = objectMapper.writeValueAsString(record); if (cleanHashAndSignature) { json = JacksonUtils.removeAttribute(json, Record.PROPERTY_SIGNATURE); json = JacksonUtils.removeAttribute(json, Record.PROPERTY_HASH); } return json; } catch(JsonProcessingException e) { - throw new TechnicalException("Unable to serialize UserEvent object", e); + throw new TechnicalException("Unable to serialize object " + record.getClass().getName(), e); } } + + } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/DateUtils.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/DateUtils.java new file mode 100644 index 00000000..85162fe8 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/DateUtils.java @@ -0,0 +1,49 @@ +package org.duniter.elasticsearch.subscription.util; + +import java.util.Calendar; +import java.util.Date; +import java.util.GregorianCalendar; + +/** + * Created by blavenie on 10/04/17. + */ +public class DateUtils { + + public static Date nextHour(int hour) { + Calendar cal = new GregorianCalendar(); + cal.setTimeInMillis(System.currentTimeMillis()); + if (cal.get(Calendar.HOUR_OF_DAY) > hour) { + // Too late for today: add 1 day (will wait tomorrow) + cal.add(Calendar.DAY_OF_YEAR, 1); + } + cal.set(Calendar.HOUR_OF_DAY, hour); + cal.set(Calendar.MINUTE, 0); + cal.set(Calendar.SECOND, 0); + cal.set(Calendar.MILLISECOND, 0); + return cal.getTime(); + } + + public static Date nextDayAndHour(int dayOfTheWeek, int hour) { + Calendar cal = new GregorianCalendar(); + cal.setTimeInMillis(System.currentTimeMillis()); + if (cal.get(Calendar.DAY_OF_WEEK) > dayOfTheWeek || (cal.get(Calendar.DAY_OF_WEEK) == dayOfTheWeek && cal.get(Calendar.HOUR_OF_DAY) > hour)) { + // Too late for this week: will wait for next week + cal.add(Calendar.WEEK_OF_YEAR, 1); + } + cal.set(Calendar.DAY_OF_WEEK, dayOfTheWeek); + cal.set(Calendar.HOUR_OF_DAY, hour); + cal.set(Calendar.MINUTE, 0); + cal.set(Calendar.SECOND, 0); + cal.set(Calendar.MILLISECOND, 0); + return cal.getTime(); + } + + public static long delayBeforeHour(int hour) { + return nextHour(hour).getTime() - System.currentTimeMillis(); + } + + public static long delayBeforeDayAndHour(int dayOfTheWeek, int hour) { + return nextDayAndHour(dayOfTheWeek, hour).getTime() - System.currentTimeMillis(); + } +} + diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java deleted file mode 100644 index fbcd6a98..00000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java +++ /dev/null @@ -1,24 +0,0 @@ -package org.duniter.elasticsearch.subscription.util.stringtemplate; - -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.StringUtils; -import org.nuiton.i18n.I18n; -import org.stringtemplate.v4.AttributeRenderer; - -import java.util.Locale; - -/** - * Created by blavenie on 10/04/17. - */ -public class I18nRenderer implements AttributeRenderer{ - - @Override - public String toString(Object key, String formatString, Locale locale) { - if (formatString == null || !formatString.startsWith("i18n")) return key.toString(); - String[] params = formatString.startsWith("i18n:") ? formatString.substring(5).split(",") : null; - if (CollectionUtils.isNotEmpty(params)) { - return I18n.l(locale, key.toString(), params); - } - return I18n.l(locale, key.toString()); - } -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java new file mode 100644 index 00000000..92b638ba --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java @@ -0,0 +1,32 @@ +package org.duniter.elasticsearch.subscription.util.stringtemplate; + +import org.duniter.core.client.model.ModelUtils; +import org.duniter.core.util.CollectionUtils; +import org.duniter.core.util.StringUtils; +import org.nuiton.i18n.I18n; +import org.stringtemplate.v4.AttributeRenderer; + +import java.util.Locale; + +/** + * Add format capabilities: i18n, pubkey + * Created by blavenie on 10/04/17. + */ +public class StringRenderer extends org.stringtemplate.v4.StringRenderer{ + + @Override + public String toString(Object o, String formatString, Locale locale) { + return formatString == null ? (String)o : + (formatString.equals("pubkey") ? ModelUtils.minifyPubkey((String)o) : + (formatString.startsWith("i18n") ? toI18nString(o, formatString, locale) : + super.toString(o, formatString, locale))); + } + + protected String toI18nString(Object key, String formatString, Locale locale) { + String[] params = formatString.startsWith("i18n:") ? formatString.substring(5).split(",") : null; + if (CollectionUtils.isNotEmpty(params)) { + return I18n.l(locale, key.toString(), params); + } + return I18n.l(locale, key.toString()); + } +} diff --git a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties index d179ccf7..a02b9131 100644 --- a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties +++ b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties @@ -1,7 +1,14 @@ -duniter4j.es.subscription.email.footer.disableHelp=You can disable this email notification service in the page <a href\="%s">Online services</a> of Cesium+. -duniter4j.es.subscription.email.footer.sendBy=This email has sent you the Cesium+ node of <a href\="%s">%s</a>. -duniter4j.es.subscription.email.header=Hello <b>%s</b>\!<br/>You received %s new notifications\: +duniter4j.es.subscription.email.footer.disableHelp=You can disable this email notification service in the page %2$% (%1$s). +duniter4j.es.subscription.email.footer.sendBy=This email has sent you the Cesium+ node of %2$s (%1$s). +duniter4j.es.subscription.email.hello=Hello %s\! +duniter4j.es.subscription.email.html.footer.disableHelp=You can disable this email notification service in <a href\="%s">online services</a> page. +duniter4j.es.subscription.email.html.footer.sendBy=This email has sent you the Cesium+ node of <a href\="%s">%s</a>.. +duniter4j.es.subscription.email.html.hello=Hello <b>%s</b>\! +duniter4j.es.subscription.email.html.pubkey=Public key\: <a href\="%s">%s</a> +duniter4j.es.subscription.email.html.unreadCount=You received <b>%s new notifications</b>. duniter4j.es.subscription.email.notificationsDivider=Notifications list\: duniter4j.es.subscription.email.openCesium=Open Cesium+ +duniter4j.es.subscription.email.pubkey=Public key\: %2$s (%1$s) duniter4j.es.subscription.email.subject=You received %s new notifications +duniter4j.es.subscription.email.unreadCount=You received %s new notifications. duniter4j.es.subscription.error.mailDisabling=Unable to process email subscriptions\: Email sending is disabled in the configuration diff --git a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties index 37f541ed..c64cf822 100644 --- a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties +++ b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties @@ -1,11 +1,14 @@ -duniter4j.es.subscription.email.footer=Cet email vous a été envoyé par Cesium+.<br/><small>vous pouvez désactiver ce service de notification par email, dans la rubrique <a href\="%s">Services en ligne</a> de Cesium+. -duniter4j.es.subscription.email.footer.disableHelp=Vous pouvez désactiver ce service de notification par email, dans la rubrique <a href\="%s">Services en ligne</a> de Cesium+. -duniter4j.es.subscription.email.footer.disableHelp.text=Vous pouvez désactiver ce service de notification par email, dans la rubrique "Services en ligne" de Cesium+ (%s). -duniter4j.es.subscription.email.footer.sendBy=Cet email vous a été envoyé le noeud Cesium+ de <a href\="%1$s">%2$s</a>. -duniter4j.es.subscription.email.footer.sendBy.text=Cet email vous a été envoyé le noeud Cesium+ de %2$s (%1$s). -duniter4j.es.subscription.email.header=Bonjour <b>%s</b> \!<br/>Vous avez %s notifications non lues. -duniter4j.es.subscription.email.header.text=Bonjour %s \!\nVous avez %s notifications non lues. +duniter4j.es.subscription.email.footer.disableHelp=Vous pouvez désactiver ce service de notification par email, dans la rubrique "Services en ligne" de Cesium+ (%s). +duniter4j.es.subscription.email.footer.sendBy=Cet email vous a été envoyé le noeud Cesium+ de %2$s (%1$s). +duniter4j.es.subscription.email.hello=Bonjour %s \! +duniter4j.es.subscription.email.html.footer.disableHelp=Vous pouvez désactiver ce service de notification par email, dans <a href\="%s">la rubrique services en ligne</a> de Cesium+. +duniter4j.es.subscription.email.html.footer.sendBy=Cet email vous a été envoyé depuis le noeud Cesium+ de <a href\="%1$s">%2$s</a>. +duniter4j.es.subscription.email.html.hello=Bonjour <b>%s</b> \! +duniter4j.es.subscription.email.html.pubkey=Clé publique \: <a href\="%s">%s</a> +duniter4j.es.subscription.email.html.unreadCount=Vous avez <b>%s notifications</b> non lues. duniter4j.es.subscription.email.notificationsDivider=Liste des notifications \: duniter4j.es.subscription.email.openCesium=Ouvrir Cesium+ +duniter4j.es.subscription.email.pubkey=Clé publique \: %2$s (%1$s) duniter4j.es.subscription.email.subject=%s nouvelles notifications non lues +duniter4j.es.subscription.email.unreadCount=Vous avez %s notifications non lues. duniter4j.es.subscription.error.mailDisabling=Impossible de traiter les abonnements email\: la fonction d'envoi d'email est désactivée dans la configuration diff --git a/duniter4j-es-subscription/src/main/resources/templates/cesium_logo.st b/duniter4j-es-subscription/src/main/resources/templates/cesium_logo.st new file mode 100644 index 00000000..83cf445c --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/cesium_logo.st @@ -0,0 +1,6 @@ +cesium_logo(url, data) ::= << +$if(data)$<img height="144" width="144" src=""> +$else$ +<img height="144" width="144" src="$url$/img/logo_128px.png"> +$endif$ +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st b/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st index 629515c4..c6fa7ed4 100644 --- a/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st +++ b/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st @@ -1,33 +1,76 @@ -html_email_content(issuer, senderPubkey, senderName, events, url) ::= << - <div class="row no-padding"> - <div class="col col-20 hidden-xs hidden-sm text-center" id="home"> - <div class="logo"></div> - </div> - <div class="col"> - <div class="padding padding-bottom row responsive-sm"> - <div class="col"> - $length(events):{count | $i18n_args("duniter4j.es.subscription.email.header", [issuer, count])$}$ - </div> - <div class="col"> - <a class="button button-positive pull-right" - href="$url$">$i18n("duniter4j.es.subscription.email.openCesium")$ >></a> - </div> +html_email_content(issuerPubkey, issuerName, senderPubkey, senderName, events, url) ::= << +<table cellspacing="0" cellpadding="0" width="100%" + style="font-size:12px;font-family:Helvetica Neue,Helvetica,Lucida Grande,tahoma,verdana,arial,sans-serif;border-spacing:0px;border-collapse:collapse;max-width:600px!important;"> + <tr> + <td> + <div style="background:#1a237e;width:100%;text-align:center;border-radius:4px;min-height:35px;"> + + $cesium_logo(url, true)$ + + <p style="margin:0px;padding:8px 0px;text-align:center;color:white;font-size:14px;"> + $i18n_args("duniter4j.es.subscription.email.html.hello", issuerName)$ + </p> </div> - <div class="list item-border-large"> - <div class="item item-divider stable-bg"> - $i18n("duniter4j.es.subscription.email.notificationsDivider")$ - </div> - $events:{e|$event_item(e)$}$ + </td> + </tr> + + <tr> + <td> + <table cellspacing="0" cellpadding="0" width="100%" > + <tr> + <td> + <p style="margin:0px;padding:16px;font-size: 12px;"> + $i18n_args("duniter4j.es.subscription.email.html.unreadCount", {$length(events)$} )$ + $if(issuerPubkey)$ + <br/> + <span style="font-size:12px;color:grey !important;"> + $i18n_args("duniter4j.es.subscription.email.html.pubkey", [{$[url, "/#/app/wot/", issuerPubkey, "/"]; separator=""$}, {$issuerPubkey; format="pubkey"$}])$ + </span> + $endif$ + </p> + + </td> + <td> + <p style="margin:0px;width:100%;text-align:right;min-height: 64px;padding: 16px 0px;"> + <a style="overflow:hidden!important;background-color:#387ef5;border-color:transparent;border-radius:2px;border-shadow: 2px 2px rgba(50,50,50,0.32);box-sizing: border-box;color:white;display:inline-block;font-size:14px;font-weight: 500;height: 47px;letter-spacing: 0.5px;line-height:42px;margin:0;min-height:47px;min-width:52px;padding-bottom:0px;padding-left:24px;padding-right:24px;padding-top:0px;text-align:center;text-decoration:none;text-transform:uppercase;" + href="$url$">$i18n("duniter4j.es.subscription.email.openCesium")$ >></a> + </p> + </td> + </tr> + </table> + </td> + </tr> + + <tr> + <td> + <div style="background-color:#f5f5f5;border: 0;box-sizing: border-box; color: rgba(0, 0, 0, 0.54);font-size: 14px;font-weight: 700;height: 48px;line-height: 48px;min-height: 48px;padding-bottom: 8px;padding-left: 16px;padding-right: 16px;padding-top: 8px;vertical-align: baseline;"> + $i18n("duniter4j.es.subscription.email.notificationsDivider")$ </div> - <div class="center padding text-center"> - <i class="ion-es-user-api"></i> - $i18n_args("duniter4j.es.subscription.email.footer.sendBy", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ - <br/> - <small> - $i18n_args("duniter4j.es.subscription.email.footer.disableHelp", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ - </small> + </td> + </tr> + + $events:{e|$html_event_item(e)$}$ + + <tr> + <td> + <div style="width:100%;text-align:center;min-height:32px;padding:8px;"> + </div> - </div> - <div class="col col-20 hidden-xs hidden-sm"> </div> - </div> ->> \ No newline at end of file + </td> + </tr> + + <tr> + <td> + <div style="background-color: rgb(236, 240, 247) !important;border-color: rgb(221, 223, 226) !important;width:100%;text-align:center;border-radius:4px;"> + <p style="margin:0px;padding:8px 0px;text-align:center;color:grey !important;text-decoration:none !important;"> + $i18n_args("duniter4j.es.subscription.email.html.footer.sendBy", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ + <br/> + <small> + $i18n_args("duniter4j.es.subscription.email.html.footer.disableHelp", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ + </small> + </p> + </div> + </td> + </tr> +</table> +>> diff --git a/duniter4j-es-subscription/src/main/resources/templates/html_event_item.st b/duniter4j-es-subscription/src/main/resources/templates/html_event_item.st new file mode 100644 index 00000000..8515d55e --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/html_event_item.st @@ -0,0 +1,10 @@ +html_event_item(e) ::= << + <tr> + <td> + <div style="border-bottom: solid 1px #ccc !important;color: rgb(68, 68, 68);display: block;font-size: 14px;font-weight: 400;line-height: 20px;margin-bottom: -1px;margin-left: -1px;margin-right: -1px;margin-top: -1px;padding-bottom: 16px;padding-left: 16px;padding-right: 16px;padding-top: 16px;white-space: normal;"> + <h3 style="color: rgb(0,0,0);font-family:-apple-system,Helvetica Neue,Helvetica,Roboto,Segoe UI,sans-serif;font-size: 14px;font-synthesis: weight style;font-weight: 500;line-height: 16.8px;margin-bottom: 4px;margin-left: 0px;margin-right: 0px;margin-top: 0px;padding: 0;">$e.description$</h3> + <h4 style="color: grey !important;font-size: 12px;font-weight: 500;line-height: 14.4px;margin: 0;padding: 0;">$e.time; format="short"$</h4> + </div> + </td> + </tr> +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/text_email.st b/duniter4j-es-subscription/src/main/resources/templates/text_email.st index b982a67a..004367d9 100644 --- a/duniter4j-es-subscription/src/main/resources/templates/text_email.st +++ b/duniter4j-es-subscription/src/main/resources/templates/text_email.st @@ -1,13 +1,15 @@ -text_email(issuer, senderPubkey, senderName, events, url) ::= << -$length(events):{count | $i18n_args("duniter4j.es.subscription.email.header.text", [issuer, count])$}$ +text_email(issuerPubkey, issuerName, senderPubkey, senderName, events, url) ::= << +$i18n_args("duniter4j.es.subscription.email.hello", issuerName)$ +$i18n_args("duniter4j.es.subscription.email.unreadCount", {$length(events)$} )$ $i18n("duniter4j.es.subscription.email.notificationsDivider")$ $events:{e|$text_event_item(e)$}$ $i18n("duniter4j.es.subscription.email.openCesium")$ : $url$ +$if(issuerPubkey)$$i18n_args("duniter4j.es.subscription.email.pubkey", [{$[url, "/#/app/wot/", issuerPubkey, "/"]; separator=""$}, {$issuerPubkey; format="pubkey"$}])$$endif$ ----------------------------------------------- -$i18n_args("duniter4j.es.subscription.email.footer.sendBy.text", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ -$i18n_args("duniter4j.es.subscription.email.footer.disableHelp.text", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ +$i18n_args("duniter4j.es.subscription.email.footer.sendBy", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ +$i18n_args("duniter4j.es.subscription.email.footer.disableHelp", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ >> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st b/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st index 51cd56fa..701a00ec 100644 --- a/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st +++ b/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st @@ -1,3 +1,3 @@ text_event_item(e) ::= << - - $e.time; format="short"$ | $e.description$ + - [$e.time; format="short"$] $e.description$ >> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java index 81b5babe..c8f9bf4a 100644 --- a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java +++ b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java @@ -102,7 +102,7 @@ public class SubscriptionServiceTest { // wait 10s Thread.sleep(10000); - service.executeEmailSubscriptions(); + service.executeEmailSubscriptions(EmailSubscription.Frequency.daily); // wait 10s Thread.sleep(10000); diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java index c526a6a2..71f43f9e 100644 --- a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java +++ b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java @@ -24,20 +24,17 @@ package org.duniter.elasticsearch.subscription.service; import org.duniter.core.client.model.ModelUtils; import org.duniter.core.exception.TechnicalException; -import org.duniter.core.test.TestResource; import org.duniter.elasticsearch.subscription.util.stringtemplate.DateRenderer; -import org.duniter.elasticsearch.subscription.util.stringtemplate.I18nRenderer; +import org.duniter.elasticsearch.subscription.util.stringtemplate.StringRenderer; import org.junit.Test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.stringtemplate.v4.ST; import org.stringtemplate.v4.STGroup; import org.stringtemplate.v4.STGroupDir; -import org.stringtemplate.v4.StringRenderer; -import java.io.File; -import java.io.FileWriter; import java.util.Date; +import java.util.Locale; import static org.junit.Assert.assertNotNull; @@ -47,10 +44,7 @@ import static org.junit.Assert.assertNotNull; public class SubscriptionTemplateTest { private static final Logger log = LoggerFactory.getLogger(SubscriptionTemplateTest.class); - private static final boolean verbose = true; - - //@ClassRule - public static final TestResource resource = TestResource.create(); + private static final boolean verbose = false; @Test public void testHtmlEmail() throws Exception{ @@ -60,36 +54,22 @@ public class SubscriptionTemplateTest { group.registerRenderer(Date.class, new DateRenderer()); group.registerRenderer(String.class, new StringRenderer()); - group.registerRenderer(String.class, new I18nRenderer()); - - ST contentEmail = group.getInstanceOf("html_email_content"); - contentEmail.add("issuer", "MyIssuerName"); - contentEmail.add("url", "https://g1.duniter.fr"); - contentEmail.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); - contentEmail.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); - contentEmail.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); - assertNotNull(contentEmail); - ST css_logo = group.getInstanceOf("css_logo"); - assertNotNull(css_logo); + ST tpl = group.getInstanceOf("html_email_content"); + tpl.add("issuerName", "MyIssuerName"); + tpl.add("issuerPubkey", "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"); + tpl.add("url", "https://g1.duniter.fr"); + tpl.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); + tpl.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); + tpl.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); + tpl.addAggr("events.{description, time}", new Object[]{"My event description 2", new Date()}); + assertNotNull(tpl); - ST htmlTpl = group.getInstanceOf("html"); - assertNotNull(htmlTpl); - htmlTpl.add("content", contentEmail.render()); - htmlTpl.add("useCss", "true"); - String html = htmlTpl.render(); + String email = tpl.render(new Locale("en", "GB")); if (verbose) { - System.out.println(html); + System.out.println(email); } - - //FileWriter fw = new FileWriter(new File(resource.getResourceDirectory("out"), "page.html")); - FileWriter fw = new FileWriter(new File("/home/blavenie/git/duniter4j/duniter4j-es-subscription/src/test/resources/test2.html")); - fw.write(html); - fw.flush(); - fw.close(); - - } catch (Exception e) { throw new TechnicalException(e); @@ -104,17 +84,17 @@ public class SubscriptionTemplateTest { group.registerRenderer(Date.class, new DateRenderer()); group.registerRenderer(String.class, new StringRenderer()); - group.registerRenderer(String.class, new I18nRenderer()); ST tpl = group.getInstanceOf("text_email"); - tpl.add("issuer", "MyIssuerName"); + tpl.add("issuerPubkey", "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"); + tpl.add("issuerName", "kimamila"); tpl.add("url", "https://g1.duniter.fr"); tpl.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); tpl.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); tpl.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); assertNotNull(tpl); - String text = tpl.render(); + String text = tpl.render(new Locale("en", "GB")); if (verbose) { System.out.println(text); diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java index 656790b3..0860736c 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java @@ -59,6 +59,7 @@ import org.elasticsearch.index.query.BoolQueryBuilder; import org.elasticsearch.index.query.QueryBuilder; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.search.SearchHit; +import org.elasticsearch.search.sort.SortOrder; import java.io.IOException; import java.util.*; @@ -240,6 +241,7 @@ public class UserEventService extends AbstractService implements ChangeService.C .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) .setFetchSource(true) .setQuery(query) + .addSort(UserEvent.PROPERTY_TIME, SortOrder.DESC) .get(); -- GitLab