diff --git a/.gitmodules b/.gitmodules index 28770e46d05930c7f2b342a98e70062be3be85ef..97112d748142dd60fbe08f5eb2cc308e332b60a8 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,6 +1,3 @@ -[submodule "duniter4j-cesium/src/main/cesium"] - path = duniter4j-cesium/src/main/cesium - url = git://github.com/duniter/cesium.git [submodule "duniter4j-elasticsearch/src/main/resources/cities"] path = duniter4j-elasticsearch/src/main/resources/cities url = git://github.com/David-Haim/CountriesToCitiesJSON.git diff --git a/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml b/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml index 46095a0f1a870e4e4400f7a142ea12d25fd12e77..184f1a706f38af1e13b7547098686fc32d0b36d2 100644 --- a/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml +++ b/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml @@ -249,4 +249,19 @@ duniter.subscription.enable: false # # Email subscription: URL to a Cesium site, for links in the email content (default: https://g1.duniter.fr) # -# duniter.subscription.email.cesium.url: 'http://domain.com/cesium' \ No newline at end of file +# duniter.subscription.email.cesium.url: 'http://domain.com/cesium' +# +# ---------------------------------- Duniter4j User (profile, message) module ------------------- +# +# +# Share link: `og:site_name` (default: 'Cesium') +# +# duniter.user.share.site.name: 'Cesium - Ğ1' +# +# Share link: `og:url` - URL to a Cesium site, for links in the email content (default: https://g1.duniter.fr) +# +# duniter.share.cesium.url: 'https://domain.com/cesium' +# +# Share link: Base URL of the ES cluster, to resolve `og:image` URL (default: none => /!\ Will use relative image path) +# +# duniter.share.base.url: 'https://data.domain.com' \ No newline at end of file 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 cd42a48aef7696c6608cfdeddb046cd5b3e8763a..124f71b8c5443388105ff26fe60d1b987a37dcab 100644 --- a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml +++ b/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml @@ -251,4 +251,19 @@ duniter.subscription.enable: true # # Email subscription: URL to a Cesium site, for links in the email content (default: https://g1.duniter.fr) # -#duniter.subscription.email.cesium.url: 'https://domain.com/cesium' \ No newline at end of file +#duniter.subscription.email.cesium.url: 'https://domain.com/cesium' +# +# ---------------------------------- Duniter4j User (profile, message) module ------------------- +# +# +# Share link: og:site_name (default: 'Cesium') +# +# duniter.user.share.site.name: 'Cesium - Ğ1' +# +# Share link : URL to a Cesium site, for links in the email content (default: https://g1.duniter.fr) +# +#duniter.share.cesium.url: 'https://domain.com/cesium' +# +# Share link : Base URL of cluster, to resolve image (default: none => /!\ Will use relative image path) +# +#duniter.share.base.url: 'http://localhost:9200' \ No newline at end of file diff --git a/duniter4j-es-core/pom.xml b/duniter4j-es-core/pom.xml index a2d3ebe77f40fd014d06a026471b2eb6e76da2e6..4a5ecfd698a69dacfc8b15a0d2ed4a1e495abfc5 100644 --- a/duniter4j-es-core/pom.xml +++ b/duniter4j-es-core/pom.xml @@ -76,6 +76,14 @@ <artifactId>jackson-databind</artifactId> </dependency> + + <dependency> + <groupId>org.antlr</groupId> + <artifactId>stringtemplate</artifactId> + <version>${stringtemplate.version}</version> + <scope>compile</scope> + </dependency> + <!-- JNA (need for OS shutdown hook) --> <dependency> <groupId>net.java.dev.jna</groupId> diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/attachment/RestImageAttachmentAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/attachment/RestImageAttachmentAction.java index f861333dfd6230a9538538c5b4135ebbe8273bbc..e46a5926a10e170d5e51eadbb46875330ee65794 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/attachment/RestImageAttachmentAction.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/attachment/RestImageAttachmentAction.java @@ -95,4 +95,17 @@ public class RestImageAttachmentAction extends BaseRestHandler { } }); } + + + public static String computeImageUrl(String index, + String type, + String id, + String imageField, + String contentType) { + + int lastSlashIndex = contentType != null ? contentType.lastIndexOf('/') : -1; + String extension = (lastSlashIndex >= 0) ? contentType.substring(lastSlashIndex+1) : contentType; + + return String.format("/%s/%s/%s/_image/%s.%s", index, type, id, imageField, extension); + } } \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/share/AbstractRestShareLinkAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/share/AbstractRestShareLinkAction.java new file mode 100644 index 0000000000000000000000000000000000000000..ff97fada10df8a93d416f847e86196295e3c881b --- /dev/null +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/share/AbstractRestShareLinkAction.java @@ -0,0 +1,91 @@ +package org.duniter.elasticsearch.rest.share; + +import org.apache.http.entity.ContentType; +import org.duniter.core.exception.BusinessException; +import org.duniter.core.util.Preconditions; +import org.duniter.core.util.StringUtils; +import org.duniter.elasticsearch.exception.DuniterElasticsearchException; +import org.duniter.elasticsearch.rest.XContentThrowableRestResponse; +import org.duniter.elasticsearch.util.opengraph.OGData; +import org.duniter.elasticsearch.util.springtemplate.STUtils; +import org.elasticsearch.client.Client; +import org.elasticsearch.common.logging.ESLogger; +import org.elasticsearch.common.logging.Loggers; +import org.elasticsearch.common.settings.Settings; +import org.elasticsearch.rest.*; +import org.nuiton.i18n.I18n; +import org.stringtemplate.v4.*; + +import java.util.Locale; + +import static org.elasticsearch.rest.RestRequest.Method.GET; +import static org.elasticsearch.rest.RestStatus.OK; + +public abstract class AbstractRestShareLinkAction extends BaseRestHandler { + + protected final ESLogger log; + + public interface OGDataResolver { + OGData resolve(String id) throws DuniterElasticsearchException, BusinessException; + } + + private OGDataResolver resolver; + private STGroup templates; + + public AbstractRestShareLinkAction(Settings settings, RestController controller, Client client, + String indexName, + String typeName, + OGDataResolver resolver + ) { + super(settings, controller, client); + log = Loggers.getLogger("duniter.rest." + indexName, settings, String.format("[%s]", indexName)); + controller.registerHandler(GET, + String.format("/%s/%s/{id}/_share", indexName, typeName), + this); + this.resolver = resolver; + + // Configure springtemplate engine + this.templates = STUtils.newSTGroup("org/duniter/elasticsearch/templates"); + Preconditions.checkNotNull(this.templates.getInstanceOf("html_share"), "Unable to load ST template for share page"); + + } + + @Override + protected void handleRequest(final RestRequest request, RestChannel restChannel, Client client) throws Exception { + String id = request.param("id"); + + try { + + OGData data = resolver.resolve(id); + Preconditions.checkNotNull(data); + Preconditions.checkNotNull(data.title); + + // Compute HTML content + ST template = templates.getInstanceOf("html_share"); + template.add("type", data.type); + template.add("title", data.title); + template.add("summary", StringUtils.truncate(data.description, 500)); + template.add("description", data.description); + template.add("siteName", data.siteName); + template.add("image", data.image); + template.add("url", data.url); + template.add("locale", data.locale); + if (StringUtils.isNotBlank(data.url)) { + Locale locale = data.locale != null ? new Locale(data.locale) : I18n.getDefaultLocale(); + template.add("redirectMessage", I18n.l(locale, "duniter4j.share.redirection.help")); + } + + String html = template.render(); + + restChannel.sendResponse(new BytesRestResponse(OK, ContentType.TEXT_HTML.getMimeType(), html)); + } + catch(DuniterElasticsearchException | BusinessException e) { + log.error(e.getMessage(), e); + restChannel.sendResponse(new XContentThrowableRestResponse(request, e)); + } + catch(Exception e) { + log.error(e.getMessage(), e); + } + } + +} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/opengraph/OGData.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/opengraph/OGData.java new file mode 100644 index 0000000000000000000000000000000000000000..0530ead6288cd913cacb01fc7fc11883688e864b --- /dev/null +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/opengraph/OGData.java @@ -0,0 +1,13 @@ +package org.duniter.elasticsearch.util.opengraph; + +public class OGData { + public String type; + public String title; + public String description; + public String image; + public String url; + public String locale; + public String imageType; + public String siteName; + +} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/DateRenderer.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/DateRenderer.java new file mode 100644 index 0000000000000000000000000000000000000000..507fd6169aa0a88bfac49f54f0117eb58e2bf46a --- /dev/null +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/DateRenderer.java @@ -0,0 +1,67 @@ +package org.duniter.elasticsearch.util.springtemplate; + +/*- + * #%L + * Duniter4j :: ElasticSearch Subscription 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 org.stringtemplate.v4.AttributeRenderer; + +import java.text.DateFormat; +import java.text.SimpleDateFormat; +import java.util.Calendar; +import java.util.Date; +import java.util.Locale; + +public class DateRenderer implements AttributeRenderer { + + public DateRenderer() { + } + + public String toString(Object o, String formatString, Locale locale) { + if(formatString == null) { + formatString = "short"; + } + + Date d; + if(o instanceof Calendar) { + d = ((Calendar)o).getTime(); + } else { + d = (Date)o; + } + + Integer styleI = (Integer)org.stringtemplate.v4.DateRenderer.formatToInt.get(formatString); + Object f; + if(styleI == null) { + f = new SimpleDateFormat(formatString, locale); + } else { + int style = styleI.intValue(); + if(formatString.startsWith("date:")) { + f = DateFormat.getDateInstance(style, locale); + } else if(formatString.startsWith("time:")) { + f = DateFormat.getTimeInstance(style, locale); + } else { + f = DateFormat.getDateTimeInstance(style, style, locale); + } + } + + return ((DateFormat)f).format(d); + } +} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/STUtils.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/STUtils.java new file mode 100644 index 0000000000000000000000000000000000000000..cd0cb61b9f40bc0a38a92d411daa633c35253073 --- /dev/null +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/STUtils.java @@ -0,0 +1,25 @@ +package org.duniter.elasticsearch.util.springtemplate; + +import org.stringtemplate.v4.DateRenderer; +import org.stringtemplate.v4.STGroup; +import org.stringtemplate.v4.STGroupDir; +import org.stringtemplate.v4.StringRenderer; + +import java.util.Date; + +public class STUtils { + + private STUtils() { + /*help class*/ + } + + public static STGroup newSTGroup(String dirName) { + // Configure springtemplate engine + STGroup templates = new STGroupDir(dirName, '$', '$'); + templates.registerRenderer(Date.class, new DateRenderer()); + templates.registerRenderer(String.class, new StringRenderer()); + return templates; + } + + +} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/StringRenderer.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/StringRenderer.java new file mode 100644 index 0000000000000000000000000000000000000000..01e3ac3e5ebfcbaf053a22c9efed87f7250854c7 --- /dev/null +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/StringRenderer.java @@ -0,0 +1,52 @@ +package org.duniter.elasticsearch.util.springtemplate; + +/*- + * #%L + * Duniter4j :: ElasticSearch Subscription 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 org.duniter.core.client.model.ModelUtils; +import org.duniter.core.util.CollectionUtils; +import org.nuiton.i18n.I18n; + +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-core/src/main/resources/i18n/duniter4j-es-core_en_GB.properties b/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_en_GB.properties index 4ada9bb98edcc894fcc032a9291ca10215d21b6d..9a1bb01daa4734bbaf2aff4f6a3c15a5847620e3 100644 --- a/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_en_GB.properties +++ b/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_en_GB.properties @@ -47,6 +47,7 @@ duniter4j.job.stopped= duniter4j.job.stopping= duniter4j.job.success= duniter4j.service.waitThenRetry=Error [%s]... will retry [%s/%s] +duniter4j.share.redirection.help=If you are not redirected automatically, follow this link\: duniter4j.task.issuer.system=System duniter4j.task.starting=Starting task... duniter4j.threadPool.clusterHealthStatus.changed=Cluster health status changed to [%s] diff --git a/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_fr_FR.properties b/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_fr_FR.properties index a2854149be7a599f50ed4b957d98b93c52e422dd..43a57823caeaca108c0a12c4a9d9f8cb79baa43a 100644 --- a/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_fr_FR.properties +++ b/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_fr_FR.properties @@ -49,6 +49,7 @@ duniter4j.job.stopped= duniter4j.job.stopping= duniter4j.job.success= duniter4j.service.waitThenRetry=Echec [%s]... tentative [%s/%s] +duniter4j.share.redirection.help=Si vous n'êtes pas redirigé automatiquement, cliquez sur le lien suivant \: duniter4j.task.issuer.system=Système duniter4j.task.starting=Démarrage du traitement... duniter4j.threadPool.clusterHealthStatus.changed=Cluster health status changed to [%s] diff --git a/duniter4j-es-subscription/src/main/resources/templates/cesium_logo.st b/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/cesium_logo.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/cesium_logo.st rename to duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/cesium_logo.st diff --git a/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/html_share.st b/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/html_share.st new file mode 100644 index 0000000000000000000000000000000000000000..664245e3511d95a8b6d5d74c8825c691e38b178d --- /dev/null +++ b/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/html_share.st @@ -0,0 +1,64 @@ +html_share(type, title, summary, description, image, siteName, locale, url, redirectMessage) ::= << +<html prefix="og: http://ogp.me/ns#"> + <head> + <meta charset="UTF-8"> + + $if(siteName)$ + <title>$siteName$ | $title$</title> + $else$ + <title>$title$</title> + $endif$ + + $if(type)$ + <meta property="og:type" content="$type$" /> + $else$ + <meta property="og:type" content="website" /> + $endif$ + + <meta property="og:title" content="$title$" /> + + $if(summary)$ + <meta property="og:description" content="$summary$" /> + $else$ + <meta property="og:description" content="$description$" /> + $endif$ + + $if(siteName)$ + <meta property="og:site_name" content="$siteName$" /> + $endif$ + + $if(image)$ + <meta property="og:image" content="$image$" /> + $endif$ + + $if(locale)$ + <meta property="og:locale" content="$locale$" /> + $endif$ + + $if(url)$ + <meta property="og:url" content="$url$" /> + <script type="text/javascript"> + window.location.href = "$url$" + </script> + <META HTTP-EQUIV="Refresh" CONTENT="0; URL=$url$"> + $endif$ + </head> + <body> + $if(image)$ + <p> + <img src="$image$"/> + </p> + $endif$ + + <h1>$title$</h1> + + <p>$description$</p> + + $if(url)$ + <p> + $redirectMessage$ <a href='$url$'>$title$</a>. + </p> + $endif$ + </body> +</html> +>> diff --git a/duniter4j-es-subscription/pom.xml b/duniter4j-es-subscription/pom.xml index c9beba4083d42dc7e961e6f0883ef92419a51fd9..c2ccb92692bfefbb8cd6eac45c731daf224d0934 100644 --- a/duniter4j-es-subscription/pom.xml +++ b/duniter4j-es-subscription/pom.xml @@ -46,7 +46,7 @@ <groupId>org.antlr</groupId> <artifactId>stringtemplate</artifactId> <version>${stringtemplate.version}</version> - <scope>compile</scope> + <scope>provided</scope> </dependency> <!-- Unit test --> 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 871c8863dfa34d6654879fae2f3508fa16b4dc30..5f93f052166cc1abf1767bf27afacf380c409d42 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 @@ -51,6 +51,7 @@ import org.duniter.elasticsearch.user.service.AdminService; import org.duniter.elasticsearch.user.service.MailService; import org.duniter.elasticsearch.user.service.UserEventService; import org.duniter.elasticsearch.user.service.UserService; +import org.duniter.elasticsearch.util.springtemplate.STUtils; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.unit.TimeValue; import org.nuiton.i18n.I18n; @@ -76,6 +77,7 @@ public class SubscriptionService extends AbstractService { private UserEventService userEventService; private UserService userService; private String emailSubjectPrefix; + private STGroup templates; @Inject public SubscriptionService(Duniter4jClient client, @@ -100,6 +102,11 @@ public class SubscriptionService extends AbstractService { if (StringUtils.isNotBlank(emailSubjectPrefix)) { emailSubjectPrefix += " "; // add one trailing space } + + // Configure springtemplate engine + templates = STUtils.newSTGroup("org/duniter/elasticsearch/subscription/templates"); + Preconditions.checkNotNull(templates.getInstanceOf("text_email"), "Missing ST template {text_email}"); + Preconditions.checkNotNull(templates.getInstanceOf("html_email_content"), "Missing ST template {html_email_content}"); } public String create(String json) { @@ -308,11 +315,7 @@ public class SubscriptionService extends AbstractService { 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()); - // 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( diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/cesium_logo.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/cesium_logo.st new file mode 100644 index 0000000000000000000000000000000000000000..83cf445c1c6d63f4dfb4612f81a06253aff3769b --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/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/css.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/css.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/css_logo.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css_logo.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/css_logo.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css_logo.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/event_item.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/event_item.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/event_item.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/event_item.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/html.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/html.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_email_content.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/html_email_content.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_email_content.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/html_event_item.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_event_item.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/html_event_item.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_event_item.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/i18n.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/i18n.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/i18n_args.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n_args.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/i18n_args.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n_args.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/text_email.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_email.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/text_email.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_email.st diff --git a/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_event_item.st similarity index 100% rename from duniter4j-es-subscription/src/main/resources/templates/text_event_item.st rename to duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_event_item.st diff --git a/duniter4j-es-subscription/src/main/resources/plugin-security.policy b/duniter4j-es-subscription/src/main/resources/plugin-security.policy index 3ea59400b9bd2a3d0e1b89091169fa50b60e1123..ea5437f6726c44547a1dce7470385fc69c055e22 100644 --- a/duniter4j-es-subscription/src/main/resources/plugin-security.policy +++ b/duniter4j-es-subscription/src/main/resources/plugin-security.policy @@ -1,4 +1,4 @@ -grant codeBase "file:${es.path.home}/plugins/duniter4j-es-gchange/"{ +grant codeBase "file:${es.path.home}/plugins/duniter4j-es-subscription/"{ permission java.io.FilePermission "/etc/ld.so.conf", "read"; permission java.io.FilePermission "/etc/ld.so.conf.d/*.conf", "read"; permission java.io.FilePermission "/usr/local/lib/*", "read"; diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java index 00601581814e13aeab27fcdb714404e86418a8f7..e640a389057facdda36a00115d2cbe8fbc794f7a 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java @@ -212,8 +212,17 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { return this.nodePubkey; } + public String getCesiumUrl() { + return this.settings.get("duniter.share.cesium.url", "https://g1.duniter.fr"); + } + public String getShareSiteName() { + return this.settings.get("duniter.user.share.site.name", "Cesium"); + } + public String getBaseUrl() { + return settings.get("duniter.share.base.url"); + } /* -- protected methods -- */ diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/Attachment.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/Attachment.java new file mode 100644 index 0000000000000000000000000000000000000000..3073c805f247a62e2eb6a0c1c8d2f62cda28178b --- /dev/null +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/Attachment.java @@ -0,0 +1,63 @@ +package org.duniter.elasticsearch.user.model; + +/* + * #%L + * Duniter4j :: Core Client API + * %% + * 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 com.fasterxml.jackson.annotation.JsonGetter; +import com.fasterxml.jackson.annotation.JsonSetter; +import org.duniter.core.client.model.elasticsearch.Record; + +/** + * Created by blavenie on 01/03/16. + */ +public class Attachment extends Record { + + public static final String JSON_PROPERTY_CONTENT_TYPE = "_content_type"; + public static final String JSON_PROPERTY_CONTENT = "_content"; + + public static final String PROPERTY_CONTENT_TYPE = "contentType"; + public static final String PROPERTY_CONTENT = "content"; + + private String contentType; + + private String content; + + @JsonSetter(JSON_PROPERTY_CONTENT_TYPE) + public void setContentType(String contentType) { + this.contentType = contentType; + } + + @JsonGetter(JSON_PROPERTY_CONTENT_TYPE) + public String getContentType() { + return contentType; + } + + @JsonGetter(JSON_PROPERTY_CONTENT) + public String getContent() { + return content; + } + + @JsonSetter(JSON_PROPERTY_CONTENT) + public void setContent(String content) { + this.content = content; + } +} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserProfile.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserProfile.java index 6ef6ecb0cc217f821fb3399e98370fbc4f8f8371..99c94b55a8ae38bad94d70b3ec0ea852565ec625 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserProfile.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserProfile.java @@ -43,6 +43,7 @@ public class UserProfile extends Record { private String locale; private String address; private String city; + private Attachment avatar; public String getTitle() { return title; @@ -91,4 +92,12 @@ public class UserProfile extends Record { public void setCity(String city) { this.city = city; } + + public Attachment getAvatar() { + return avatar; + } + + public void setAvatar(Attachment avatar) { + this.avatar = avatar; + } } 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 1422b77c0aac986165ae788fa60932df524ed010..fd91b23d0607055b152d9848d1e520c408abadba 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 @@ -50,6 +50,7 @@ public class RestModule extends AbstractModule implements Module { bind(RestUserEventMarkAsReadAction.class).asEagerSingleton(); bind(RestUserEventSearchAction.class).asEagerSingleton(); bind(RestUserAvatarAction.class).asEagerSingleton(); + bind(RestUserShareLinkAction.class).asEagerSingleton(); // Group bind(RestGroupIndexAction.class).asEagerSingleton(); @@ -76,6 +77,7 @@ public class RestModule extends AbstractModule implements Module { bind(RestPageCommentUpdateAction.class).asEagerSingleton(); bind(RestPageCategoryAction.class).asEagerSingleton(); bind(RestPageImageAction.class).asEagerSingleton(); + bind(RestPageShareLinkAction.class).asEagerSingleton(); // Mixed search bind(RestMixedSearchAction.class).asEagerSingleton(); diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageShareLinkAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageShareLinkAction.java new file mode 100644 index 0000000000000000000000000000000000000000..ecd1c817499ab22f60f370ff4ba7e231ba7da755 --- /dev/null +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageShareLinkAction.java @@ -0,0 +1,108 @@ +package org.duniter.elasticsearch.user.rest.page; + +import com.google.common.html.HtmlEscapers; +import org.duniter.core.exception.BusinessException; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.util.StringUtils; +import org.duniter.elasticsearch.exception.DuniterElasticsearchException; +import org.duniter.elasticsearch.rest.attachment.RestImageAttachmentAction; +import org.duniter.elasticsearch.rest.share.AbstractRestShareLinkAction; +import org.duniter.elasticsearch.user.PluginSettings; +import org.duniter.elasticsearch.user.dao.page.PageIndexDao; +import org.duniter.elasticsearch.user.dao.page.PageRecordDao; +import org.duniter.elasticsearch.user.model.UserProfile; +import org.duniter.elasticsearch.user.model.page.RegistryRecord; +import org.duniter.elasticsearch.user.service.PageService; +import org.duniter.elasticsearch.user.service.UserService; +import org.duniter.elasticsearch.util.opengraph.OGData; +import org.elasticsearch.client.Client; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.settings.Settings; +import org.elasticsearch.rest.RestController; +import org.nuiton.i18n.I18n; + +import java.io.UnsupportedEncodingException; +import java.net.URLEncoder; +import java.util.Locale; + +public class RestPageShareLinkAction extends AbstractRestShareLinkAction { + + @Inject + public RestPageShareLinkAction(final Settings settings, final RestController controller, final Client client, + final PluginSettings pluginSettings, + final PageService service) { + super(settings, controller, client, PageIndexDao.INDEX, PageRecordDao.TYPE, + createResolver(pluginSettings, service)); + } + + protected static OGDataResolver createResolver( + final PluginSettings pluginSettings, + final PageService service) throws DuniterElasticsearchException, BusinessException { + + return (id) -> { + try { + RegistryRecord record = service.getPageForSharing(id); + + OGData data = new OGData(); + + if (record != null) { + + // og:title + if (StringUtils.isNotBlank(record.getTitle())) { + data.title = record.getTitle(); + } + else { + data.title = pluginSettings.getShareSiteName(); + } + + // og:description + data.description = HtmlEscapers.htmlEscaper().escape(record.getDescription()); + + // og:image + if (record.getThumbnail() != null && StringUtils.isNotBlank(record.getThumbnail().get("_content_type"))) { + String baseUrl = pluginSettings.getBaseUrl(); + data.image = StringUtils.isBlank(baseUrl) ? "" : baseUrl; + data.image += RestImageAttachmentAction.computeImageUrl(PageIndexDao.INDEX, PageRecordDao.TYPE, id, RegistryRecord.PROPERTY_THUMBNAIL, record.getThumbnail().get("_content_type")); + } + + // og:url + data.url = String.format("%s/#/app/page/view/%s/%s", + pluginSettings.getCesiumUrl(), + id, + URLEncoder.encode(record.getTitle(), "UTF-8")); + } + else { + + // og:title + data.title = pluginSettings.getShareSiteName(); + + // og:description + data.description = I18n.t("duniter.user.share.description"); + + // og:url + data.url = String.format("%s/#/app/page/view/%s/%s", + pluginSettings.getCesiumUrl(), + id, + ""); + } + + // og:type + data.type = "website"; + + // og:site_name + data.siteName = pluginSettings.getShareSiteName(); + + // default og:image + if (StringUtils.isBlank(data.image)) { + data.image = pluginSettings.getCesiumUrl() + "/img/logo_128px.png"; + data.imageType = "image/png"; + } + + return data; + } + catch(UnsupportedEncodingException e) { + throw new TechnicalException(e); + } + }; + } +} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserShareLinkAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserShareLinkAction.java new file mode 100644 index 0000000000000000000000000000000000000000..540e6e4e0fcdf07c324b0b6a248d078763e7b2c2 --- /dev/null +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserShareLinkAction.java @@ -0,0 +1,130 @@ +package org.duniter.elasticsearch.user.rest.user; + +import com.google.common.collect.Maps; +import com.google.common.html.HtmlEscapers; +import org.duniter.core.exception.BusinessException; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.util.StringUtils; +import org.duniter.elasticsearch.exception.DuniterElasticsearchException; +import org.duniter.elasticsearch.rest.attachment.RestImageAttachmentAction; +import org.duniter.elasticsearch.rest.share.AbstractRestShareLinkAction; +import org.duniter.elasticsearch.user.PluginSettings; +import org.duniter.elasticsearch.user.model.UserProfile; +import org.duniter.elasticsearch.user.service.UserService; +import org.duniter.elasticsearch.util.opengraph.OGData; +import org.elasticsearch.client.Client; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.settings.Settings; +import org.elasticsearch.rest.RestController; +import org.nuiton.i18n.I18n; + +import java.io.UnsupportedEncodingException; +import java.net.URLEncoder; +import java.util.Locale; + +public class RestUserShareLinkAction extends AbstractRestShareLinkAction { + + @Inject + public RestUserShareLinkAction(final Settings settings, final RestController controller, final Client client, + final PluginSettings pluginSettings, + final UserService userService) { + super(settings, controller, client, UserService.INDEX, UserService.PROFILE_TYPE, + createResolver(pluginSettings, userService)); + + if (StringUtils.isBlank(pluginSettings.getBaseUrl())) { + log.warn(I18n.t("duniter4j.es.share.error.noBaseUrl", "duniter.share.base.url")); + } + } + + protected static AbstractRestShareLinkAction.OGDataResolver createResolver( + final PluginSettings pluginSettings, + final UserService userService) throws DuniterElasticsearchException, BusinessException { + + return (id) -> { + try { + UserProfile profile = userService.getUserProfileForSharing(id); + + OGData data = new OGData(); + + if (profile != null) { + + // og:locale + Locale locale; + if (StringUtils.isNotBlank(profile.getLocale())) { + locale = new Locale(profile.getLocale()); + data.locale = profile.getLocale(); + } + else { + locale = I18n.getDefaultLocale(); + } + + String pubkey = I18n.l(locale, "duniter.user.share.pubkey", id); + + // og:title + if (StringUtils.isNotBlank(profile.getTitle())) { + data.title = profile.getTitle(); + data.description = pubkey; + } + else { + data.title = pubkey; + data.description = ""; + } + + // og:description + if (StringUtils.isNotBlank(data.description)) data.description += " | "; + if (StringUtils.isNotBlank(profile.getDescription())) { + data.description += HtmlEscapers.htmlEscaper().escape(profile.getDescription()); + } + else { + data.description += I18n.l(locale, "duniter.user.share.description"); + } + + // og:image + if (profile.getAvatar() != null && StringUtils.isNotBlank(profile.getAvatar().getContentType())) { + String baseUrl = pluginSettings.getBaseUrl(); + data.image = StringUtils.isBlank(baseUrl) ? "" : baseUrl; + data.image += RestImageAttachmentAction.computeImageUrl(UserService.INDEX, UserService.PROFILE_TYPE, id, UserProfile.PROPERTY_AVATAR, profile.getAvatar().getContentType()); + } + + // og:url + data.url = String.format("%s/#/app/wot/%s/%s", + pluginSettings.getCesiumUrl(), + id, + URLEncoder.encode(profile.getTitle(), "UTF-8")); + } + else { + + // og:title + String pubkey = I18n.t("duniter.user.share.pubkey", id); + data.title = pubkey; + + // og:description + data.description = I18n.t("duniter.user.share.description"); + + // og:url + data.url = String.format("%s/#/app/wot/%s/%s", + pluginSettings.getCesiumUrl(), + id, + ""); + } + + // og:type + data.type = "website"; + + // og:site_name + data.siteName = pluginSettings.getShareSiteName(); + + // default og:image + if (StringUtils.isBlank(data.image)) { + data.image = pluginSettings.getCesiumUrl() + "/img/logo_128px.png"; + data.imageType = "image/png"; + } + + return data; + } + catch(UnsupportedEncodingException e) { + throw new TechnicalException(e); + } + }; + } +} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/PageService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/PageService.java index 881fd85ef293b7d62b20afade1c87d375c4d5cd0..f4b4429e5af3cca16913fdff7eaf0fc9742d8c80 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/PageService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/PageService.java @@ -32,6 +32,8 @@ import org.duniter.elasticsearch.user.PluginSettings; import org.duniter.elasticsearch.user.dao.page.PageCommentDao; import org.duniter.elasticsearch.user.dao.page.PageIndexDao; import org.duniter.elasticsearch.user.dao.page.PageRecordDao; +import org.duniter.elasticsearch.user.model.UserProfile; +import org.duniter.elasticsearch.user.model.page.RegistryRecord; import org.elasticsearch.common.inject.Inject; /** @@ -138,6 +140,13 @@ public class PageService extends AbstractService { commentDao.update(id, json); } + public RegistryRecord getPageForSharing(String id) { + + return client.getSourceByIdOrNull(recordDao.getIndex(), recordDao.getType(), id, RegistryRecord.class, + RegistryRecord.PROPERTY_TITLE, + RegistryRecord.PROPERTY_DESCRIPTION, + RegistryRecord.PROPERTY_THUMBNAIL); + } /* -- Internal methods -- */ diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java index 745f5b828e6eb0bfe35efa97b875556329b5c4d9..2d080a88ba9d1309ec607cb8aa9a5345c5103003 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java @@ -24,19 +24,26 @@ package org.duniter.elasticsearch.user.service; import com.fasterxml.jackson.databind.JsonNode; +import org.apache.lucene.queryparser.flexible.core.util.StringUtils; import org.duniter.core.util.Preconditions; import org.apache.commons.collections4.MapUtils; import org.duniter.core.client.model.ModelUtils; -import org.duniter.core.client.model.elasticsearch.UserProfile; +import org.duniter.elasticsearch.user.model.Attachment; +import org.duniter.elasticsearch.user.model.UserProfile; import org.duniter.core.service.CryptoService; import org.duniter.elasticsearch.client.Duniter4jClient; +import org.duniter.elasticsearch.rest.attachment.RestImageAttachmentAction; +import org.duniter.elasticsearch.rest.share.AbstractRestShareLinkAction; import org.duniter.elasticsearch.user.PluginSettings; import org.duniter.elasticsearch.exception.AccessDeniedException; import org.duniter.elasticsearch.service.AbstractService; import org.duniter.elasticsearch.user.dao.profile.UserIndexDao; import org.duniter.elasticsearch.user.dao.profile.UserProfileDao; import org.duniter.elasticsearch.user.dao.profile.UserSettingsDao; +import org.duniter.elasticsearch.util.opengraph.OGData; import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.rest.BytesRestResponse; +import org.elasticsearch.rest.RestStatus; import java.util.HashMap; import java.util.Map; @@ -226,6 +233,15 @@ public class UserService extends AbstractService { return sb.substring(separator.length()); } + public UserProfile getUserProfileForSharing(String id) { + + return client.getSourceByIdOrNull(INDEX, PROFILE_TYPE, id, UserProfile.class, + UserProfile.PROPERTY_TITLE, + UserProfile.PROPERTY_DESCRIPTION, + UserProfile.PROPERTY_LOCALE, + UserProfile.PROPERTY_AVATAR); + } + /* -- Internal methods -- */ } 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 98d218d1ac2674b2a99cc8e0d673c1eb85d6576e..3c47a593117794059ed2628a702289ef4d3798b0 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 @@ -15,4 +15,7 @@ duniter.user.event.NODE_BMA_UP=Duniter node [%1$s\:%2$s] is UP again. duniter.user.event.NODE_STARTED=Your node ES API [%1$s] is UP. duniter.user.event.TX_RECEIVED=You received a payment from %2$s. duniter.user.event.TX_SENT=Your payment to %2$s was executed. +duniter.user.share.description=Follow your "libre money" wallets easily +duniter.user.share.pubkey=Public key\: %1$s +duniter4j.es.share.error.noBaseUrl=Image path are relative for share links (og\:image). /\!\\ Set [%s] in the configuration (recommended). duniter4j.ws.user.open=User [%1$s] connecting with id [%2$s] with locale [%3$s] 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 e4e31362232b4af585877a5b90a3541bf51c8da1..ec9ace345c37859ec4eebeece40f906bd7d5233d 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 @@ -15,4 +15,7 @@ duniter.user.event.NODE_BMA_UP=Noeud Duniter [%1$s\:%2$s] à nouveau accessible. duniter.user.event.NODE_STARTED=Noeud ES API [%1$s] est démarré. duniter.user.event.TX_RECEIVED=Vous avez recu un paiement de %2$s. duniter.user.event.TX_SENT=Votre paiement à %2$s a bien été executé. +duniter.user.share.description=Suivez vos comptes de monnaie libre en toute simplicité \! +duniter.user.share.pubkey=Clé publique \: %1$s +duniter4j.es.share.error.noBaseUrl=Chemins relatif pour les images, dans les partages (og\:image). /\!\\ Renseignez [%s] dans la configuration (conseillé). duniter4j.ws.user.open=Utilisateur [%1$s] connecté id\=[%2$s] sur la locale [%3$s] diff --git a/pom.xml b/pom.xml index ea42a4859c7081124d9c4a627b95c62efbdd3942..a58713254a8ae40c7e5eecc5a9478632480fd1dd 100644 --- a/pom.xml +++ b/pom.xml @@ -33,8 +33,8 @@ <slf4j.version>1.7.5</slf4j.version> <guava.version>22.0</guava.version> <xml-apis.version>2.0.2</xml-apis.version> - <kalium.version>0.5.0_blavenie</kalium.version> - <jnr-ffi.version>2.1.0</jnr-ffi.version> + <kalium.version>0.6.0_PR64</kalium.version> + <jnr-ffi.version>2.1.7</jnr-ffi.version> <scrypt.version>1.4.0</scrypt.version> <elasticsearch.version>2.4.6</elasticsearch.version> <jna.version>4.2.0</jna.version> diff --git a/src/scripts/start-es-node.sh b/src/scripts/start-es-node.sh new file mode 100755 index 0000000000000000000000000000000000000000..a9be23c2a5700cafb7bcfa08b37ab480d64a0ea4 --- /dev/null +++ b/src/scripts/start-es-node.sh @@ -0,0 +1,16 @@ +#!/bin/bash + + +PID=`ps -efl | grep duniter4j-es | grep g1/lib | awk '/^([0-9]+) ([^\s]) ([a-zA-Z0-9]+) ([0-9]+).*/ {printf "%s", $4}'` + +if [ "$PID" != "" ]; +then + echo "Error: ES node already started!" + exit -1 +else + cd /opt/duniter4j-es-g1/bin + ./elasticsearch -d + echo "ES node started !" + echo "...to follow log: tail -f /opt/duniter4j-es-g1/logs/g1-es-data.log" +fi + diff --git a/src/scripts/stop-es-nodes.sh b/src/scripts/stop-es-nodes.sh new file mode 100755 index 0000000000000000000000000000000000000000..8181a733c99af4acddd1e759dcb3b3effc6cfcca --- /dev/null +++ b/src/scripts/stop-es-nodes.sh @@ -0,0 +1,32 @@ +#!/bin/bash + +# Get ES PID +PID=`ps -efl | grep duniter4j-es | grep lib | awk '{printf "%s", $4}'` + +if [ "$PID" != "" ]; +then + echo "Stopping ES node running on PID $PID..." + sudo kill -15 $PID + + sleep 5s + + # Check if still alive + PID=`ps -efl | grep duniter4j-es | grep g1/lib | awk '{printf "%s", $4}'` + if [ "$PID" != "" ]; + then + sleep 10s + fi + + PID=`ps -efl | grep duniter4j-es | grep g1/lib | awk '{printf "%s", $4}'` + if [ "$PID" != "" ]; + then + echo "Error: Unable to stop ES node !" + exit -1 + else + echo "ES node stopped" + fi + +else + echo "ES node not running!" +fi + diff --git a/src/scripts/update-es-node.sh b/src/scripts/update-es-node.sh new file mode 100755 index 0000000000000000000000000000000000000000..28702aadeaff634210f4a1ecbd87ac75b1b60860 --- /dev/null +++ b/src/scripts/update-es-node.sh @@ -0,0 +1,90 @@ +#!/bin/bash + +VERSION=$1 +OLD_VERSION=$2 + +if [ "${VERSION}" == "" ]; then + echo "ERROR: Missing version argument !" + echo " " + echo "usage: sudo ./update-es.sh <version> [<old_version>]" + exit +fi +if [ "${OLD_VERSION}" == "" ]; then + OLD_VERSION=`ps -efl | grep duniter4j-es | grep g1/lib | sed -r 's/.*duniter4j-es-([0-9.]+)-g1.*/\1/g'` + if [ "${OLD_VERSION}" == "" ]; then + echo "Error: unable to known previous version" + exit + fi +fi + +READLINK=`which readlink` +if [ -z "$READLINK" ]; then + message "Required tool 'readlink' is missing. Please install before launch \"$0\" file." + exit 1 +fi + +# ------------------------------------------------------------------ +# Ensure BASEDIR points to the directory where the soft is installed. +# ------------------------------------------------------------------ +SCRIPT_LOCATION=$0 +if [ -x "$READLINK" ]; then + while [ -L "$SCRIPT_LOCATION" ]; do + SCRIPT_LOCATION=`"$READLINK" -e "$SCRIPT_LOCATION"` + done +fi + +export BASEDIR=`dirname "$SCRIPT_LOCATION"` +cd $BASEDIR + +echo "--- Downloading duniter4j-es-standalone v$VERSION... ----------------------" + +if [ -f "downloads/duniter4j-es-${VERSION}-standalone.zip" ]; then + echo "...removing file, as it already exists in ./downloads/duniter4j-es-${VERSION}-standalone.zip" + rm ./downloads/duniter4j-es-${VERSION}-standalone.zip +fi + +if [ ! -e "downloads" ]; then + mkdir downloads +fi + +cd downloads +wget -kL https://github.com/duniter/duniter4j/releases/download/duniter4j-${VERSION}/duniter4j-es-${VERSION}-standalone.zip +cd .. + +if [ -f "downloads/duniter4j-es-${VERSION}-standalone.zip" ]; then + echo "" +else + echo "Error: unable to dowlonad this version!" + exit -1 +fi + +echo "--- Installating duniter4j-es v$VERSION... ---------------------" +if [ -d "/opt/duniter4j-es-${VERSION}-g1" ]; then + echo "Error: Already installed in /opt/duniter4j-es-${VERSION}-g1 !" + exit -1 +fi + +unzip -o ./downloads/duniter4j-es-${VERSION}-standalone.zip +mv duniter4j-es-${VERSION} duniter4j-es-${VERSION}-g1 +sudo mv duniter4j-es-${VERSION}-g1 /opt/ +sudo rm /opt/duniter4j-es-g1 +sudo ln -s /opt/duniter4j-es-${VERSION}-g1 /opt/duniter4j-es-g1 + +mkdir /opt/duniter4j-es-${VERSION}-g1/data +mv /opt/duniter4j-es-${VERSION}-g1/config/elasticsearch.yml /opt/duniter4j-es-${VERSION}-g1/config/elasticsearch.yml.ori + +./stop-es-nodes.sh + +if [ "$OLD_VERSION" != "$VERSION" ]; +then + echo "--- Restoring files (data+config) from previous version $OLD_VERSION... ---------------------" + tar -cvf /opt/duniter4j-es-${OLD_VERSION}-g1/data/save.tar.gz /opt/duniter4j-es-${OLD_VERSION}-g1/data/g1-* + mv /opt/duniter4j-es-${OLD_VERSION}-g1/data/g1-* /opt/duniter4j-es-${VERSION}-g1/data + cp /opt/duniter4j-es-${OLD_VERSION}-g1/config/elasticsearch.yml /opt/duniter4j-es-${VERSION}-g1/config +fi + +#./start-es-nodes.sh + +echo "--- Successfully installed duniter4j-es v$VERSION ! -------------" +echo "" +