diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Event.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Event.java
new file mode 100644
index 0000000000000000000000000000000000000000..d535ad58ac20900c119e114a00f4adea169c33a4
--- /dev/null
+++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Event.java
@@ -0,0 +1,56 @@
+package org.duniter.core.client.model.elasticsearch;
+
+import org.nuiton.i18n.I18n;
+
+import java.util.Locale;
+
+/**
+ * Created by blavenie on 29/11/16.
+ */
+public class Event extends Record {
+
+    public static final String PROPERTY_TYPE="type";
+    public static final String PROPERTY_CODE="code";
+    public static final String PROPERTY_MESSAGE="message";
+    public static final String PROPERTY_PARAMS="params";
+
+    private String type;
+
+    private String code;
+
+    private String message;
+
+    private String[] params;
+
+    public String getType() {
+        return type;
+    }
+
+    public void setType(String type) {
+        this.type = type;
+    }
+
+    public String getCode() {
+        return code;
+    }
+
+    public void setCode(String code) {
+        this.code = code;
+    }
+
+    public String getMessage() {
+        return message;
+    }
+
+    public void setMessage(String message) {
+        this.message = message;
+    }
+
+    public String[] getParams() {
+        return params;
+    }
+
+    public void setParams(String[] params) {
+        this.params = params;
+    }
+}
diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/ServiceLocator.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/ServiceLocator.java
index 5f4b669631058ac10953a4ec58dffa722579fa51..e0f14a353c8b514c68bada786d197d278d4bb241 100644
--- a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/ServiceLocator.java
+++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/ServiceLocator.java
@@ -30,6 +30,7 @@ import org.duniter.core.client.service.elasticsearch.CurrencyRegistryRemoteServi
 import org.duniter.core.client.service.local.CurrencyService;
 import org.duniter.core.client.service.local.PeerService;
 import org.duniter.core.service.CryptoService;
+import org.duniter.core.service.MailService;
 import org.slf4j.Logger;
 import org.slf4j.LoggerFactory;
 
@@ -139,6 +140,10 @@ public class ServiceLocator implements Closeable {
         return getBean(CurrencyRegistryRemoteService.class);
     }
 
+    public MailService getMaiLService() {
+        return getBean(MailService.class);
+    }
+
     public <S extends Bean> S getBean(Class<S> clazz) {
         if (beanFactory == null) {
             initBeanFactory();
diff --git a/duniter4j-core-shared/pom.xml b/duniter4j-core-shared/pom.xml
index f3061fcfeb3665d832bca9bcba21c80191239df6..fa32b08e294e1d952226e812c3700732c0c05930 100644
--- a/duniter4j-core-shared/pom.xml
+++ b/duniter4j-core-shared/pom.xml
@@ -58,6 +58,10 @@
       <groupId>org.nuiton.i18n</groupId>
       <artifactId>nuiton-i18n</artifactId>
     </dependency>
+    <dependency>
+      <groupId>javax.mail</groupId>
+      <artifactId>mail</artifactId>
+    </dependency>
 
     <!-- Unit test -->
     <dependency>
diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java
new file mode 100644
index 0000000000000000000000000000000000000000..3bb0859189f10e3be065383d02050baf2420866f
--- /dev/null
+++ b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java
@@ -0,0 +1,32 @@
+package org.duniter.core.service;
+
+import org.duniter.core.beans.Bean;
+import org.duniter.core.exception.TechnicalException;
+
+import javax.mail.internet.ContentType;
+import javax.mail.internet.ParseException;
+
+/**
+ * Created by blavenie on 28/11/16.
+ */
+public interface MailService extends Bean {
+
+    void sendTextEmail(String smtpHost,
+                       int smtpPort,
+                       String smtpUsername,
+                       String smtpPassword,
+                       String issuer,
+                       String recipients,
+                       String subject,
+                       String textContent);
+
+    void sendEmail(String smtpHost,
+                   int smtpPort,
+                   String smtpUsername,
+                   String smtpPassword,
+                   String issuer,
+                   String recipients,
+                   String subject,
+                   ContentType contentType,
+                   String content);
+}
diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java
new file mode 100644
index 0000000000000000000000000000000000000000..d217524284f599a30280685c129820f15139a4f9
--- /dev/null
+++ b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java
@@ -0,0 +1,169 @@
+package org.duniter.core.service;
+
+import org.duniter.core.exception.TechnicalException;
+import org.duniter.core.util.StringUtils;
+
+import javax.mail.*;
+import javax.mail.internet.ContentType;
+import javax.mail.internet.InternetAddress;
+import javax.mail.internet.MimeMessage;
+import javax.mail.internet.ParseException;
+import java.io.Closeable;
+import java.util.Properties;
+
+public class MailServiceImpl implements MailService, Closeable {
+
+    private static Session session;
+    private static Transport transport;
+
+    public MailServiceImpl() {
+
+    }
+
+    @Override
+    public void sendTextEmail(String smtpHost,
+                          int smtpPort,
+                          String smtpUsername,
+                          String smtpPassword,
+                          String issuer,
+                          String recipients,
+                          String subject,
+                          String textContent) {
+        try{
+            ContentType contentType = new ContentType("text/plain");
+            contentType.setParameter("charset", "UTF-8");
+
+            sendEmail(smtpHost, smtpPort, smtpUsername, smtpPassword,
+                      issuer,
+                      recipients,
+                      subject,
+                      contentType,
+                      textContent);
+        }
+        catch(ParseException e) {
+            // Should never occur
+            throw new TechnicalException(e);
+        }
+
+    }
+
+    @Override
+    public void sendEmail(String smtpHost,
+                          int smtpPort,
+                          String smtpUsername,
+                          String smtpPassword,
+                          String issuer,
+                          String recipients,
+                          String subject,
+                          ContentType contentType,
+                          String content) {
+
+        // check arguments
+        if (StringUtils.isBlank(smtpHost) || smtpPort <= 0) {
+            throw new TechnicalException("Invalid arguments: 'smtpHost' could not be null or empty, and 'smtpPort' could not be <= 0");
+        }
+        if (StringUtils.isBlank(issuer)) {
+            throw new TechnicalException("Invalid arguments: 'issuer' could not be null or empty");
+        }
+        if (StringUtils.isBlank(recipients) || StringUtils.isBlank(subject) || StringUtils.isBlank(content) || contentType == null) {
+            throw new TechnicalException("Invalid arguments: 'recipients', 'subject', 'contentType' or 'content' could not be null or empty");
+        }
+
+        if (!isConnected()) {
+            connect(smtpHost, smtpPort, smtpUsername, smtpPassword, issuer);
+        }
+
+        // send email to recipients
+        try {
+            Message message = new MimeMessage(session);
+            message.setFrom(new InternetAddress(issuer));
+            message.setRecipients(Message.RecipientType.TO, InternetAddress.parse(recipients));
+            message.setSubject(subject);
+
+            message.setContent(content, contentType.toString());
+            message.saveChanges();
+            transport.sendMessage(message, message.getAllRecipients());
+
+        } catch (MessagingException e) {
+            throw new TechnicalException(String.format("Error while sending email to [%s] using smtp server [%s]",
+                    recipients,
+                    getSmtpServerAsString(smtpHost, smtpPort, smtpUsername)
+            ), e);
+        } catch (IllegalStateException e) {
+            throw new TechnicalException(String.format("Error while sending email to [%s] using smtp server [%s]",
+                    recipients,
+                    getSmtpServerAsString(smtpHost, smtpPort, smtpUsername)
+            ), e);
+        }
+    }
+
+    public void close() {
+        if (isConnected()) {
+            try {
+                transport.close();
+            }
+            catch(Exception e) {
+                // silent is gold
+            }
+            transport = null;
+            session = null;
+        }
+    }
+
+    /* -- private methods -- */
+
+    private String getSmtpServerAsString(String smtpHost, int smtpPort, String smtpUsername) {
+        StringBuilder buffer = new StringBuilder();
+        if (StringUtils.isNotBlank(smtpUsername)) {
+            buffer.append(smtpUsername).append("@");
+        }
+        return buffer.append(smtpHost)
+                .append(":")
+                .append(smtpPort)
+                .toString();
+
+    }
+
+    private void connect(String smtpHost, int smtpPort,
+                         String smtpUsername,
+                         String smtpPassword,
+                         String issuer) {
+        Properties props = new Properties();
+
+        props.put("mail.smtp.host", smtpHost);
+        props.put("mail.smtp.port", smtpPort);
+        if (StringUtils.isNotBlank(issuer)) {
+            props.put("mail.from", issuer);
+        }
+        boolean useAuth = false;
+        // auto set authentification if smtp user name is provided
+        if (StringUtils.isNotBlank(smtpUsername)) {
+            props.put("mail.smtp.auth", "true");
+            //props.put("mail.smtp.starttls.enable", "true");
+            useAuth = true;
+        }
+
+        session = Session.getInstance(props);
+
+        try {
+            transport = session.getTransport("smtp");
+            if (useAuth) {
+                transport.connect(smtpUsername, smtpPassword);
+            } else {
+                transport.connect();
+            }
+        } catch (NoSuchProviderException e) {
+            throw new TechnicalException(e);
+        } catch (MessagingException e) {
+            throw new TechnicalException(e);
+        }
+    }
+
+    private boolean isConnected() {
+        if ((session == null) || (transport == null)) {
+            return false;
+        }
+        return transport.isConnected();
+    }
+
+}
\ No newline at end of file
diff --git a/duniter4j-core-shared/src/main/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-core-shared/src/main/resources/META-INF/services/org.duniter.core.beans.Bean
index 9f44afdb46ce426cae8cc2c9237126f47a4c41f7..b5182a1c7baa184f4080da10fded7615ded6b94c 100644
--- a/duniter4j-core-shared/src/main/resources/META-INF/services/org.duniter.core.beans.Bean
+++ b/duniter4j-core-shared/src/main/resources/META-INF/services/org.duniter.core.beans.Bean
@@ -1 +1,2 @@
-org.duniter.core.service.Ed25519CryptoServiceImpl
\ No newline at end of file
+org.duniter.core.service.Ed25519CryptoServiceImpl
+org.duniter.core.service.MailServiceImpl
\ No newline at end of file
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/PluginSettings.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/PluginSettings.java
index 37a994cc03b80f168cc6bee7eac13b03f692e22e..265473a9daec1ccd72ca42c4a87bbb489b9b880a 100644
--- a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/PluginSettings.java
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/PluginSettings.java
@@ -241,6 +241,34 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> {
         return settings.getAsInt("duniter.data.sync.port", 80);
     }
 
+    public String getMailSmtpHost()  {
+        return settings.get("duniter.mail.smtp.host", "localhost");
+    }
+
+    public int getMailSmtpPort()  {
+        return settings.getAsInt("duniter.mail.smtp.port", 25);
+    }
+
+    public String getMailSmtpUsername()  {
+        return settings.get("duniter.mail.smtp.username");
+    }
+
+    public String getMailSmtpPassword()  {
+        return settings.get("duniter.mail.smtp.password");
+    }
+
+    public String getMailAdmin()  {
+        return settings.get("duniter.mail.admin");
+    }
+
+    public String getMailFrom()  {
+        return settings.get("duniter.mail.from", "no-reply@duniter.fr");
+    }
+
+    public String getMailSubjectPrefix()  {
+        return settings.get("duniter.mail.subject.prefix", "[Duniter4j ES]");
+    }
+
     /* protected methods */
 
     protected void initI18n() throws IOException {
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/node/DuniterNode.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/node/DuniterNode.java
index 0aec5df77b696f9bb65754321f5b42e18a53bde4..1f6d9019d64ea18069dc3984e0ac9cc978a568a0 100644
--- a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/node/DuniterNode.java
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/node/DuniterNode.java
@@ -27,8 +27,12 @@ import org.duniter.core.client.model.local.Peer;
 import org.duniter.elasticsearch.PluginSettings;
 import org.duniter.elasticsearch.action.security.RestSecurityController;
 import org.duniter.elasticsearch.service.*;
+import org.duniter.elasticsearch.service.event.Event;
+import org.duniter.elasticsearch.service.event.EventCodes;
+import org.duniter.elasticsearch.service.event.EventService;
 import org.duniter.elasticsearch.service.synchro.SynchroService;
 import org.duniter.elasticsearch.threadpool.ThreadPool;
+import org.elasticsearch.client.Client;
 import org.elasticsearch.cluster.health.ClusterHealthStatus;
 import org.elasticsearch.common.component.AbstractLifecycleComponent;
 import org.elasticsearch.common.inject.Inject;
@@ -47,13 +51,17 @@ public class DuniterNode extends AbstractLifecycleComponent<DuniterNode> {
     private final ThreadPool threadPool;
     private final Injector injector;
     private final static ESLogger logger = Loggers.getLogger("node");
+    private final Client client;
+    private final String clusterName;
 
     @Inject
-    public DuniterNode(Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) {
+    public DuniterNode(Client client, Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) {
         super(settings);
         this.pluginSettings = pluginSettings;
         this.threadPool = threadPool;
         this.injector = injector;
+        this.client = client;
+        this.clusterName = settings.get("cluster.name");
     }
 
     @Override
@@ -66,6 +74,16 @@ public class DuniterNode extends AbstractLifecycleComponent<DuniterNode> {
                 synchronize();
             }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN);
         }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN);
+
+        // When started
+        threadPool.scheduleOnStarted(() -> {
+            // Notify admin
+            injector.getInstance(EventService.class)
+                    .notifyAdmin(new Event(
+                            Event.EventType.INFO,
+                            EventCodes.NODE_STARTED.name(),
+                            new String[]{clusterName}));
+        });
     }
 
     @Override
@@ -104,6 +122,11 @@ public class DuniterNode extends AbstractLifecycleComponent<DuniterNode> {
                     .deleteIndex()
                     .createIndexIfNotExists();
 
+            injector.getInstance(EventService.class)
+                    .deleteIndex()
+                    .createIndexIfNotExists();
+
+
             if (logger.isInfoEnabled()) {
                 logger.info("Reloading all Duniter indices... [OK]");
             }
@@ -117,6 +140,7 @@ public class DuniterNode extends AbstractLifecycleComponent<DuniterNode> {
             injector.getInstance(MessageService.class).createIndexIfNotExists();
             injector.getInstance(UserService.class).createIndexIfNotExists();
             injector.getInstance(HistoryService.class).createIndexIfNotExists();
+            injector.getInstance(EventService.class).createIndexIfNotExists();
 
             if (logger.isInfoEnabled()) {
                 logger.info("Checking Duniter indices... [OK]");
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java
index 70f2298177a346c05fb9e2c702b5d3eddf016a58..d2b283fdd496814605ca1b1fa9990c45b6c08e64 100644
--- a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java
@@ -40,6 +40,8 @@ import org.duniter.core.client.service.local.PeerServiceImpl;
 import org.duniter.core.exception.TechnicalException;
 import org.duniter.core.service.CryptoService;
 import org.duniter.core.service.Ed25519CryptoServiceImpl;
+import org.duniter.core.service.MailService;
+import org.duniter.core.service.MailServiceImpl;
 import org.elasticsearch.common.inject.Inject;
 import org.elasticsearch.common.inject.Injector;
 import org.elasticsearch.common.inject.Singleton;
@@ -90,6 +92,7 @@ public class ServiceLocator
                 .bind(WotRemoteService.class, WotRemoteServiceImpl.class)
                 .bind(TransactionRemoteService.class, TransactionRemoteServiceImpl.class)
                 .bind(CryptoService.class, Ed25519CryptoServiceImpl.class)
+                .bind(MailService.class, MailServiceImpl.class)
                 .bind(PeerService.class, PeerServiceImpl.class)
                 .bind(CurrencyService.class, CurrencyServiceImpl.class)
                 .bind(HttpService.class, HttpServiceImpl.class)
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java
index c2604584460c86c9c6180616adb622f166ae1dba..00890a8cfadc92ca62fb24747a88fe2dfda8f2be 100644
--- a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java
@@ -35,6 +35,7 @@ import org.duniter.core.client.service.local.CurrencyService;
 import org.duniter.core.client.service.local.PeerService;
 import org.duniter.core.service.CryptoService;
 import org.duniter.elasticsearch.PluginSettings;
+import org.duniter.elasticsearch.service.event.EventService;
 import org.duniter.elasticsearch.service.synchro.SynchroService;
 import org.elasticsearch.common.inject.AbstractModule;
 import org.elasticsearch.common.inject.Module;
@@ -46,6 +47,7 @@ public class ServiceModule extends AbstractModule implements Module {
 
         // ES common service
         bind(PluginSettings.class).asEagerSingleton();
+        bind(EventService.class).asEagerSingleton();
 
         // ES indexation services
         bind(RegistryService.class).asEagerSingleton();
@@ -62,7 +64,6 @@ public class ServiceModule extends AbstractModule implements Module {
         bindWithLocator(NetworkRemoteService.class);
         bindWithLocator(WotRemoteService.class);
         bindWithLocator(TransactionRemoteService.class);
-        bindWithLocator(CryptoService.class);
         bindWithLocator(PeerService.class);
         bindWithLocator(CurrencyService.class);
         bindWithLocator(HttpService.class);
@@ -70,6 +71,10 @@ public class ServiceModule extends AbstractModule implements Module {
         bindWithLocator(PeerDao.class);
         bindWithLocator(DataContext.class);
 
+        // Duniter Shared API beans
+        bindWithLocator(CryptoService.class);
+        bindWithLocator(org.duniter.core.service.MailService.class);
+
 /*
         bindWithLocator(BlockchainRemoteServiceImpl.class);
         bindWithLocator(NetworkRemoteServiceImpl.class);
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/UserService.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/UserService.java
index cc079c45e97919dfffdfe511b6d2442ce571aa2f..4163b6e2f936922ad3f53c6b986de9d8d05fac84 100644
--- a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/UserService.java
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/UserService.java
@@ -29,6 +29,7 @@ import org.duniter.core.client.service.bma.BlockchainRemoteService;
 import org.duniter.core.client.service.bma.WotRemoteService;
 import org.duniter.core.exception.TechnicalException;
 import org.duniter.core.service.CryptoService;
+import org.duniter.core.service.MailService;
 import org.duniter.elasticsearch.PluginSettings;
 import org.duniter.elasticsearch.exception.AccessDeniedException;
 import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder;
@@ -54,7 +55,8 @@ public class UserService extends AbstractService {
     @Inject
     public UserService(Client client,
                        PluginSettings settings,
-                       CryptoService cryptoService) {
+                       CryptoService cryptoService,
+                       MailService mailService) {
         super("gchange." + INDEX, client, settings,cryptoService);
     }
 
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/Event.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/Event.java
new file mode 100644
index 0000000000000000000000000000000000000000..6b5d517646c118040d3e1170d3a896913b096413
--- /dev/null
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/Event.java
@@ -0,0 +1,81 @@
+package org.duniter.elasticsearch.service.event;
+
+import org.nuiton.i18n.I18n;
+
+import java.util.Locale;
+
+/**
+ * Created by blavenie on 29/11/16.
+ */
+public class Event {
+
+    private EventType type;
+
+    private String code;
+
+    private long time;
+
+    private String message;
+
+
+    private String[] params;
+
+    public Event(EventType type, String code) {
+        this(type, code, null);
+    }
+
+    public Event(EventType type, String code, String[] params) {
+        this.type = type;
+        this.code = code;
+        this.params = params;
+        // default
+        this.message = I18n.t("duniter4j.event." + code, params);
+        this.time = Math.round(1d * System.currentTimeMillis() / 1000);
+    }
+
+    public EventType getType() {
+        return type;
+    }
+
+    public void setType(EventType type) {
+        this.type = type;
+    }
+
+    public String getCode() {
+        return code;
+    }
+
+    public void setCode(String code) {
+        this.code = code;
+    }
+
+    public String getMessage() {
+        return message;
+    }
+
+    public String getLocalizedMessage(Locale locale) {
+        return I18n.l(locale, "duniter4j.event." + code, params);
+    }
+
+    public void setMessage(String message) {
+        this.message = message;
+    }
+
+    public String[] getParams() {
+        return params;
+    }
+
+    public void setParams(String[] params) {
+        this.params = params;
+    }
+
+    public long getTime() {
+        return time;
+    }
+
+    public enum EventType {
+        INFO,
+        WARN,
+        ERROR
+    }
+}
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/EventCodes.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/EventCodes.java
new file mode 100644
index 0000000000000000000000000000000000000000..f790e04670df91961cc3c46d7d839193b4d9e91a
--- /dev/null
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/EventCodes.java
@@ -0,0 +1,9 @@
+package org.duniter.elasticsearch.service.event;
+
+/**
+ * Created by blavenie on 29/11/16.
+ */
+public enum EventCodes {
+
+    NODE_STARTED
+}
diff --git a/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/EventService.java b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/EventService.java
new file mode 100644
index 0000000000000000000000000000000000000000..745063d57705025440ba88191c78d4b23e409403
--- /dev/null
+++ b/duniter4j-elasticsearch/src/main/java/org/duniter/elasticsearch/service/event/EventService.java
@@ -0,0 +1,338 @@
+package org.duniter.elasticsearch.service.event;
+
+/*
+ * #%L
+ * UCoin Java Client :: Core API
+ * %%
+ * Copyright (C) 2014 - 2015 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.core.JsonProcessingException;
+import com.fasterxml.jackson.databind.JsonNode;
+import org.duniter.core.exception.TechnicalException;
+import org.duniter.core.service.CryptoService;
+import org.duniter.core.service.MailService;
+import org.duniter.core.util.CollectionUtils;
+import org.duniter.core.util.StringUtils;
+import org.duniter.core.util.crypto.CryptoUtils;
+import org.duniter.core.util.crypto.KeyPair;
+import org.duniter.elasticsearch.PluginSettings;
+import org.duniter.elasticsearch.service.AbstractService;
+import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder;
+import org.elasticsearch.action.index.IndexResponse;
+import org.elasticsearch.client.Client;
+import org.elasticsearch.common.inject.Inject;
+import org.elasticsearch.common.settings.Settings;
+import org.elasticsearch.common.xcontent.XContentBuilder;
+import org.elasticsearch.common.xcontent.XContentFactory;
+import org.nuiton.i18n.I18n;
+
+import java.io.IOException;
+import java.util.Locale;
+
+/**
+ * Created by Benoit on 30/03/2015.
+ */
+public class EventService extends AbstractService {
+
+    public static final String INDEX = "user";
+    public static final String EVENT_TYPE = "event";
+
+    private final MailService mailService;
+    public final KeyPair nodeKeyPair;
+    public final String nodePubkey;
+
+    @Inject
+    public EventService(Client client, PluginSettings settings, CryptoService cryptoService, MailService mailService) {
+        super("duniter.event." + INDEX, client, settings, cryptoService);
+        this.mailService = mailService;
+        this.nodeKeyPair = getNodeKeyPairOrNull(pluginSettings);
+        this.nodePubkey = getNodePubKey(this.nodeKeyPair);
+    }
+
+    /**
+     * Notify cluster admin
+     */
+    public void notifyAdmin(Event event) {
+        Locale locale = I18n.getDefaultLocale(); // TODO get locale from admin
+
+        // Add new event to index
+        if (StringUtils.isNotBlank(nodePubkey)) {
+            indexEvent(nodePubkey, locale, event);
+        }
+
+        // Retrieve admin email
+        String adminEmail = pluginSettings.getMailAdmin();
+        if (StringUtils.isBlank(adminEmail) && StringUtils.isNotBlank(nodePubkey)) {
+            adminEmail = getEmailByPk(nodePubkey);
+        }
+
+        // Send email to admin
+        if (StringUtils.isNotBlank(adminEmail)) {
+            String subjectPrefix = pluginSettings.getMailSubjectPrefix();
+            sendEmail(adminEmail,
+                    I18n.l(locale, "duniter4j.event.subject."+event.getType().name(), subjectPrefix),
+                    event.getLocalizedMessage(locale));
+        }
+    }
+
+    /**
+     * Notify a user
+     */
+    public void notifyUser(String recipient, Event event) {
+
+        String email = getEmailByPk(recipient);
+        Locale locale = I18n.getDefaultLocale(); // TODO get locale
+
+        // Add new event to index
+        indexEvent(recipient, locale, event);
+
+        // Send email to user
+        if (StringUtils.isNotBlank(email)) {
+            String subjectPrefix = pluginSettings.getMailSubjectPrefix();
+            sendEmail(email,
+                    I18n.l(locale, "duniter4j.event.subject."+event.getType().name(), subjectPrefix),
+                    event.getLocalizedMessage(locale));
+        }
+
+    }
+
+    /**
+     * Delete blockchain index, and all data
+     * @throws JsonProcessingException
+     */
+    public EventService deleteIndex() {
+        deleteIndexIfExists(INDEX);
+        return this;
+    }
+
+    public boolean existsIndex() {
+        return super.existsIndex(INDEX);
+    }
+
+    /**
+     * Create index need for blockchain registry, if need
+     */
+    public EventService createIndexIfNotExists() {
+        try {
+            if (!existsIndex(INDEX)) {
+                createIndex();
+            }
+        }
+        catch(JsonProcessingException e) {
+            throw new TechnicalException(String.format("Error while creating index [%s]", INDEX));
+        }
+
+        return this;
+    }
+
+    /**
+     * Create index need for category registry
+     * @throws JsonProcessingException
+     */
+    public EventService createIndex() throws JsonProcessingException {
+        logger.info(String.format("Creating index [%s/%s]", INDEX, EVENT_TYPE));
+
+        CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX);
+        Settings indexSettings = Settings.settingsBuilder()
+                .put("number_of_shards", 2)
+                .put("number_of_replicas", 1)
+                //.put("analyzer", createDefaultAnalyzer())
+                .build();
+        createIndexRequestBuilder.setSettings(indexSettings);
+        createIndexRequestBuilder.addMapping(EVENT_TYPE, createEventType());
+        createIndexRequestBuilder.execute().actionGet();
+
+        return this;
+    }
+
+    public String indexEvent(String recipient, Locale locale, Event event) {
+        // Generate json
+        String eventJson;
+        if (StringUtils.isNotBlank(nodePubkey)) {
+            eventJson = toJson(nodePubkey, recipient, locale, event, null);
+            String signature = cryptoService.sign(eventJson, nodeKeyPair.getSecKey());
+            eventJson = toJson(nodePubkey, recipient, locale, event, signature);
+        } else {
+            // Node has not keyring : TODO no issuer ?
+            eventJson = toJson(recipient, recipient, locale, event, null);
+        }
+
+        if (logger.isDebugEnabled()) {
+            logger.debug(String.format("Indexing a event to recipient [%s]", recipient.substring(0, 8)));
+        }
+
+        // do indexation
+        return indexEvent(eventJson, false /*checkSignature*/);
+    }
+
+    public String indexEvent(String eventJson) {
+        return indexEvent(eventJson, true);
+    }
+
+    public String indexEvent(String eventJson, boolean checkSignature) {
+
+        if (checkSignature) {
+            JsonNode jsonNode = readAndVerifyIssuerSignature(eventJson);
+            String recipient = jsonNode.get(org.duniter.core.client.model.elasticsearch.Event.PROPERTY_ISSUER).asText();
+            if (logger.isDebugEnabled()) {
+                logger.debug(String.format("Indexing a event to recipient [%s]", recipient.substring(0, 8)));
+            }
+        }
+        if (logger.isTraceEnabled()) {
+            logger.trace(eventJson);
+        }
+
+        IndexResponse response = client.prepareIndex(INDEX, EVENT_TYPE)
+                .setSource(eventJson)
+                .setRefresh(false)
+                .execute().actionGet();
+
+        return response.getId();
+    }
+
+    /* -- Internal methods -- */
+
+    public XContentBuilder createEventType() {
+        try {
+            XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(EVENT_TYPE)
+                    .startObject("properties")
+
+                    // type
+                    .startObject("type")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // issuer
+                    .startObject("issuer")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // recipient
+                    .startObject("recipient")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // time
+                    .startObject("time")
+                    .field("type", "integer")
+                    .endObject()
+
+                    // code
+                    .startObject("code")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // params
+                    .startObject("params")
+                    .field("type", "string")
+                    .endObject()
+
+                    // message
+                    .startObject("message")
+                    .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", INDEX, EVENT_TYPE, ioe.getMessage()), ioe);
+        }
+    }
+
+    private String getEmailByPk(String issuerPk) {
+        return "benoit.lavenier@e-is.pro";
+    }
+
+    private String getEmailSubject(Locale locale, Event event) {
+
+        return  I18n.l(locale, "duniter4j.event.subject."+event.getType().name());
+    }
+
+    /**
+     * Send email
+     */
+    private void sendEmail(String recipients, String subject, String textContent) {
+        String smtpHost =  pluginSettings.getMailSmtpHost();
+        int smtpPort =  pluginSettings.getMailSmtpPort();
+        String smtpUsername =  pluginSettings.getMailSmtpUsername();
+        String smtpPassword =  pluginSettings.getMailSmtpPassword();
+        String from =  pluginSettings.getMailFrom();
+
+        try {
+            mailService.sendTextEmail(smtpHost, smtpPort, smtpUsername, smtpPassword, from, recipients, subject, textContent);
+        }
+        catch(TechnicalException e) {
+            if (logger.isDebugEnabled()) {
+                logger.error(String.format("Error while trying to send email: %s", e.getMessage()), e);
+            }
+            else {
+                logger.error(String.format("Error while trying to send email: %s", e.getMessage()));
+            }
+        }
+    }
+
+    private String toJson(String issuer, String recipient, Locale locale, Event event, String signature) {
+        try {
+            XContentBuilder eventObject = XContentFactory.jsonBuilder().startObject()
+                    .field("type", event.getType().name())
+                    .field("issuer", issuer) // TODO isuer = node pubkey
+                    .field("recipient", recipient)
+                    .field("time", event.getTime())
+                    .field("code", event.getCode())
+                    .field("message", event.getLocalizedMessage(locale));
+            if (CollectionUtils.isNotEmpty(event.getParams())) {
+                eventObject.array("params", event.getParams());
+            }
+            if (StringUtils.isNotBlank(signature)) {
+                eventObject.field("signature", signature);
+            }
+            eventObject.endObject();
+            return eventObject.string();
+        }
+        catch(IOException e) {
+            throw new TechnicalException(e);
+        }
+
+    }
+
+    private KeyPair getNodeKeyPairOrNull(PluginSettings pluginSettings) {
+
+        if (StringUtils.isNotBlank(pluginSettings.getKeyringSalt()) &&
+                StringUtils.isNotBlank(pluginSettings.getKeyringPassword())) {
+            return cryptoService.getKeyPair(pluginSettings.getKeyringSalt(),
+                    pluginSettings.getKeyringPassword());
+        }
+
+        return null;
+    }
+
+    private String getNodePubKey(KeyPair nodeKeyPair) {
+        if (nodeKeyPair == null) return null;
+        return CryptoUtils.encodeBase58(nodeKeyPair.getPubKey());
+    }
+}
diff --git a/duniter4j-elasticsearch/src/main/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-elasticsearch/src/main/resources/META-INF/services/org.duniter.core.beans.Bean
index 061f29c029c33d670d9a5a18dfaf66e233b482ae..0f39d76137c59998de28bf005b0fb022bd52251e 100644
--- a/duniter4j-elasticsearch/src/main/resources/META-INF/services/org.duniter.core.beans.Bean
+++ b/duniter4j-elasticsearch/src/main/resources/META-INF/services/org.duniter.core.beans.Bean
@@ -4,6 +4,7 @@ org.duniter.core.client.service.bma.WotRemoteServiceImpl
 org.duniter.core.client.service.bma.TransactionRemoteServiceImpl
 org.duniter.core.client.service.elasticsearch.CurrencyRegistryRemoteServiceImpl
 org.duniter.core.service.Ed25519CryptoServiceImpl
+org.duniter.core.service.MailServiceImpl
 org.duniter.core.client.service.HttpServiceImpl
 org.duniter.core.client.service.DataContext
 org.duniter.core.client.service.local.PeerServiceImpl
diff --git a/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_en_GB.properties b/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_en_GB.properties
index a6067b359c9b1b3288467e75f3b34faaa2ce093d..d5a57ab3a335453416ed273e845cb91391aec15c 100644
--- a/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_en_GB.properties
+++ b/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_en_GB.properties
@@ -34,6 +34,10 @@ duniter4j.config.option.tasks.queueCapacity.description=
 duniter4j.config.option.temp.directory.description=
 duniter4j.config.option.version.description=
 duniter4j.config.parse.error=
+duniter4j.event.NODE_STARTED=Node started on cluster Duniter4j ES [%s]
+duniter4j.event.subject.ERROR=[%s] Error message
+duniter4j.event.subject.INFO=[%s] Information message
+duniter4j.event.subject.WARN=[%s] Warning message
 duniter4j.executor.task.waitingExecution=
 duniter4j.job.stopped=
 duniter4j.job.stopping=
diff --git a/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_fr_FR.properties b/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_fr_FR.properties
index 1ed787a11c6d3a2ab895f504f3a0940cdd8bc226..5c2ac7bf0f54f978cc5aa836d456b53f56dd835b 100644
--- a/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_fr_FR.properties
+++ b/duniter4j-elasticsearch/src/main/resources/i18n/duniter4j-elasticsearch_fr_FR.properties
@@ -4,7 +4,7 @@ duniter4j.blockIndexerService.detectFork.invalidBlockchain=[%s] [%s] Peer has an
 duniter4j.blockIndexerService.detectFork.remoteBlockNotFound=[%s] [%s] Unable to get block \#%s from peer\: %s
 duniter4j.blockIndexerService.detectFork.resync=[%s] [%s] Rollback index from block \#%s, and resync
 duniter4j.blockIndexerService.indexBlock=[%s] [%s] Indexing block \#%s - hash [%s]
-duniter4j.blockIndexerService.indexLastBlocks.invalidBlockchain=
+duniter4j.blockIndexerService.indexLastBlocks.invalidBlockchain=[%s] [%s] Peer has another blockchain (no common blocks \!). Skipping last blocks indexation.
 duniter4j.blockIndexerService.indexLastBlocks.otherPeers.task=Indexing missing blocks of [%s] from other peers
 duniter4j.blockIndexerService.indexLastBlocks.progress=[%s] [%s] Indexing block \#%s / %s (%s%%)...
 duniter4j.blockIndexerService.indexLastBlocks.remoteParametersError=[%s] Error when calling [/blockchain/parameters]\: %s
@@ -34,6 +34,10 @@ duniter4j.config.option.tasks.queueCapacity.description=
 duniter4j.config.option.temp.directory.description=
 duniter4j.config.option.version.description=
 duniter4j.config.parse.error=
+duniter4j.event.NODE_STARTED=Noeud démarré sur le cluster Duniter4j ES [%s]
+duniter4j.event.subject.ERROR=%s Message d'erreur
+duniter4j.event.subject.INFO=%s Message d'information
+duniter4j.event.subject.WARN=%s Message d'avertissement
 duniter4j.executor.task.waitingExecution=
 duniter4j.job.stopped=
 duniter4j.job.stopping=
@@ -42,4 +46,3 @@ duniter4j.removeServiceUtils.waitThenRetry=Remote request failed [%s]. Waiting t
 duniter4j.task.issuer.system=Système
 duniter4j.task.starting=Démarrage du traitement...
 duniter4j.threadPool.clusterHealthStatus.changed=Cluster health status changed to [%s]. Executing pending job...
-uniter4j.blockIndexerService.indexLastBlocks.invalidBlockchain=[%s] [%s] Peer has another blockchain (no common blocks \!). Skipping last blocks indexation.
diff --git a/duniter4j-elasticsearch/src/test/es-home/config/elasticsearch.yml b/duniter4j-elasticsearch/src/test/es-home/config/elasticsearch.yml
index a8dc50e68730ec3fcfc31373ed75c801dbfd027e..fc10df3236a31e354731c5a630787aec4fecc1c4 100644
--- a/duniter4j-elasticsearch/src/test/es-home/config/elasticsearch.yml
+++ b/duniter4j-elasticsearch/src/test/es-home/config/elasticsearch.yml
@@ -121,7 +121,7 @@ duniter.string.analyzer: french
 #
 # Enabling node blockchain synchronization
 #
-duniter.blockchain.sync.enable: true
+duniter.blockchain.sync.enable: false
 #
 # Duniter node to synchronize
 #
@@ -130,6 +130,9 @@ duniter.port: 9330
 #
 # ---------------------------------- Duniter4j security -------------------------
 #
+duniter.keyring.salt: abc
+duniter.keyring.password: def
+
 # Enable security, to disable HTTP access to the default ES admin API
 #
 duniter.security.enable: false
@@ -148,4 +151,25 @@ duniter.security.enable: false
 #
 duniter.data.sync.enable: false
 #duniter.data.sync.host: data.duniter.fr
-#duniter.data.sync.port: 80
\ No newline at end of file
+#duniter.data.sync.port: 80
+
+# ---------------------------------- Duniter4j SMTP server -------------------------
+#
+# SMTP server configuration (host and port)
+#
+#duniter.mail.smtp.host: localhost
+#duniter.mail.smtp.port: 25
+#
+# Mail 'from' address
+#
+#duniter.mail.from: no-reply@domain.com
+duniter.mail.from: root@EIS-DEV
+#
+# Mail: admin address
+#
+#duniter.mail.admin: user@domain.com
+duniter.mail.admin: blavenie@EIS-DEV
+#
+# Mail subject prefix
+#
+#duniter.mail.subject.prefix: [Duniter4j ES]
diff --git a/duniter4j-elasticsearch/src/test/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-elasticsearch/src/test/resources/META-INF/services/org.duniter.core.beans.Bean
index 7214110991979625a568d261dae19efb24c08e82..1d327a744e4eec6703b417254f5b19dfdbc09fa4 100644
--- a/duniter4j-elasticsearch/src/test/resources/META-INF/services/org.duniter.core.beans.Bean
+++ b/duniter4j-elasticsearch/src/test/resources/META-INF/services/org.duniter.core.beans.Bean
@@ -3,6 +3,7 @@ org.duniter.core.client.service.bma.NetworkRemoteServiceImpl
 org.duniter.core.client.service.bma.WotRemoteServiceImpl
 org.duniter.core.client.service.bma.TransactionRemoteServiceImpl
 org.duniter.core.service.Ed25519CryptoServiceImpl
+org.duniter.core.service.MailServiceImpl
 org.duniter.core.client.service.HttpServiceImpl
 org.duniter.core.client.service.DataContext
 org.duniter.core.client.service.local.PeerServiceImpl
diff --git a/pom.xml b/pom.xml
index 7b75013c03cdbb31cc9fe2c3ce9a40917034c84d..493d3164aa66694d6f2d004720a4e72463c8e77c 100644
--- a/pom.xml
+++ b/pom.xml
@@ -314,6 +314,11 @@
         <artifactId>javax.websocket-api</artifactId>
         <version>1.1</version>
       </dependency>
+      <dependency>
+        <groupId>javax.mail</groupId>
+        <artifactId>mail</artifactId>
+        <version>1.4.7</version>
+      </dependency>
 
         <!-- NaCL lib -->
       <dependency>