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")$ &gt;&gt;</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")$ &gt;&gt;</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">&nbsp;</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