From 99c72b492943d63e8b3f687fb6f72c926c074548 Mon Sep 17 00:00:00 2001
From: blavenie <benoit.lavenier@e-is.pro>
Date: Fri, 3 Mar 2017 18:46:08 +0100
Subject: [PATCH] - add index for invitations - update group index: remove
 group name (use attribute 'id' as ES id)

---
 .../client/model/elasticsearch/UserGroup.java |  10 -
 .../src/test/es-home/config/elasticsearch.yml |  10 +-
 .../service/AbstractService.java              |   3 +-
 .../service/AbstractSynchroService.java       |   1 +
 .../elasticsearch/user/PluginInit.java        |   4 +
 .../user/model/UserEventCodes.java            |   4 +-
 .../elasticsearch/user/rest/RestModule.java   |   6 +
 ...estInvitationCertificationIndexAction.java |  44 ++++
 ...vitationCertificationMarkAsReadAction.java |  44 ++++
 .../user/service/GroupService.java            |  67 ++++--
 .../user/service/HistoryService.java          |   6 +-
 .../user/service/MessageService.java          |   8 +-
 .../user/service/ServiceModule.java           |   2 +
 .../user/service/SynchroService.java          |   5 +
 .../user/service/UserInvitationService.java   | 215 ++++++++++++++++++
 duniter4j-es-user/src/main/misc/curl_test.sh  |  28 ++-
 .../src/main/misc/query_many_indices.sh       |  17 ++
 .../i18n/duniter4j-es-user_en_GB.properties   |   1 +
 .../i18n/duniter4j-es-user_fr_FR.properties   |   1 +
 19 files changed, 429 insertions(+), 47 deletions(-)
 create mode 100644 duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationIndexAction.java
 create mode 100644 duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationMarkAsReadAction.java
 create mode 100644 duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java
 create mode 100755 duniter4j-es-user/src/main/misc/query_many_indices.sh

diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/UserGroup.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/UserGroup.java
index 4144d10d..1f3f5126 100644
--- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/UserGroup.java
+++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/UserGroup.java
@@ -27,24 +27,14 @@ package org.duniter.core.client.model.elasticsearch;
  */
 public class UserGroup extends Record {
 
-    public static final String PROPERTY_NAME="name";
     public static final String PROPERTY_TITLE="title";
     public static final String PROPERTY_DESCRIPTION="description";
     public static final String PROPERTY_CREATION_TIME="creationTime";
 
-    private String name;
     private String title;
     private String description;
     private Long creationTime;
 
-    public String getName() {
-        return name;
-    }
-
-    public void setName(String name) {
-        this.name = name;
-    }
-
     public String getTitle() {
         return title;
     }
diff --git a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml
index 850b5ddd..6bd2985f 100644
--- a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml
+++ b/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml
@@ -138,8 +138,8 @@ duniter.keyring.password: def
 
 # Enable security, to disable HTTP access to the default ES admin API
 #
-#duniter.security.enable: false
-duniter.security.enable: true
+duniter.security.enable: false
+#duniter.security.enable: true
 #
 # Security token prefix (default: 'duniter-')
 #
@@ -153,9 +153,9 @@ duniter.security.enable: true
 #
 # Should synchronize data using P2P
 #
-duniter.data.sync.enable: false
-#duniter.data.sync.host: data.duniter.fr
-#duniter.data.sync.port: 80
+duniter.data.sync.enable: true
+duniter.data.sync.host: data.gtest.duniter.fr
+duniter.data.sync.port: 80
 
 # ---------------------------------- Duniter4j SMTP server -------------------------
 #
diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractService.java
index 93b705e0..eb115021 100644
--- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractService.java
+++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractService.java
@@ -237,7 +237,6 @@ public abstract class AbstractService implements Bean {
         }
     }
 
-
     protected String getIssuer(JsonNode actualObj) {
         return  getMandatoryField(actualObj, Record.PROPERTY_ISSUER).asText();
     }
@@ -259,7 +258,7 @@ public abstract class AbstractService implements Bean {
         Map<String, Object> fields = getFieldsById(index, type, docId, fieldNames);
         if (MapUtils.isEmpty(fields)) throw new NotFoundException(String.format("Document [%s/%s/%s] not exists.", index, type, docId));
         Arrays.stream(fieldNames).forEach((fieldName) -> {
-            if (!fields.containsKey(fieldName)) throw new NotFoundException(String.format("Document [%s/%s/%s] should have the madatory field [%s].", index, type, docId, fieldName));
+            if (!fields.containsKey(fieldName)) throw new NotFoundException(String.format("Document [%s/%s/%s] should have the mandatory field [%s].", index, type, docId, fieldName));
         });
         return fields;
     }
diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractSynchroService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractSynchroService.java
index dca7183e..1ef7130b 100644
--- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractSynchroService.java
+++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractSynchroService.java
@@ -172,6 +172,7 @@ public abstract class AbstractSynchroService extends AbstractService {
         JsonNode node;
         try {
             HttpPost httpPost = new HttpPost(httpService.getPath(peer, fromIndex, fromType, "_search"));
+            httpPost.setHeader("Content-Type", "application/json;charset=UTF-8");
             httpPost.setEntity(new ByteArrayEntity(bos.bytes().array()));
             if (logger.isDebugEnabled()) {
                 logger.debug(String.format("[%s] [%s/%s] Sending POST request: %s", peer, fromIndex, fromType, new String(bos.bytes().array())));
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java
index 9d05f816..c827707d 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java
@@ -106,6 +106,9 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> {
             injector.getInstance(GroupService.class)
                     .deleteIndex()
                     .createIndexIfNotExists();
+            injector.getInstance(UserInvitationService.class)
+                    .deleteIndex()
+                    .createIndexIfNotExists();
 
             if (logger.isInfoEnabled()) {
                 logger.info("Reloading all Duniter indices... [OK]");
@@ -119,6 +122,7 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> {
             injector.getInstance(UserService.class).createIndexIfNotExists();
             injector.getInstance(MessageService.class).createIndexIfNotExists();
             injector.getInstance(GroupService.class).createIndexIfNotExists();
+            injector.getInstance(UserInvitationService.class).createIndexIfNotExists();
 
             if (logger.isInfoEnabled()) {
                 logger.info("Checking Duniter indices... [OK]");
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEventCodes.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEventCodes.java
index 8ff71637..db84a9a7 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEventCodes.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEventCodes.java
@@ -45,6 +45,8 @@ public enum UserEventCodes {
     CERT_RECEIVED,
 
     // Message
-    MESSAGE_RECEIVED
+    MESSAGE_RECEIVED,
 
+    // Invitation
+    INVITATION_TO_CERTIFY
 }
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/RestModule.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/RestModule.java
index fc6d2ded..db6d8302 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/RestModule.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/RestModule.java
@@ -25,6 +25,8 @@ package org.duniter.elasticsearch.user.rest;
 import org.duniter.elasticsearch.user.rest.group.RestGroupIndexAction;
 import org.duniter.elasticsearch.user.rest.group.RestGroupUpdateAction;
 import org.duniter.elasticsearch.user.rest.history.RestHistoryDeleteIndexAction;
+import org.duniter.elasticsearch.user.rest.invitation.RestInvitationCertificationIndexAction;
+import org.duniter.elasticsearch.user.rest.invitation.RestInvitationCertificationMarkAsReadAction;
 import org.duniter.elasticsearch.user.rest.message.RestMessageInboxIndexAction;
 import org.duniter.elasticsearch.user.rest.message.RestMessageInboxMarkAsReadAction;
 import org.duniter.elasticsearch.user.rest.message.RestMessageOutboxIndexAction;
@@ -61,6 +63,10 @@ public class RestModule extends AbstractModule implements Module {
         bind(RestMessageOutboxIndexAction.class).asEagerSingleton();
         bind(RestMessageInboxMarkAsReadAction.class).asEagerSingleton();
 
+        // Invitation
+        bind(RestInvitationCertificationIndexAction.class).asEagerSingleton();
+        bind(RestInvitationCertificationMarkAsReadAction.class).asEagerSingleton();
+
         // Backward compatibility
         {
             // message/record
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationIndexAction.java
new file mode 100644
index 00000000..b33c7af9
--- /dev/null
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationIndexAction.java
@@ -0,0 +1,44 @@
+package org.duniter.elasticsearch.user.rest.invitation;
+
+/*
+ * #%L
+ * duniter4j-elasticsearch-plugin
+ * %%
+ * Copyright (C) 2014 - 2016 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 org.duniter.elasticsearch.rest.AbstractRestPostIndexAction;
+import org.duniter.elasticsearch.rest.security.RestSecurityController;
+import org.duniter.elasticsearch.user.service.UserInvitationService;
+import org.elasticsearch.client.Client;
+import org.elasticsearch.common.inject.Inject;
+import org.elasticsearch.common.settings.Settings;
+import org.elasticsearch.rest.RestController;
+
+public class RestInvitationCertificationIndexAction extends AbstractRestPostIndexAction {
+
+    @Inject
+    public RestInvitationCertificationIndexAction(Settings settings, RestController controller, Client client,
+                                                  RestSecurityController securityController,
+                                                  final UserInvitationService service) {
+        super(settings, controller, client, securityController,
+                UserInvitationService.INDEX,
+                UserInvitationService.CERTIFICATION_TYPE,
+                json -> service.indexCertificationInvitationFromJson(json));
+    }
+}
\ No newline at end of file
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationMarkAsReadAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationMarkAsReadAction.java
new file mode 100644
index 00000000..6f65826f
--- /dev/null
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationMarkAsReadAction.java
@@ -0,0 +1,44 @@
+package org.duniter.elasticsearch.user.rest.invitation;
+
+/*
+ * #%L
+ * duniter4j-elasticsearch-plugin
+ * %%
+ * Copyright (C) 2014 - 2016 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 org.duniter.elasticsearch.rest.AbstractRestPostMarkAsReadAction;
+import org.duniter.elasticsearch.rest.security.RestSecurityController;
+import org.duniter.elasticsearch.user.service.UserInvitationService;
+import org.elasticsearch.client.Client;
+import org.elasticsearch.common.inject.Inject;
+import org.elasticsearch.common.settings.Settings;
+import org.elasticsearch.rest.RestController;
+
+public class RestInvitationCertificationMarkAsReadAction extends AbstractRestPostMarkAsReadAction {
+
+    @Inject
+    public RestInvitationCertificationMarkAsReadAction(Settings settings, RestController controller, Client client,
+                                                       RestSecurityController securityController,
+                                                       UserInvitationService service) {
+        super(settings, controller, client, securityController, UserInvitationService.INDEX, UserInvitationService.CERTIFICATION_TYPE,
+                (id, signature) -> {
+                    service.markInvitationAsRead(id, signature);
+                });
+    }
+}
\ No newline at end of file
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java
index a1f5f542..8a06c09d 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java
@@ -115,15 +115,17 @@ public class GroupService extends AbstractService {
     public String indexRecordProfileFromJson(String profileJson) {
 
         JsonNode actualObj = readAndVerifyIssuerSignature(profileJson);
-        String name = getName(actualObj);
+        String title = getTitle(actualObj);
+        String id = computeIdFromTitle(title);
+        String issuer = getIssuer(actualObj);
 
         if (logger.isDebugEnabled()) {
-            logger.debug(String.format("Indexing a user profile from issuer [%s]", name.substring(0, 8)));
+            logger.debug(String.format("Indexing group [%s] from issuer [%s]", id, issuer.substring(0, 8)));
         }
 
         IndexResponse response = client.prepareIndex(INDEX, RECORD_TYPE)
                 .setSource(profileJson)
-                .setId(name) // always use the name as id
+                .setId(id)
                 .setRefresh(false)
                 .execute().actionGet();
         return response.getId();
@@ -133,39 +135,29 @@ public class GroupService extends AbstractService {
      * Update a record
      * @param recordJson
      */
-    public ListenableActionFuture<UpdateResponse> updateRecordFromJson(String recordJson, String id) {
+    public ListenableActionFuture<UpdateResponse> updateRecordFromJson(String id, String recordJson) {
 
         JsonNode actualObj = readAndVerifyIssuerSignature(recordJson);
-        String name = getName(actualObj);
 
-        if (!Objects.equals(name, id)) {
-            throw new AccessDeniedException(String.format("Could not update this document: not issuer."));
-        }
         if (logger.isDebugEnabled()) {
-            logger.debug(String.format("Updating a group from name [%s]", name));
+            logger.debug(String.format("Updating group [%s]", id));
         }
 
-        return client.prepareUpdate(INDEX, RECORD_TYPE, name)
+        return client.prepareUpdate(INDEX, RECORD_TYPE, id)
                 .setDoc(recordJson)
                 .execute();
     }
 
+    public String getTitleById(String id) {
 
-
-    protected String getName(JsonNode actualObj) {
-        return  getMandatoryField(actualObj, UserGroup.PROPERTY_NAME).asText();
-    }
-
-    public String getTitleByName(String name) {
-
-        Object title = getFieldById(INDEX, RECORD_TYPE, name, UserGroup.PROPERTY_NAME);
+        Object title = getFieldById(INDEX, RECORD_TYPE, id, UserGroup.PROPERTY_TITLE);
         if (title == null) return null;
         return title.toString();
     }
 
-    public Map<String, String> getTitlesByNames(Set<String> names) {
+    public Map<String, String> getTitlesByNames(Set<String> ids) {
 
-        Map<String, Object> titles = getFieldByIds(INDEX, RECORD_TYPE, names, UserGroup.PROPERTY_NAME);
+        Map<String, Object> titles = getFieldByIds(INDEX, RECORD_TYPE, ids, UserGroup.PROPERTY_TITLE);
         if (MapUtils.isEmpty(titles)) return null;
         Map<String, String> result = new HashMap<>();
         titles.entrySet().stream().forEach((entry) -> result.put(entry.getKey(), entry.getValue().toString()));
@@ -175,6 +167,29 @@ public class GroupService extends AbstractService {
     /* -- Internal methods -- */
 
 
+    protected String getTitle(JsonNode actualObj) {
+        return  getMandatoryField(actualObj, UserGroup.PROPERTY_TITLE).asText();
+    }
+
+    protected String computeIdFromTitle(String title) {
+        return computeIdFromTitle(title, 0);
+    }
+
+    protected String computeIdFromTitle(String title, int counter) {
+
+        String id = title.replaceAll("\\s+", "");
+        id  = id.replaceAll("[^a-zA−Z_-]+", "");
+        if (counter > 0) {
+            id += "_" + counter;
+        }
+
+        if (!isDocumentExists(INDEX, RECORD_TYPE, id)) {
+            return id;
+        }
+
+        return computeIdFromTitle(title, counter+1);
+    }
+
     public XContentBuilder createRecordType() {
         String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer();
 
@@ -210,6 +225,18 @@ public class GroupService extends AbstractService {
                     .field("index", "not_analyzed")
                     .endObject()
 
+                    // hash
+                    .startObject("hash")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // signature
+                    .startObject("signature")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
                     // avatar
                     .startObject("avatar")
                         .field("type", "attachment")
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java
index beaf8da4..8cfb537b 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java
@@ -124,10 +124,14 @@ public class HistoryService extends AbstractService {
             throw new NotFoundException(String.format("Index [%s] not exists.", index));
         }
 
-        // Special case for message: check if issuer is recipient
+        // Special case for message: check if deletion issuer is the message recipient
         if (MessageService.INDEX.equals(index) && MessageService.INBOX_TYPE.equals(type)) {
             checkSameDocumentField(index, type, id, MessageRecord.PROPERTY_RECIPIENT, issuer);
         }
+        // Special case for invitation: check if deletion issuer is the invitation recipient
+        else if (UserInvitationService.INDEX.equals(index)) {
+            checkSameDocumentField(index, type, id, MessageRecord.PROPERTY_RECIPIENT, issuer);
+        }
         else {
             // Check document issuer
             checkSameDocumentIssuer(index, type, id, issuer);
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java
index 71dee94f..40bb0e15 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java
@@ -218,7 +218,13 @@ public class MessageService extends AbstractService {
                     .field("index", "not_analyzed")
                     .endObject()
 
-                    // content
+                    // title (encrypted)
+                    .startObject("title")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // content (encrypted)
                     .startObject("content")
                     .field("type", "string")
                     .field("index", "not_analyzed")
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java
index c0b55796..e1986861 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java
@@ -37,6 +37,8 @@ public class ServiceModule extends AbstractModule implements Module {
 
         bind(UserEventService.class).asEagerSingleton();
 
+        bind(UserInvitationService.class).asEagerSingleton();
+
         bind(BlockchainUserEventService.class).asEagerSingleton();
 
         bind(SynchroService.class).asEagerSingleton();
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/SynchroService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/SynchroService.java
index f39c137e..6d683a46 100644
--- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/SynchroService.java
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/SynchroService.java
@@ -67,6 +67,7 @@ public class SynchroService extends AbstractSynchroService {
         importUserChanges(result, peer, sinceTime);
         importMessageChanges(result, peer, sinceTime);
         importGroupChanges(result, peer, sinceTime);
+        importInvitationChanges(result, peer, sinceTime);
 
         long duration = System.currentTimeMillis() - time;
         logger.info(String.format("[%s] Synchronizing user data since %s [OK] %s (ins %s ms)", peer.toString(), sinceTime, result.toString(), duration));
@@ -96,4 +97,8 @@ public class SynchroService extends AbstractSynchroService {
         importChanges(result, peer, GroupService.INDEX, GroupService.RECORD_TYPE,  sinceTime);
     }
 
+    protected void importInvitationChanges(SynchroResult result, Peer peer, long sinceTime) {
+        importChanges(result, peer, UserInvitationService.INDEX, UserInvitationService.CERTIFICATION_TYPE,  sinceTime);
+    }
+
 }
diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java
new file mode 100644
index 00000000..aa0548d4
--- /dev/null
+++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java
@@ -0,0 +1,215 @@
+package org.duniter.elasticsearch.user.service;
+
+/*
+ * #%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.client.model.ModelUtils;
+import org.duniter.core.exception.TechnicalException;
+import org.duniter.core.service.CryptoService;
+import org.duniter.elasticsearch.exception.InvalidSignatureException;
+import org.duniter.elasticsearch.user.PluginSettings;
+import org.duniter.elasticsearch.user.model.Message;
+import org.duniter.elasticsearch.user.model.UserEvent;
+import org.duniter.elasticsearch.user.model.UserEventCodes;
+import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder;
+import org.elasticsearch.action.index.IndexResponse;
+import org.elasticsearch.action.update.UpdateRequestBuilder;
+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.Map;
+
+/**
+ * Created by Benoit on 30/03/2015.
+ */
+public class UserInvitationService extends AbstractService {
+
+    public static final String INDEX = "invitation";
+    public static final String CERTIFICATION_TYPE = "certification";
+
+
+    private final UserEventService userEventService;
+
+    @Inject
+    public UserInvitationService(Client client, PluginSettings settings,
+                                 CryptoService cryptoService, UserEventService userEventService) {
+        super("duniter." + INDEX, client, settings, cryptoService);
+        this.userEventService = userEventService;
+    }
+
+    /**
+     * Delete blockchain index, and all data
+     * @throws JsonProcessingException
+     */
+    public UserInvitationService deleteIndex() {
+        deleteIndexIfExists(INDEX);
+        return this;
+    }
+
+    public boolean existsIndex() {
+        return super.existsIndex(INDEX);
+    }
+
+    /**
+     * Create index need for blockchain registry, if need
+     */
+    public UserInvitationService 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 UserInvitationService createIndex() throws JsonProcessingException {
+        logger.info(String.format("Creating index [%s/%s]", INDEX, CERTIFICATION_TYPE));
+
+        CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX);
+        Settings indexSettings = Settings.settingsBuilder()
+                .put("number_of_shards", 2)
+                .put("number_of_replicas", 1)
+                .build();
+        createIndexRequestBuilder.setSettings(indexSettings);
+        createIndexRequestBuilder.addMapping(CERTIFICATION_TYPE, createCertificationType());
+        createIndexRequestBuilder.execute().actionGet();
+
+        return this;
+    }
+
+    public String indexCertificationInvitationFromJson(String recordJson) {
+
+        JsonNode actualObj = readAndVerifyIssuerSignature(recordJson);
+        String issuer = getIssuer(actualObj);
+        String recipient = getMandatoryField(actualObj, Message.PROPERTY_RECIPIENT).asText();
+        Long time = getMandatoryField(actualObj, Message.PROPERTY_TIME).asLong();
+
+        if (logger.isDebugEnabled()) {
+            logger.debug(String.format("Indexing a invitation to certify from issuer [%s]", issuer.substring(0, 8)));
+        }
+
+        IndexResponse response = client.prepareIndex(INDEX, CERTIFICATION_TYPE)
+                .setSource(recordJson)
+                .setRefresh(false)
+                .execute().actionGet();
+
+        String invitationId = response.getId();
+
+        // Notify recipient
+        userEventService.notifyUser(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.INVITATION_TO_CERTIFY.name())
+                .setRecipient(recipient)
+                .setMessage(I18n.n("duniter.invitation.cert.received"), issuer, ModelUtils.minifyPubkey(issuer))
+                .setTime(time)
+                .setReference(INDEX, CERTIFICATION_TYPE, invitationId)
+                .build());
+
+        return invitationId;
+    }
+
+    public void markInvitationAsRead(String id, String signature) {
+        Map<String, Object> fields = getMandatoryFieldsById(INDEX, CERTIFICATION_TYPE, id, Message.PROPERTY_HASH, Message.PROPERTY_RECIPIENT);
+        String recipient = fields.get(UserEvent.PROPERTY_RECIPIENT).toString();
+        String hash = fields.get(UserEvent.PROPERTY_HASH).toString();
+
+        // Check signature
+        boolean valid = cryptoService.verify(hash, signature, recipient);
+        if (!valid) {
+            throw new InvalidSignatureException("Invalid signature: only the recipient can mark an message as read.");
+        }
+
+        UpdateRequestBuilder request = client.prepareUpdate(INDEX, CERTIFICATION_TYPE, id)
+                .setDoc("read_signature", signature);
+        request.execute();
+    }
+
+    /* -- Internal methods -- */
+
+    public XContentBuilder createCertificationType() {
+        return createMapping(CERTIFICATION_TYPE);
+    }
+
+    public XContentBuilder createMapping(String typeName) {
+        try {
+            XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(typeName)
+                    .startObject("properties")
+
+                    // 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()
+
+                    // nonce
+                    .startObject("nonce")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // content (encrypted)
+                    .startObject("content")
+                    .field("type", "string")
+                    .field("index", "not_analyzed")
+                    .endObject()
+
+                    // read_signature
+                    .startObject("read_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", INDEX, CERTIFICATION_TYPE, ioe.getMessage()), ioe);
+        }
+    }
+}
diff --git a/duniter4j-es-user/src/main/misc/curl_test.sh b/duniter4j-es-user/src/main/misc/curl_test.sh
index 1c88ff48..be2865e1 100755
--- a/duniter4j-es-user/src/main/misc/curl_test.sh
+++ b/duniter4j-es-user/src/main/misc/curl_test.sh
@@ -88,20 +88,34 @@ curl -XPOST "http://127.0.0.1:9200/user/event/_search?pretty" -d'
 
 
 echo "--- GET market pictures content_type--- "
-curl -XPOST "http://127.0.0.1:9200/market/record/_search?pretty" -d'
+curl -XPOST "http://data.gtest.duniter.fr/user/profile/_search?pretty" -d'
 {
   query: {
     constant_score: {
         filter: [
-          {term: { issuer: "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"}}
+          {terms: { issuer: ["5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"]}}
         ]
       }
   },
-  sort : [
-    { "time" : {"order" : "desc"}}
-  ],
   from: 0,
-  size: 3,
-  _source: ["avatar._content_type"]
+  size: 100,
+  _source: ["title", "avatar._content_type"]
+}'
+
+
+echo "--- GET user event count --- "
+curl -XPOST "http://data.gtest.duniter.fr/user/event/_search?pretty" -d'
+{
+  from: 0,
+  size: 0,
+  _source: false
 }'
 
+
+echo "--- GET message count --- "
+curl -XPOST "http://data.gtest.duniter.fr/message/record/_search?pretty" -d'
+{
+  from: 0,
+  size: 10,
+  _source: ["nonce"]
+}'
diff --git a/duniter4j-es-user/src/main/misc/query_many_indices.sh b/duniter4j-es-user/src/main/misc/query_many_indices.sh
new file mode 100755
index 00000000..22a30bdb
--- /dev/null
+++ b/duniter4j-es-user/src/main/misc/query_many_indices.sh
@@ -0,0 +1,17 @@
+#!/bin/sh
+
+echo "--- COUNT query --- "
+curl -XPOST "http://127.0.0.1:9200/_search?pretty" -d'
+{
+  query: {
+    indices : {
+            "indices" : ["user", "registry", "currency"],
+            "query" : { "term" : { "tags" : "gtest" } },
+            "no_match_query" : { "term" : { "tags" : "gtest" } }
+            }
+  },
+  from: 0,
+  size: 100
+}'
+
+
diff --git a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties
index 50ee1d02..e149f6a6 100644
--- a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties
+++ b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties
@@ -1,6 +1,7 @@
 duniter.event.NODE_BMA_DOWN=Duniter node [%1$s\:%2$s] is DOWN\: no access from ES node [%3$s]. Last connexion at %4$d. Blockchain indexation waiting.
 duniter.event.NODE_BMA_UP=Duniter node [%1$s\:%2$s] is UP again.
 duniter.event.NODE_STARTED=Node started on cluster Duniter4j ES [%s]
+duniter.invitation.cert.received=
 duniter.user.event.active=
 duniter.user.event.cert.received=
 duniter.user.event.cert.sent=
diff --git a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties
index c41fd1df..ab9df823 100644
--- a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties
+++ b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties
@@ -1,6 +1,7 @@
 duniter.event.NODE_BMA_DOWN=Noeud Duniter [%1$s\:%2$s] non joignable, depuis le noeud ES API [%3$s]. Dernière connexion à %4$d. Indexation de blockchain en attente.
 duniter.event.NODE_BMA_UP=Noeud Duniter [%1$s\:%2$s] à nouveau accessible.
 duniter.event.NODE_STARTED=Noeud ES API démarré sur le cluster Duniter [%1$s]
+duniter.invitation.cert.received=%2$s vous invite à certifier une identité.
 duniter.user.event.cert.received=%2$s vous a certifié (certification prise en compte).
 duniter.user.event.cert.sent=Votre certification de %2$s a été pris en compte.
 duniter.user.event.message.received=Vous avez reçu un message de %2$s
-- 
GitLab