diff --git a/README.md b/README.md index 3cfba466191afb315b2e18483a0bfa628425f2dd..d8e25e2121bfafcc1fe510543b3f16987388b78a 100644 --- a/README.md +++ b/README.md @@ -15,10 +15,4 @@ Duniter4j is a Java Toolkit for [Duniter](http://duniter.org). - `duniter4j-core-client`: [a Java API](./src/site/markdown/Java_API.md) to help Java developers to communicate with a Duniter network. - - `duniter4j-elasticsearch`: [a ElastiSearch node](./src/site/markdown/ES.md) used to store (with full-text capabilities) all blockchain data, and additional user data. - - * It comes with an [HTTP API](./src/site/markdown/ES_API.md) to store and retrieve all this data. - - * This API is used by [Cesium+](https://www.github.com/duniter/cesium) (a Duniter wallet). - diff --git a/duniter4j-client/src/license/THIRD-PARTY.properties b/duniter4j-client/src/license/THIRD-PARTY.properties index 290ad644adf0b6fdfd6aed29e52ca005550953b6..1ce5233532388fe08b2e42f64ea060867702feaa 100644 --- a/duniter4j-client/src/license/THIRD-PARTY.properties +++ b/duniter4j-client/src/license/THIRD-PARTY.properties @@ -21,6 +21,7 @@ # Please fill the missing licenses for dependencies : # # -#Tue Nov 21 18:04:39 CET 2017 +#Wed Jun 20 16:56:38 CEST 2018 +commons-primitives--commons-primitives--1.0=The Apache Software License, Version 2.0 dnl.utils--j-text-utils--0.3.3=GNU General Lesser Public License (LGPL) version 3.0 net.sf.opencsv--opencsv--2.3=GNU General Lesser Public License (LGPL) version 3.0 diff --git a/duniter4j-client/src/main/assembly/min/duniter4j-client.config b/duniter4j-client/src/main/assembly/min/duniter4j-client.config index 1b8b35f3c1331dca1badbe9277af2a0e80c325b2..78b06ab84837b87ef20101c079d530b589bc368e 100644 --- a/duniter4j-client/src/main/assembly/min/duniter4j-client.config +++ b/duniter4j-client/src/main/assembly/min/duniter4j-client.config @@ -1 +1,15 @@ -# Duniter4j Configuration +# -------------------------------------- +# Duniter4j Client Configuration file +# -------------------------------------- + +# +# Duniter node +# +duniter4j.node.host=g1.duniter.org +duniter4j.node.port=10901 + +# +# Cesium+ node (aka Duniter4j-Elasticsearch) +# +duniter4j.node.elasticsearch.host=g1.data.duniter.fr +duniter4j.node.elasticsearch.port=443 diff --git a/duniter4j-client/src/main/filtered-resources/duniter4j-client.config b/duniter4j-client/src/main/filtered-resources/duniter4j-client.config index 9ad61c03358ef8203a174530c2c17c1560cf18e2..2e90086dd45ee4f525394530a558d29038b11181 100644 --- a/duniter4j-client/src/main/filtered-resources/duniter4j-client.config +++ b/duniter4j-client/src/main/filtered-resources/duniter4j-client.config @@ -3,7 +3,9 @@ duniter4j.version=${project.version} duniter4j.inceptionYear=${project.inceptionYear} duniter4j.organizationName=${license.organizationName} - +# +# Duniter node +# duniter4j.node.host=192.168.0.5 duniter4j.node.port=10901 diff --git a/duniter4j-client/src/main/filtered-resources/log4j.properties b/duniter4j-client/src/main/filtered-resources/log4j.properties index 135fd9c4a4d3509072c55b024bace0abee7421fa..ad5e6c445328789634532fb9fbdeb5111cc9fa29 100644 --- a/duniter4j-client/src/main/filtered-resources/log4j.properties +++ b/duniter4j-client/src/main/filtered-resources/log4j.properties @@ -19,8 +19,8 @@ log4j.appender.file.layout.ConversionPattern=%d{ISO8601} %5p %c - %m%n # Duniter4j levels log4j.logger.org.duniter=INFO log4j.logger.org.duniter.core=WARN -# Avoid warning on leaf not found (Duniter issue) -log4j.logger.org.duniter.core.client.service.local.NetworkServiceImpl=ERROR +# Avoid warning on leaf not found (Duniter issue ?) +#log4j.logger.org.duniter.core.client.service.local.NetworkServiceImpl=WARN # Other frameworks levels log4j.logger.org.apache.http=ERROR diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/config/Configuration.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/config/Configuration.java index 43530e98b25a02a26875eb08cc61528c83697951..7bb45477b84d9f900371713799579b01b6a663d4 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/config/Configuration.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/config/Configuration.java @@ -35,6 +35,7 @@ import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.io.File; +import java.net.MalformedURLException; import java.net.URL; import java.util.Locale; import java.util.Set; @@ -274,6 +275,17 @@ public class Configuration { } public URL getNodeElasticSearchUrl() { - return applicationConfig.getOptionAsURL(ConfigurationOption.NODE_ELASTICSEARCH_URL.getKey()); + // Force SSL for 443 port + if (getNodeElasticSearchPort() == 443) { + try { + return new URL(applicationConfig.getOption(ConfigurationOption.NODE_ELASTICSEARCH_URL.getKey()) + .replaceAll("http://", "https://")); + } catch(MalformedURLException e) { + return applicationConfig.getOptionAsURL(ConfigurationOption.NODE_ELASTICSEARCH_URL.getKey()); + } + } + else { + return applicationConfig.getOptionAsURL(ConfigurationOption.NODE_ELASTICSEARCH_URL.getKey()); + } } } diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/NetworkPeers.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/NetworkPeers.java index c413492d9586f1980fd14cfc5b3d91e6d222c406..c7e4455c63b86aa7f638081fdbecdcb31d6bbca0 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/NetworkPeers.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/NetworkPeers.java @@ -47,6 +47,7 @@ public class NetworkPeers implements Serializable { public String version; public String currency; public String status; + public Long statusTS; public String block; public String signature; public String pubkey; @@ -79,6 +80,16 @@ public class NetworkPeers implements Serializable { this.status = status; } + @JsonGetter("statusTS") + public Long getStatusTS() { + return statusTS; + } + + @JsonSetter("statusTS") + public void setStatusTS(Long statusTS) { + this.statusTS = statusTS; + } + public String getBlock() { return block; } diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java index 31821fda20bf8c8f5f47882450c04daa0902b4e1..7c4d3addcc564d19bb087d998ea3b81b98d7001e 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java @@ -22,6 +22,8 @@ package org.duniter.core.client.model.bma.jackson; * #L% */ +import com.fasterxml.jackson.core.JsonGenerator; +import com.fasterxml.jackson.databind.DeserializationFeature; import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.module.SimpleModule; import org.duniter.core.client.model.bma.BlockchainBlock; @@ -63,7 +65,8 @@ public abstract class JacksonUtils extends SimpleModule { objectMapper.registerModule(module); // Adding features - //objectMapper.getFactory().configure(JsonGenerator.Feature., true); + objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); + //objectMapper.getFactory().configure(JsonGenerator.Feature.IGNORE_UNKNOWN, true); return objectMapper; } diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Peer.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Peer.java index d2e2c649cd54b577a3346a96ad3c02d6d7b53f2e..9ed441bc4d4429e0ba97500be56778e5deec5498 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Peer.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Peer.java @@ -149,9 +149,9 @@ public class Peer implements LocalEntity<String>, Serializable { public Peer build() { int port = this.port != null ? this.port : 80; - boolean useSsl = this.useSsl != null ? this.useSsl : - (port == 443 || this.api == EndpointApi.BMAS.name()); String api = this.api != null ? this.api : EndpointApi.BASIC_MERKLED_API.name(); + boolean useSsl = this.useSsl != null ? this.useSsl : + (port == 443 || EndpointApi.BMAS.name().equals(this.api)); Peer ep = new Peer(api, dns, ipv4, ipv6, port, useSsl); if (StringUtils.isNotBlank(this.epId)) { ep.setEpId(this.epId); diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/HttpServiceImpl.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/HttpServiceImpl.java index 4fbdddbd58208035833d0144fa64b82751e29327..d3e55f9520fa2885516f8f088b22ae955b019489 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/HttpServiceImpl.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/HttpServiceImpl.java @@ -51,6 +51,7 @@ import org.duniter.core.client.model.bma.jackson.JacksonUtils; import org.duniter.core.client.model.local.Peer; import org.duniter.core.client.service.bma.BmaTechnicalException; import org.duniter.core.client.service.exception.*; +import org.duniter.core.exception.BusinessException; import org.duniter.core.exception.TechnicalException; import org.duniter.core.util.ObjectUtils; import org.duniter.core.util.StringUtils; @@ -349,7 +350,7 @@ public class HttpServiceImpl implements HttpService, Closeable, InitializingBean catch (SocketTimeoutException | ConnectTimeoutException e) { throw new HttpTimeoutException(I18n.t("duniter4j.client.core.timeout"), e); } - catch (TechnicalException e) { + catch (TechnicalException | BusinessException e) { throw e; } catch (Throwable e) { diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java index 34fef4cd0636e645690c06d5015307911364b84d..66c0d61d709d5b4473f9cf8e43db3ad81a96970b 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java @@ -272,7 +272,7 @@ public class NetworkServiceImpl extends BaseRemoteServiceImpl implements Network Filter filterDef = new Filter(); filterDef.filterType = null; filterDef.filterStatus = Peer.PeerStatus.UP; - filterDef.filterEndpoints = ImmutableList.of(EndpointApi.BASIC_MERKLED_API.name(), EndpointApi.BMAS.name()); + filterDef.filterEndpoints = ImmutableList.of(EndpointApi.BASIC_MERKLED_API.name(), EndpointApi.BMAS.name(), EndpointApi.WS2P.name()); filterDef.currency = parameters.getCurrency(); // Default sort @@ -451,7 +451,7 @@ public class NetworkServiceImpl extends BaseRemoteServiceImpl implements Network List<Peer> result = new ArrayList<>(); // If less than 100 node, get it in ONE call - if (leaves.size() < 100) { + if (leaves.size() <= 2000) { List<Peer> peers = networkRemoteService.getPeers(peer); if (CollectionUtils.isNotEmpty(peers)) { @@ -496,9 +496,12 @@ public class NetworkServiceImpl extends BaseRemoteServiceImpl implements Network NetworkPeers.Peer peer = networkRemoteService.getPeerLeaf(requestedPeer, leaf); addEndpointsAsPeers(peer, result, leaf, filterEndpoints); - } catch(HttpNotFoundException | TechnicalException e) { + } catch(HttpNotFoundException hnfe) { log.debug("Peer not found for leaf=" + leaf); // skip + } catch(TechnicalException e) { + log.warn("Error while getting peer leaf=" + leaf, e.getMessage()); + // skip } } } diff --git a/duniter4j-es-assembly/LICENSE.txt b/duniter4j-es-assembly/LICENSE.txt deleted file mode 100644 index 94a9ed024d3859793618152ea559a168bbcbb5e2..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/LICENSE.txt +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - <one line to give the program's name and a brief idea of what it does.> - Copyright (C) <year> <name of author> - - 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/>. - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - <program> Copyright (C) <year> <name of author> - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -<http://www.gnu.org/licenses/>. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -<http://www.gnu.org/philosophy/why-not-lgpl.html>. diff --git a/duniter4j-es-assembly/pom.xml b/duniter4j-es-assembly/pom.xml deleted file mode 100644 index 0d51c4b9498d406e37aab0476b1a1ca16c2c0560..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/pom.xml +++ /dev/null @@ -1,388 +0,0 @@ -<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> - <modelVersion>4.0.0</modelVersion> - - <parent> - <groupId>org.duniter</groupId> - <artifactId>duniter4j</artifactId> - <version>1.0.4-SNAPSHOT</version> - </parent> - - <artifactId>duniter4j-es-assembly</artifactId> - <packaging>pom</packaging> - <name>Duniter4j :: ElasticSearch Assembly</name> - <description>Build a ElasticSearch releases with all Duniter4j plugins</description> - - <properties> - <!-- bundle configuration --> - <bundlePrefix>duniter4j-es-${project.version}</bundlePrefix> - - <!-- i18n configuration --> - <i18n.bundleOutputName>duniter4j-es-i18n</i18n.bundleOutputName> - <i18n.generateCsvFile>true</i18n.generateCsvFile> - <i18n.bundleCsvFile> - ${maven.gen.dir}/resources/META-INF/${i18n.bundleOutputName}.csv - </i18n.bundleCsvFile> - <config.i18nBundleName>${i18n.bundleOutputName}</config.i18nBundleName> - <assembly.skip>false</assembly.skip> - </properties> - - <dependencies> - <dependency> - <groupId>javax.websocket</groupId> - <artifactId>javax.websocket-api</artifactId> - <version>1.1</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-client</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-server</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-client</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-server</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - <version>${jna.version}</version> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna-platform</artifactId> - <version>${jna.version}</version> - <exclusions> - <exclusion> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - </exclusion> - </exclusions> - </dependency> - </dependencies> - - <build> - - <plugins> - <plugin> - <artifactId>maven-dependency-plugin</artifactId> - <executions> - <!-- unpack ES --> - <execution> - <id>unpack-elasticsearch</id> - <goals> - <goal>unpack</goal> - </goals> - <phase>prepare-package</phase> - <configuration> - <artifactItems> - <artifactItem> - <groupId>org.elasticsearch.distribution.zip</groupId> - <artifactId>elasticsearch</artifactId> - <version>${elasticsearch.version}</version> - <type>zip</type> - </artifactItem> - </artifactItems> - <outputDirectory>${project.build.directory}/</outputDirectory> - <silent>true</silent> - <skip>${assembly.skip}</skip> - </configuration> - </execution> - - <!-- unpack attachment plugin --> - <execution> - <id>unpack-mapper-attachments-plugin</id> - <goals> - <goal>unpack</goal> - </goals> - <phase>prepare-package</phase> - <configuration> - <artifactItems> - <artifactItem> - <groupId>org.elasticsearch.plugin</groupId> - <artifactId>mapper-attachments</artifactId> - <version>${elasticsearch.version}</version> - <type>zip</type> - </artifactItem> - </artifactItems> - <outputDirectory>${project.build.directory}/elasticsearch-${elasticsearch.version}/plugins/mapper-attachments</outputDirectory> - <silent>true</silent> - <skip>${assembly.skip}</skip> - </configuration> - </execution> - - <!-- unpack ES core plugin --> - <execution> - <id>unpack-es-core-plugin</id> - <goals> - <goal>unpack</goal> - </goals> - <phase>prepare-package</phase> - <configuration> - <artifactItems> - <artifactItem> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-core</artifactId> - <version>${project.version}</version> - <type>zip</type> - </artifactItem> - </artifactItems> - <outputDirectory>${project.build.directory}/elasticsearch-${elasticsearch.version}/plugins/duniter4j-es-core</outputDirectory> - <silent>true</silent> - <skip>${assembly.skip}</skip> - </configuration> - </execution> - - <!-- unpack ES user plugin --> - <execution> - <id>unpack-es-user-plugin</id> - <goals> - <goal>unpack</goal> - </goals> - <phase>prepare-package</phase> - <configuration> - <artifactItems> - <artifactItem> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-user</artifactId> - <version>${project.version}</version> - <type>zip</type> - </artifactItem> - </artifactItems> - <outputDirectory>${project.build.directory}/elasticsearch-${elasticsearch.version}/plugins/duniter4j-es-user</outputDirectory> - <silent>true</silent> - <skip>${assembly.skip}</skip> - </configuration> - </execution> - - <!-- unpack ES subscription plugin --> - <execution> - <id>unpack-es-subscription-plugin</id> - <goals> - <goal>unpack</goal> - </goals> - <phase>prepare-package</phase> - <configuration> - <artifactItems> - <artifactItem> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-subscription</artifactId> - <version>${project.version}</version> - <type>zip</type> - </artifactItem> - </artifactItems> - <outputDirectory>${project.build.directory}/elasticsearch-${elasticsearch.version}/plugins/duniter4j-es-subscription</outputDirectory> - <silent>true</silent> - <skip>${assembly.skip}</skip> - </configuration> - </execution> - </executions> - </plugin> - - <plugin> - <artifactId>maven-assembly-plugin</artifactId> - <executions> - <execution> - <id>assembly-standalone</id> - <phase>package</phase> - <goals> - <goal>single</goal> - </goals> - <configuration> - <attach>true</attach> - <finalName>${bundlePrefix}</finalName> - <descriptors> - <descriptor> - ${basedir}/src/main/assembly/standalone.xml - </descriptor> - </descriptors> - <skipAssembly>${assembly.skip}</skipAssembly> - </configuration> - </execution> - </executions> - </plugin> - </plugins> - </build> - - <profiles> - <!-- use this profile to run the main class --> - <profile> - <id>run</id> - <activation> - <activeByDefault>false</activeByDefault> - </activation> - <build> - <defaultGoal>integration-test</defaultGoal> - <plugins> - <plugin> - <groupId>org.apache.maven.plugins</groupId> - <artifactId>maven-antrun-plugin</artifactId> - <dependencies> - <dependency> - <groupId>ant-contrib</groupId> - <artifactId>ant-contrib</artifactId> - <version>1.0b3</version> - <exclusions> - <exclusion> - <groupId>ant</groupId> - <artifactId>ant</artifactId> - </exclusion> - </exclusions> - </dependency> - </dependencies> - <executions> - <execution> - <id>install-duniter-plugin-jar</id> - <phase>pre-integration-test</phase> - <goals> - <goal>run</goal> - </goals> - <configuration> - <target> - <ac:if xmlns:ac="antlib:net.sf.antcontrib"> - <istrue value="${assembly.skip}" /> - <!-- reuse standalone files --> - <then> - <delete failonerror="false"> - <fileset dir="${run.es.home}/plugins" includes="**/duniter4j-*.jar" /> - </delete> - <copy todir="${run.es.home}/plugins/duniter4j-es-core" overwrite="true"> - <fileset dir="../duniter4j-core-client/target" includes="duniter4j-*${project.version}.jar"> - </fileset> - <fileset dir="../duniter4j-core-shared/target" includes="duniter4j-*${project.version}.jar"> - </fileset> - <fileset dir="../duniter4j-es-core/target" includes="duniter4j-*${project.version}.jar"> - </fileset> - </copy> - <copy todir="${run.es.home}/plugins/duniter4j-es-core" overwrite="true"> - <fileset dir="../duniter4j-es-core/target" includes="duniter4j-*${project.version}.jar"> - </fileset> - </copy> - <copy todir="${run.es.home}/plugins/duniter4j-es-user" overwrite="true"> - <fileset dir="../duniter4j-es-user/target" includes="duniter4j-*${project.version}.jar"> - </fileset> - </copy> - <copy todir="${run.es.home}/plugins/duniter4j-es-subscription" overwrite="true"> - <fileset dir="../duniter4j-es-subscription/target" includes="duniter4j-*${project.version}.jar"> - </fileset> - </copy> - </then> - <else> - <delete dir="${project.build.directory}/${bundlePrefix}" /> - <delete dir="${run.es.home}" /> - <!-- Unzip standalone zip--> - <unzip src="${project.build.directory}/${bundlePrefix}-standalone.zip" dest="${project.build.directory}" overwrite="true"> - </unzip> - <move file="${project.build.directory}/${bundlePrefix}" tofile="${run.es.home}" /> - </else> - </ac:if> - - <!-- Use files from src/test/es-home --> - <copy todir="${run.es.home}" overwrite="true"> - <fileset dir="${project.basedir}/src/test/es-home" includes="**/*.*"> - </fileset> - </copy> - </target> - </configuration> - </execution> - </executions> - </plugin> - - <plugin> - <groupId>org.codehaus.mojo</groupId> - <artifactId>exec-maven-plugin</artifactId> - <dependencies> - <dependency> - <groupId>org.elasticsearch</groupId> - <artifactId>elasticsearch</artifactId> - <version>${elasticsearch.version}</version> - </dependency> - <dependency> - <groupId>log4j</groupId> - <artifactId>log4j</artifactId> - <version>${log4j.version}</version> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - <version>${jna.version}</version> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna-platform</artifactId> - <version>${jna.version}</version> - <exclusions> - <exclusion> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - </exclusion> - </exclusions> - </dependency> - <dependency> - <groupId>javax.websocket</groupId> - <artifactId>javax.websocket-api</artifactId> - <version>1.1</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-client</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-client</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-server</artifactId> - <version>${tyrus.version}</version> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-server</artifactId> - <version>${tyrus.version}</version> - </dependency> - </dependencies> - <executions> - <execution> - <id>run</id> - <goals> - <goal>java</goal> - </goals> - <phase>integration-test</phase> - <configuration> - <mainClass>org.elasticsearch.bootstrap.Elasticsearch</mainClass> - <arguments> - <argument>start</argument> - </arguments> - <includeProjectDependencies>false</includeProjectDependencies> - <includePluginDependencies>true</includePluginDependencies> - <systemProperties> - <systemProperty> - <key>es.path.home</key> - <value>${run.es.home}</value> - </systemProperty> - </systemProperties> - </configuration> - </execution> - </executions> - </plugin> - </plugins> - </build> - - <properties> - <run.es.home>${project.build.directory}/es-run-home</run.es.home> - </properties> - </profile> - </profiles> -</project> diff --git a/duniter4j-es-assembly/src/license/THIRD-PARTY.properties b/duniter4j-es-assembly/src/license/THIRD-PARTY.properties deleted file mode 100644 index 8610ec5bda0d6673a7c224b7fd16226cc1bcd664..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/license/THIRD-PARTY.properties +++ /dev/null @@ -1,26 +0,0 @@ -# Generated by org.codehaus.mojo.license.AddThirdPartyMojo -#------------------------------------------------------------------------------- -# Already used licenses in project : -# - ASL, version 2 -# - Apache License 2.0 -# - Apache License Version 2.0 -# - BSD License -# - CC0 1.0 Universal -# - COMMON DEVELOPMENT AND DISTRIBUTION LICENSE (CDDL) Version 1.0 -# - Eclipse Public License 1.0 -# - General Public License (GPL) v3 -# - Indiana University Extreme! Lab Software License, vesion 1.1.1 -# - LGPL, version 2.1 -# - Lesser General Public License (LGPL) v 3.0 -# - Lesser General Public License (LPGL) -# - Lesser General Public License (LPGL) v 2.1 -# - MIT License -# - New BSD License -# - Public Domain, per Creative Commons CC0 -# - The Apache Software License, Version 2.0 -#------------------------------------------------------------------------------- -# Please fill the missing licenses for dependencies : -# -# -#Tue Jan 05 15:24:57 CET 2016 -commons-primitives--commons-primitives--1.0=The Apache Software License, Version 2.0 diff --git a/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml b/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml deleted file mode 100644 index 184f1a706f38af1e13b7547098686fc32d0b36d2..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml +++ /dev/null @@ -1,267 +0,0 @@ -# ======================== Elasticsearch Configuration ========================= -# -# NOTE: Elasticsearch comes with reasonable defaults for most settings. -# Before you set out to tweak and tune the configuration, make sure you -# understand what are you trying to accomplish and the consequences. -# -# The primary way of configuring a node is via this file. This template lists -# the most important settings you may want to configure for a production cluster. -# -# Please see the documentation for further information on configuration options: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/setup-configuration.html> -# -# ---------------------------------- Cluster ----------------------------------- -# -# Use a descriptive name for your cluster: -# -cluster.name: duniter4j-es-g1 -# -# ------------------------------------ Node ------------------------------------ -# -# Use a descriptive name for the node: -# -# node.name: node-1 -# -# Add custom attributes to the node: -# -# node.rack: r1 -# -# ----------------------------------- Paths ------------------------------------ -# -# Path to directory where to store the data (separate multiple locations by comma): -# -# path.data: /path/to/data -# -# Path to log files: -# -# path.logs: /path/to/logs -# -# ----------------------------------- Memory ----------------------------------- -# -# Lock the memory on startup: -# -# bootstrap.mlockall: true -# -# Make sure that the `ES_HEAP_SIZE` environment variable is set to about half the memory -# available on the system and that the owner of the process is allowed to use this limit. -# -# Elasticsearch performs poorly when the system is swapping the memory. -# -# ---------------------------------- Network ----------------------------------- -# -# Set the bind address to a specific IP (IPv4 or IPv6): -# -# network.host: 192.168.233.118 -# -# Set a custom port for HTTP: -# -# http.port: 9200-9300 - -http.cors.allow-origin: "/.*/" -http.cors.enabled: true - -# Internal transport layer -# -# transport.tcp.port: 9210-9220 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-network.html> -# -# --------------------------------- Discovery ---------------------------------- -# -# Pass an initial list of hosts to perform discovery when new node is started: -# The default list of hosts is ["127.0.0.1", "[::1]"] -# -# discovery.zen.ping.unicast.hosts: ["host1", "host2"] -# discovery.zen.ping.unicast.hosts: ["127.0.0.1", ""] -# -# Prevent the "split brain" by configuring the majority of nodes (total number of nodes / 2 + 1): -# -# discovery.zen.minimum_master_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-discovery.html> -# -# ---------------------------------- Gateway ----------------------------------- -# -# Block initial recovery after a full cluster restart until N nodes are started: -# -# gateway.recover_after_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-gateway.html> -# -# ---------------------------------- Various ----------------------------------- -# -# Disable starting multiple nodes on a single system: -# -# node.max_local_storage_nodes: 1 -# -# Require explicit names when deleting indices: -# -# action.destructive_requires_name: true -# -# Security to isolate plugin classpath - /!\ WARNING: should be DISABLE for Duniter4j -# -security.manager.enabled: false -# -# ---------------------------------- Duniter4j --------------------------------- -# -# Enable duniter4j plugins -# -# duniter.enabled: false -# -# Delete then create all indices at startup - /!\ WARNING: DO NOT set to true in production -# -# duniter.indices.reload: true -# -# Default string analyzer -# -duniter.string.analyzer: french -# -# Enabling blockchain synchronization (default: false) -# -duniter.blockchain.enable: true -# -# Force blockchain full synchronization - /!\ WARNING: all user events will be reset to 'unread' -# -# duniter.blockchain.reload: true -# duniter.blockchain.reload.from: 18900 -# duniter.blockchain.reload.to: 19000 -# -# Duniter node address -# -duniter.host: g1.duniter.org -duniter.port: 10901 -# duniter.useSsl: true -# -# Compute statistics on indices (each hour) ? (default: true) -# -# duniter.stats.enable: false -# -# ---------------------------------- Duniter4j security module ------------------- -# -# Keyring, use to sign emitted documents (user events, subscription, etc.). -# If not set, random keys will be generated. -# -# duniter.keyring.salt: -# duniter.keyring.password: -# -# Enable security - will restrict HTTP access to only Duniter4j known indices - /!\ WARNING: should be enable for production use -# -duniter.security.enable: true -# -# ---------------------------------- Duniter4j P2P module ------------------------- -# -# Enable P2P synchronize between ES peers ? (default: true) -# -# duniter.p2p.enable: false -# -# Enable P2P synchronisation using websocket ? (default: true) -# -# duniter.p2p.ws.enable: false -# -# Time delay (in seconds) to request last documents to peer (e.g. if peer's clock is late). (default: 3600s = 1h) -# -# duniter.p2p.peerTimeOffset: 3600 -# -# Enable discovery on network peers, to automatically synchronize this peers (default: true) -# -# duniter.p2p.discovery.enable: false -# -# Pass a list of hosts to always synchronize (default: <empty>) -# -duniter.p2p.includes.endpoints: [ - "ES_USER_API g1.data.duniter.fr 443", - "ES_SUBSCRIPTION_API g1.data.duniter.fr 443" -] -# -# Pass a list of pubkeys to always synchronize (default: <empty>) -# -# duniter.p2p.includes.pubkeys: [""] -# -# ---------------------------------- Duniter4j document moderation --------------- -# -# Filter too old document, if time older that 'maxPastDelta' (in seconds). (default: 7200 =2h) -# -# duniter.document.time.maxPastDelta: 7200 -# -# Filter document in the futur, if time greater that 'maxFutureDelta' (in seconds). (default: 600 =10min) -# -# duniter.document.time.maxFutureDelta: 600 -# -# Allow admin (define in duniter.keyring) to delete documents ? (default: true) -# -# duniter.document.allowAdminDeletion: true -# -# ---------------------------------- Duniter4j Mail module ----------------------- -# -# Enable mail module ? -# -duniter.mail.enable: false -# -# Mail: SMTP server configuration (host and port) -# -# duniter.mail.smtp.host: localhost -# duniter.mail.smtp.port: 25 -# -# Mail: SMTP server SSL security -# -# duniter.mail.smtp.ssl: true -# duniter.mail.smtp.starttls: true -# -# Mail: SMTP server authentication -# -# duniter.mail.smtp.username: -# duniter.mail.smtp.password: -# -# Mail: 'from' address -# -# duniter.mail.from: no-reply@domain.com -# -# Mail: admin address -# -# duniter.mail.admin: user@domain.com -# -# Mail: subject prefix -# -# duniter.mail.subject.prefix: '[Cesium+]' - -# ---------------------------------- Duniter4j Websocket server ---------------------- -# -# Websocket port (default: 9400-9410) -# -duniter.ws.port: 9400-9410 -# -# ---------------------------------- Duniter4j Subscription module ------------------- -# -# Enable subscription module (Need to enable mail features) -# -duniter.subscription.enable: false -# -# Email subscription: Day of the week to trigger weekly (default: 2 = monday) -# -# duniter.subscription.email.dayOfWeek: 2 -# -# Email subscription: Hour in day to trigger daily email subscription (default: 3 AM) -# -# duniter.subscription.email.hourOfDay: 3 -# -# 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' -# -# ---------------------------------- 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/main/assembly/config/logging.yml b/duniter4j-es-assembly/src/main/assembly/config/logging.yml deleted file mode 100644 index 37e2601508fa9dba747cfc275d0db34765f9187d..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/main/assembly/config/logging.yml +++ /dev/null @@ -1,98 +0,0 @@ -# you can override this using by setting a system property, for example -Des.logger.level=DEBUG -es.logger.level: INFO -rootLogger: ${es.logger.level}, console, file -logger: - # log action execution errors for easier debugging - action: DEBUG - - # deprecation logging, turn to DEBUG to see them - deprecation: INFO, deprecation_log_file - - # reduce the logging for aws, too much is logged under the default INFO - com.amazonaws: WARN - # aws will try to do some sketchy JMX stuff, but its not needed. - com.amazonaws.jmx.SdkMBeanRegistrySupport: ERROR - com.amazonaws.metrics.AwsSdkMetrics: ERROR - - duniter: INFO - #duniter.p2p: TRACE - security: INFO - cluster.metadata: ERROR - cluster.routing.allocation: ERROR - - org.duniter: INFO - org.nuiton.i18n: ERROR - org.nuiton.config: ERROR - org.nuiton.converter: WARN - org.apache.http: WARN - org.apache.http.client: ERROR - org.glassfish.grizzly: WARN - org.glassfish.tyrus: WARN - - # gateway - #gateway: DEBUG - #index.gateway: DEBUG - - # peer shard recovery - #indices.recovery: DEBUG - - # discovery - #discovery: TRACE - - index.search.slowlog: TRACE, index_search_slow_log_file - index.indexing.slowlog: TRACE, index_indexing_slow_log_file - -additivity: - index.search.slowlog: false - index.indexing.slowlog: false - deprecation: false - -appender: - console: - type: console - layout: - type: consolePattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %.10000m%n" - - # Use the following log4j-extras RollingFileAppender to enable gzip compression of log files. - # For more information see https://logging.apache.org/log4j/extras/apidocs/org/apache/log4j/rolling/RollingFileAppender.html - #file: - #type: extrasRollingFile - #file: ${path.logs}/${cluster.name}.log - #rollingPolicy: timeBased - #rollingPolicy.FileNamePattern: ${path.logs}/${cluster.name}.log.%d{yyyy-MM-dd}.gz - #layout: - #type: pattern - #conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - deprecation_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_deprecation.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_search_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_search_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_indexing_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_indexing_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" diff --git a/duniter4j-es-assembly/src/main/assembly/standalone.xml b/duniter4j-es-assembly/src/main/assembly/standalone.xml deleted file mode 100644 index be7c8367cb3f92b3c99c89a1a4a335c4b86c2978..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/main/assembly/standalone.xml +++ /dev/null @@ -1,92 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> -<!-- - #%L - allegro-obsdeb :: UI :: Swing - $Id:$ - $HeadURL:$ - %% - Copyright (C) 2009 - 2013 Ifremer - %% - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU Affero 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 Affero General Public License - along with this program. If not, see <http://www.gnu.org/licenses/>. - #L% - --> - -<assembly - xmlns="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.2" - xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" - xsi:schemaLocation="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.2 http://maven.apache.org/xsd/assembly-1.1.2.xsd"> - <id>standalone</id> - <formats> - <format>zip</format> - </formats> - - <fileSets> - - <fileSet> - <directory>target/elasticsearch-${elasticsearch.version}</directory> - <outputDirectory/> - <excludes> - <exclude>bin/elasticsearch</exclude> - <exclude>bin/plugin</exclude> - <exclude>config/elasticsearch.yml</exclude> - <exclude>config/logging.yml</exclude> - </excludes> - </fileSet> - <fileSet> - <directory>target/elasticsearch-${elasticsearch.version}</directory> - <outputDirectory/> - <includes> - <include>bin/elasticsearch</include> - <include>bin/plugin</include> - </includes> - <fileMode>0755</fileMode> - </fileSet> - - <!-- default configuration file --> - <fileSet> - <directory>src/main/assembly/config</directory> - <outputDirectory>config</outputDirectory> - <includes> - <include>elasticsearch.yml</include> - <include>logging.yml</include> - </includes> - </fileSet> - - <!-- websocket lib (tyrus ) - <fileSet> - <directory>target/elasticsearch-${elasticsearch.version}</directory> - <outputDirectory>lib</outputDirectory> - <includes> - <include>elasticsearch.yml</include> - <include>logging.yml</include> - </includes> - </fileSet>--> - </fileSets> - - <dependencySets> - <dependencySet> - <outputDirectory>lib</outputDirectory> - <useProjectArtifact>false</useProjectArtifact> - <useTransitiveFiltering>true</useTransitiveFiltering> - <includes> - <include>javax.websocket:javax.websocket-api</include> - <include>org.glassfish.tyrus:tyrus-client</include> - <include>org.glassfish.tyrus:tyrus-container-grizzly-client</include> - <include>org.glassfish.tyrus:tyrus-server</include> - <include>org.glassfish.tyrus:tyrus-container-grizzly-server</include> - </includes> - <fileMode>0555</fileMode> - </dependencySet> - </dependencySets> -</assembly> diff --git a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml deleted file mode 100644 index 124f71b8c5443388105ff26fe60d1b987a37dcab..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml +++ /dev/null @@ -1,269 +0,0 @@ -# ======================== Elasticsearch Configuration ========================= -# -# NOTE: Elasticsearch comes with reasonable defaults for most settings. -# Before you set out to tweak and tune the configuration, make sure you -# understand what are you trying to accomplish and the consequences. -# -# The primary way of configuring a node is via this file. This template lists -# the most important settings you may want to configure for a production cluster. -# -# Please see the documentation for further information on configuration options: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/setup-configuration.html> -# -# ---------------------------------- Cluster ----------------------------------- -# -# Use a descriptive name for your cluster: -# -cluster.name: g1-es-data-test -# -# ------------------------------------ Node ------------------------------------ -# -# Use a descriptive name for the node: -# -node.name: EIS-DEV -# -# Add custom attributes to the node: -# -# node.rack: r1 -# -# ----------------------------------- Paths ------------------------------------ -# -# Path to directory where to store the data (separate multiple locations by comma): -# -# path.data: /path/to/data -# -# Path to log files: -# -# path.logs: /path/to/logs -# -# ----------------------------------- Memory ----------------------------------- -# -# Lock the memory on startup: -# -# bootstrap.mlockall: true -# -# Make sure that the `ES_HEAP_SIZE` environment variable is set to about half the memory -# available on the system and that the owner of the process is allowed to use this limit. -# -# Elasticsearch performs poorly when the system is swapping the memory. -# -# ---------------------------------- Network ----------------------------------- -# -# Set the bind address to a specific IP (IPv4 or IPv6): -# -# network.host: 192.168.233.118 -# -# Set a custom port for HTTP: -# -# http.port: 9200-9300 - -http.cors.allow-origin: "/.*/" -http.cors.enabled: true - -# Internal transport layer -# -# transport.tcp.port: 9210-9220 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-network.html> -# -# --------------------------------- Discovery ---------------------------------- -# -# Pass an initial list of hosts to perform discovery when new node is started: -# The default list of hosts is ["127.0.0.1", "[::1]"] -# -# discovery.zen.ping.unicast.hosts: ["host1", "host2"] -#discovery.zen.ping.unicast.hosts: ["127.0.0.1", ""] -# -# Prevent the "split brain" by configuring the majority of nodes (total number of nodes / 2 + 1): -# -# discovery.zen.minimum_master_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-discovery.html> -# -# ---------------------------------- Gateway ----------------------------------- -# -# Block initial recovery after a full cluster restart until N nodes are started: -# -# gateway.recover_after_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-gateway.html> -# -# ---------------------------------- Various ----------------------------------- -# -# Disable starting multiple nodes on a single system: -# -# node.max_local_storage_nodes: 1 -# -# Require explicit names when deleting indices: -# -# rest.destructive_requires_name: true -# -# Security to isolate plugin classpath - /!\ WARNING: should be DISABLE for Duniter4j -# -security.manager.enabled: false -# -# ---------------------------------- Duniter4j --------------------------------- -# -# Enable duniter4j plugins -# -# duniter.enabled: false -# -# Delete then create all indices at startup - /!\ WARNING: DO NOT set to true in production -# -# duniter.indices.reload: true -# -# Default string analyzer -# -duniter.string.analyzer: french -# -# Enabling blockchain synchronization -# -duniter.blockchain.enable: true -# -# Force blockchain full synchronization - /!\ WARNING: all user events will be reset to 'unread' -# -# duniter.blockchain.reload: true -# duniter.blockchain.reload.from: 18900 -# duniter.blockchain.reload.to: 19000 -# -# Duniter node address -# -duniter.host: g1.duniter.fr -duniter.port: 443 -duniter.useSsl: true -# -# Compute statistics on indices (each hour) ? (default: true) -# -# duniter.stats.enable: false -# -# ---------------------------------- Duniter4j security module ------------------- -# -# Keyring, use to sign emitted documents (user events, subscription, etc.). -# If not set, random keys will be generated. -# -duniter.keyring.salt: 'abc' -duniter.keyring.password: 'def' -# -# Enable security - will restrict HTTP access to only Duniter4j known indices - /!\ WARNING: should be enable for production use -# -duniter.security.enable: true -# -# Security token prefix (default: 'duniter-') -# -# duniter.auth.token.prefix: duniter- -# -# Token validity duration, in seconds (default: 600) -# -# duniter.auth.tokenValidityDuration: 3600 # = 1hour - -# ---------------------------------- Duniter4j P2P module ------------------------- -# -# Enable P2P synchronize between ES peers ? (default: true) -# -# duniter.p2p.enable: false -# -# Enable P2P synchronisation using websocket ? (default: true) -# -# duniter.p2p.ws.enable: false -# -# Time delay (in seconds) to request last documents to peer (e.g. if peer's clock is late). (default: 3600s = 1h) -# -# duniter.p2p.peerTimeOffset: 3600 -# -# Enable discovery on network peers, to automatically synchronize this peers (default: true) -# -duniter.p2p.discovery.enable: false -# -# Pass a list of hosts to always synchronize (default: <empty>) -# -duniter.p2p.includes.endpoints: [ - "ES_USER_API g1.data.duniter.fr 443", - "ES_SUBSCRIPTION_API g1.data.duniter.fr 443" -] -# -# Pass a list of pubkeys to always synchronize (default: <empty>) -# -#duniter.p2p.includes.pubkeys: [ -# "38MEAZN68Pz1DTvT3tqgxx4yQP6snJCQhPqEFxbDk4aE" -#] -#duniter.p2p.fullResyncAtStartup: true -# -# ---------------------------------- Duniter4j Mail module ----------------------- -# -# Enable mail module ? -# -duniter.mail.enable: false -# -# Mail: SMTP server configuration (host and port) -# -#duniter.mail.smtp.host: localhost -#duniter.mail.smtp.port: 25 -# -# Mail: SMTP server SSL security -# -#duniter.mail.smtp.ssl: true -#duniter.mail.smtp.starttls: true -# -# Mail: SMTP server authentication -# -#duniter.mail.smtp.username: -#duniter.mail.smtp.password: -# -# Mail: 'from' address -# -#duniter.mail.from: no-reply@domain.com -# -# Mail: admin address -# -#duniter.mail.admin: user@domain.com -# -# Mail: subject prefix -# -#duniter.mail.subject.prefix: '[Cesium+]' - -# ---------------------------------- Duniter4j Websocket server ---------------------- -# -# Websocket port (default: 9400-9410) -# -duniter.ws.port: 9400-9410 -# -# ---------------------------------- Duniter4j Subscription module ------------------- -# -# Enable subscription module (Need to enable mail features) -# -duniter.subscription.enable: true -# -# Opions to DEBUG this features -# -#duniter.subscription.email.atStartup: false -#duniter.subscription.email.debug: false -# -# Email subscription: Day of the week to trigger weekly (default: 2 = monday) -# -#duniter.subscription.email.dayOfWeek: 2 -# -# Email subscription: Hour in day to trigger daily email subscription (default: 3 AM) -# -#duniter.subscription.email.hourOfDay: 3 -# -# 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' -# -# ---------------------------------- 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-assembly/src/test/es-home/config/logging.yml b/duniter4j-es-assembly/src/test/es-home/config/logging.yml deleted file mode 100644 index 9963d24c4caa4f672f11dcd7113bc22fe81bb61c..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/es-home/config/logging.yml +++ /dev/null @@ -1,114 +0,0 @@ -# you can override this using by setting a system property, for example -Des.logger.level=DEBUG -es.logger.level: INFO -rootLogger: ${es.logger.level}, console, file -logger: - # log action execution errors for easier debugging - action: DEBUG - - # deprecation logging, turn to DEBUG to see them - deprecation: INFO, deprecation_log_file - - # reduce the logging for aws, too much is logged under the default INFO - com.amazonaws: WARN - # aws will try to do some sketchy JMX stuff, but its not needed. - com.amazonaws.jmx.SdkMBeanRegistrySupport: ERROR - com.amazonaws.metrics.AwsSdkMetrics: ERROR - - duniter: DEBUG - #duniter.core: DEBUG - #duniter.security: ERROR - #duniter.user.event: DEBUG - #duniter.network.p2p: DEBUG - #duniter.network.peer: DEBUG - #duniter.mail: DEBUG - #duniter.subscription: DEBUG - #duniter.p2p: TRACE - - security: DEBUG - cluster.metadata: ERROR - cluster.routing.allocation: ERROR - - org.duniter: DEBUG - #org.duniter.core.util.LockManager: DEBUG - #org.duniter.core.beans: DEBUG - #org.duniter.core.client.service: DEBUG - #org.duniter.elasticsearch: DEBUG - #org.duniter.elasticsearch.service: DEBUG - #org.duniter.elasticsearch.user.service: DEBUG - #org.duniter.elasticsearch.subscription.service: DEBUG - - org.nuiton.i18n: ERROR - org.nuiton.config: ERROR - org.nuiton.converter: WARN - org.apache.http: WARN - org.apache.http.client: ERROR - org.glassfish.grizzly: WARN - org.glassfish.tyrus: WARN - - # gateway - #gateway: DEBUG - #index.gateway: DEBUG - - # peer shard recovery - #indices.recovery: DEBUG - - # discovery - #discovery: TRACE - - index.search.slowlog: TRACE, index_search_slow_log_file - index.indexing.slowlog: TRACE, index_indexing_slow_log_file - -additivity: - index.search.slowlog: false - index.indexing.slowlog: false - deprecation: false - -appender: - console: - type: console - layout: - type: consolePattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %.10000m%n" - - # Use the following log4j-extras RollingFileAppender to enable gzip compression of log files. - # For more information see https://logging.apache.org/log4j/extras/apidocs/org/apache/log4j/rolling/RollingFileAppender.html - #file: - #type: extrasRollingFile - #file: ${path.logs}/${cluster.name}.log - #rollingPolicy: timeBased - #rollingPolicy.FileNamePattern: ${path.logs}/${cluster.name}.log.%d{yyyy-MM-dd}.gz - #layout: - #type: pattern - #conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - deprecation_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_deprecation.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_search_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_search_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_indexing_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_indexing_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" diff --git a/duniter4j-es-assembly/src/test/misc/blocksByIssuer.sh b/duniter4j-es-assembly/src/test/misc/blocksByIssuer.sh deleted file mode 100755 index 26220b2841fa812ba6649242514cd673cbd46d8d..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/blocksByIssuer.sh +++ /dev/null @@ -1,21 +0,0 @@ -#!/bin/sh - -curl -XPOST 'http://localhost:9200/g1/block/_search?pretty' -d ' - { - "size": 0, - "aggs": { - "blocksByIssuer": { - "terms": { - "field": "issuer", - "size": 0 - }, - "aggs" : { - "difficulty_stats" : { - "stats" : { - "field" : "powMin" - } - } - } - } - } - }' diff --git a/duniter4j-es-assembly/src/test/misc/movementByRange.sh b/duniter4j-es-assembly/src/test/misc/movementByRange.sh deleted file mode 100755 index 679019927b2dac6f6787cd8501e3d4601c3db004..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/movementByRange.sh +++ /dev/null @@ -1,36 +0,0 @@ -#!/bin/sh -curl -XPOST 'http://localhost:9200/gtest/movement/_search?pretty' -d ' - { - "size": 1000, - "query": { - "bool": { - "filter": [ - {"term": {"recipient" : "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of" }}, - {"terms": {"code" : ["MEMBER_JOIN","MEMBER_ACTIVE","MEMBER_LEAVE","MEMBER_EXCLUDE","MEMBER_REVOKE"] }} - ] - } - }, - "aggs": { - "tx": { - "range": { - "field" : "medianTime", - "ranges" : [ - { "from" : 0, "to" : 1492041600 } - ] - }, - "aggs" : { - "received" : { - "filter": {"term": {"recipient": "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"}}, - "aggs": { - "received_stats": { - "stats": { - "field": "amount" - } - } - } - } - } - } - } - }' - diff --git a/duniter4j-es-assembly/src/test/misc/test_docstat.sh b/duniter4j-es-assembly/src/test/misc/test_docstat.sh deleted file mode 100755 index 408b4458d41bb8e8a3fdf303aee3f67a927c661d..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/test_docstat.sh +++ /dev/null @@ -1,38 +0,0 @@ - -curl -XPOST 'https://g1-test.data.duniter.fr/docstat/record/_search?pretty' -d ' - { - "size": 0, - "aggs": { - "range": { - "range": { - "field": "time", - "ranges": [ - {"from":1506016800, "to": 1506178800 } - ] - }, - "aggs": { - "index": { - "terms": { - "field": "index", - "size": 0 - }, - "aggs" : { - "type": { - "terms": { - "field": "indexType", - "size": 0 - }, - "aggs": { - "max" : { - "max" : { - "field" : "count" - } - } - } - } - } - } - } - } - } - }' \ No newline at end of file diff --git a/duniter4j-es-assembly/src/test/misc/test_es_query.sh b/duniter4j-es-assembly/src/test/misc/test_es_query.sh deleted file mode 100755 index 66d191e02b73c9fbd9705df8996700280a8e84c7..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/test_es_query.sh +++ /dev/null @@ -1,72 +0,0 @@ -#!/bin/sh - -curl -XPOST 'http://localhost:9200/g1/block/_search?pretty' -d ' - { - "size": 0, - "aggs": { - "txByRange": { - "range": { - "field" : "medianTime", - "ranges" : [ - { "from" : 1491955200, "to" : 1492041600 } - ] - }, - "aggs" : { - "tx_stats" : { - "stats" : { - "script" : { - "inline" : "txcount", - "lang": "native" - } - } - }, - "time" : { - "stats" : { "field" : "medianTime" } - } - } - } - } - }' - - -curl -XPOST 'http://localhost:9200/g1/block/_search?pretty' -d ' - { - "size": 0, - "aggs": { - "blocksByIssuer": { - "terms": { - "field": "issuer", - "size": 0 - }, - "aggs" : { - "difficulty_stats" : { - "stats" : { - "field" : "difficulty" - } - } - } - } - } - }' - -curl -XPOST 'http://localhost:9200/g1/peer/_search?pretty' -d ' -{ - "size" : 1000, - "query" : { - "constant_score" : { - "filter" : { - "bool" : { - "must" : [ { - "bool" : { - "must" : { - "term" : { - "api" : "ES_USER_API" - } - } - } - }] - } - } - } - } -}' \ No newline at end of file diff --git a/duniter4j-es-assembly/src/test/misc/test_geo_distance.sh b/duniter4j-es-assembly/src/test/misc/test_geo_distance.sh deleted file mode 100755 index 30c7538d44ce6a115c0f9c9345500b880ea0ab7e..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/test_geo_distance.sh +++ /dev/null @@ -1,20 +0,0 @@ -#!/bin/sh - -curl -XPOST 'https://g1.data.duniter.fr/page/record/_search?pretty&_source=title' -d ' - { - "size": 100, - "query": { - "bool": { - "filter": [{ - "geo_distance": { - "distance": "500km", - "geoPoint": { - "lat": 47.2186371, - "lon": -1.5541362 - } - } - }] - } - } - }' - diff --git a/duniter4j-es-assembly/src/test/misc/test_queries_graph.sh b/duniter4j-es-assembly/src/test/misc/test_queries_graph.sh deleted file mode 100755 index 257c5976f75ab10ed73b0246cfce2ff1bfb25ad9..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/test_queries_graph.sh +++ /dev/null @@ -1,18 +0,0 @@ -#!/bin/sh -curl -XPOST 'http://gtest.data.duniter.fr:80/user/event/_search?pretty' -d ' - { - "size": 1000, - "query": { - "bool": { - "filter": [ - {"term": {"recipient" : "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of" }}, - {"terms": {"code" : ["MEMBER_JOIN","MEMBER_ACTIVE","MEMBER_LEAVE","MEMBER_EXCLUDE","MEMBER_REVOKE"] }} - ] - } - }, - "sort" : [ - { "time" : {"order" : "desc"}} - ], - _source: ["code", "time"] - }' - diff --git a/duniter4j-es-assembly/src/test/misc/test_scroll.sh b/duniter4j-es-assembly/src/test/misc/test_scroll.sh deleted file mode 100755 index f98fa5515f432e94720f134b418163d9a956b000..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/test_scroll.sh +++ /dev/null @@ -1,55 +0,0 @@ -#!/bin/sh - -#curl -XPOST 'https://g1-test.data.duniter.fr/user/profile/_search?pretty' -d ' -# { -# "query":{"bool":{"should":{"range":{"time":{"gte":0}}}}}, -# "from":0, -# "scroll":"1m", -# "size": 0 -# }' - - -#curl -XPOST 'https://g1.data.duniter.fr/subscription/execution/_search?pretty' -d ' -# { -# "query":{"bool":{"should":{"range":{"time":{"gte":0}}}}}, -# "size": 1000, -# "sort": "time" -# }' - - -#curl -XPOST 'http://localhost:9200/user/profile/_search/scroll?scroll=1m' -d 'cXVlcnlUaGVuRmV0Y2g7Mzs4OTU6dU5jU2NMeFlRRi0xbVZGSlVxc3dndzs4OTY6dU5jU2NMeFlRRi0xbVZGSlVxc3dndzs4OTQ6dU5jU2NMeFlRRi0xbVZGSlVxc3dndzswOw==' - -curl -XPOST 'http://localhost:9200/history/delete/_search?scroll=1m' - -#curl -XPOST 'http://localhost:9200/history/delete/_search/scroll?scroll=1m' -d 'cXVlcnlUaGVuRmV0Y2g7Mjs3MToxNlZjRUplMVMyaW1sZERvdVU2dHZnOzcyOjE2VmNFSmUxUzJpbWxkRG91VTZ0dmc7MDs=' - -curl -XPOST 'http://localhost:9200/g1-test/peer/_search' -d '{ - "constant_score" : { - "filter" : { - "bool" : { - "must" : [ { - "bool" : { - "filter" : { - "term" : { - "api" : "ES_USER_API" - } - } - } - }, { - "nested" : { - "query" : { - "bool" : { - "filter" : { - "term" : { - "stats.status" : "UP" - } - } - } - }, - "path" : "stats" - } - } ] - } - } - } -}'' \ No newline at end of file diff --git a/duniter4j-es-assembly/src/test/misc/test_synchro.sh b/duniter4j-es-assembly/src/test/misc/test_synchro.sh deleted file mode 100755 index 7f564cb028fab99e845c11c11ceb7e03c31413ec..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/test_synchro.sh +++ /dev/null @@ -1,37 +0,0 @@ - -curl -XPOST 'https://g1.data.le-sou.org/g1/synchro/_search?pretty' -d ' - { - "size": 0, - "aggs": { - "range": { - "range": { - "field": "time", - "ranges": [ - {"from":0, "to": 9996178800 } - ] - }, - "aggs": { - "peer": { - "terms": { - "field": "peer", - "size": 0 - }, - "aggs" : { - "result": { - "nested": { - "path": "result" - }, - "aggs" : { - "inserts" : { - "stats": { - "field" : "result.inserts" - } - } - } - } - } - } - } - } - } - }' \ No newline at end of file diff --git a/duniter4j-es-assembly/src/test/misc/udByPeriods.sh b/duniter4j-es-assembly/src/test/misc/udByPeriods.sh deleted file mode 100755 index 7eac90e70b8bf21eda9c2668e1c795b1b9131b43..0000000000000000000000000000000000000000 --- a/duniter4j-es-assembly/src/test/misc/udByPeriods.sh +++ /dev/null @@ -1,33 +0,0 @@ -#!/bin/sh - -curl -XPOST 'http://localhost:9200/g1/block/_search?pretty' -d ' - { - "size": 1000, - query: { - filtered: { - filter: { - - bool: { - must: [ - { - exists: { - field: "dividend" - } - }, - { - range: { - medianTime: { - from: 1506837759, to: 201507961583 - } - } - } - ] - } - } - } - }, - _source: ["medianTime", "number", "dividend", "monetaryMass", "membersCount", "unitbase"], - sort: { - "medianTime" : "asc" - } - }' diff --git a/duniter4j-es-core/LICENSE.txt b/duniter4j-es-core/LICENSE.txt deleted file mode 100644 index 94a9ed024d3859793618152ea559a168bbcbb5e2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/LICENSE.txt +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - <one line to give the program's name and a brief idea of what it does.> - Copyright (C) <year> <name of author> - - 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/>. - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - <program> Copyright (C) <year> <name of author> - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -<http://www.gnu.org/licenses/>. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -<http://www.gnu.org/philosophy/why-not-lgpl.html>. diff --git a/duniter4j-es-core/pom.xml b/duniter4j-es-core/pom.xml deleted file mode 100644 index 9668b41fd689668bb30a9810ba30c309d428cada..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/pom.xml +++ /dev/null @@ -1,206 +0,0 @@ -<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> - <modelVersion>4.0.0</modelVersion> - - <parent> - <groupId>org.duniter</groupId> - <artifactId>duniter4j</artifactId> - <version>1.0.4-SNAPSHOT</version> - </parent> - - <artifactId>duniter4j-es-core</artifactId> - <packaging>jar</packaging> - <name>Duniter4j :: ElasticSearch Core plugin</name> - <description>A ElasticSearch plugin that can index data from a Duniter currency</description> - - <properties> - <!-- i18n configuration --> - <i18n.bundleOutputName>duniter4j-es-core-i18n</i18n.bundleOutputName> - <i18n.generateCsvFile>true</i18n.generateCsvFile> - <i18n.bundleCsvFile> - ${maven.gen.dir}/resources/META-INF/${i18n.bundleOutputName}.csv - </i18n.bundleCsvFile> - <config.i18nBundleName>${i18n.bundleOutputName}</config.i18nBundleName> - - </properties> - - <dependencies> - <dependency> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-core-client</artifactId> - <version>${project.version}</version> - <exclusions> - <exclusion> - <groupId>com.google.guava</groupId> - <artifactId>guava</artifactId> - </exclusion> - <exclusion> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-client</artifactId> - </exclusion> - <exclusion> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-client</artifactId> - </exclusion> - <exclusion> - <groupId>javax.websocket</groupId> - <artifactId>javax.websocket-api</artifactId> - </exclusion> - </exclusions> - </dependency> - - <!-- LOGGING DEPENDENCIES - SLF4J --> - <dependency> - <groupId>org.slf4j</groupId> - <artifactId>slf4j-api</artifactId> - </dependency> - <dependency> - <groupId>org.slf4j</groupId> - <artifactId>slf4j-log4j12</artifactId> - <optional>true</optional> - </dependency> - <dependency> - <groupId>log4j</groupId> - <artifactId>log4j</artifactId> - <optional>true</optional> - <scope>runtime</scope> - </dependency> - - <!-- Elastic Search --> - <dependency> - <groupId>org.elasticsearch</groupId> - <artifactId>elasticsearch</artifactId> - <scope>provided</scope> - </dependency> - <dependency> - <groupId>com.fasterxml.jackson.core</groupId> - <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> - <artifactId>jna</artifactId> - <scope>provided</scope> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna-platform</artifactId> - <scope>provided</scope> - <exclusions> - <exclusion> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - </exclusion> - </exclusions> - </dependency> - - <!-- Websocket --> - <dependency> - <groupId>javax.websocket</groupId> - <artifactId>javax.websocket-api</artifactId> - <scope>provided</scope> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-server</artifactId> - <scope>provided</scope> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-server</artifactId> - <scope>provided</scope> - </dependency> - - <!-- Unit test --> - <dependency> - <groupId>junit</groupId> - <artifactId>junit</artifactId> - <scope>test</scope> - </dependency> - </dependencies> - - <build> - <resources> - <resource> - <directory>src/main/filtered-resources</directory> - <filtering>true</filtering> - <includes> - <include>*.config</include> - <include>**/*.properties</include> - </includes> - </resource> - <resource> - <directory>src/main/resources</directory> - <filtering>false</filtering> - </resource> - </resources> - - <plugins> - <plugin> - <groupId>org.nuiton.i18n</groupId> - <artifactId>i18n-maven-plugin</artifactId> - - <executions> - <execution> - <id>scan-sources</id> - <configuration> - <entries> - <entry> - <specificGoal>parserValidation</specificGoal> - <basedir>${maven.src.dir}/main/java/</basedir> - <includes> - <param>**/**-validation.xml</param> - </includes> - </entry> - </entries> - </configuration> - <goals> - <goal>parserJava</goal> - <goal>parserValidation</goal> - <goal>gen</goal> - </goals> - </execution> - <execution> - <id>make-bundle</id> - <goals> - <goal>bundle</goal> - </goals> - </execution> - </executions> - </plugin> - - <plugin> - <artifactId>maven-assembly-plugin</artifactId> - <executions> - <execution> - <id>assembly-plugin</id> - <phase>package</phase> - <goals> - <goal>single</goal> - </goals> - <configuration> - <attach>true</attach> - <appendAssemblyId>false</appendAssemblyId> - <finalName>${project.artifactId}-${project.version}</finalName> - <descriptors> - <descriptor> - ${basedir}/src/main/assembly/plugin.xml - </descriptor> - </descriptors> - <skipAssembly>${assembly.skip}</skipAssembly> - </configuration> - </execution> - </executions> - </plugin> - </plugins> - </build> - -</project> diff --git a/duniter4j-es-core/src/license/THIRD-PARTY.properties b/duniter4j-es-core/src/license/THIRD-PARTY.properties deleted file mode 100644 index 10a89e2bcaf4082c40f141057fa9e10147c47390..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/license/THIRD-PARTY.properties +++ /dev/null @@ -1,35 +0,0 @@ -# Generated by org.codehaus.mojo.license.AddThirdPartyMojo -#------------------------------------------------------------------------------- -# Already used licenses in project : -# - ASL, version 2 -# - Apache 2.0 -# - Apache License 2.0 -# - Apache License Version 2.0 -# - BSD License -# - BSD licence -# - CC0 1.0 Universal -# - CDDL -# - CDDL+GPL -# - COMMON DEVELOPMENT AND DISTRIBUTION LICENSE (CDDL) Version 1.0 -# - Common Development and Distribution License (CDDL) v1.0 -# - Dual license consisting of the CDDL v1.1 and GPL v2 -# - Eclipse Public License 1.0 -# - GPLv2+CE -# - General Public License (GPL) v3 -# - Indiana University Extreme! Lab Software License, vesion 1.1.1 -# - LGPL, version 2.1 -# - Lesser General Public License (LGPL) v 3.0 -# - Lesser General Public License (LPGL) -# - Lesser General Public License (LPGL) v 2.1 -# - Lesser General Public License (LPGL) version 3.0 -# - MIT License -# - New BSD License -# - Public Domain, per Creative Commons CC0 -# - The Apache Software License, Version 2.0 -#------------------------------------------------------------------------------- -# Please fill the missing licenses for dependencies : -# -# -#Fri May 18 18:26:32 CEST 2018 -commons-primitives--commons-primitives--1.0=The Apache Software License, Version 2.0 -org.antlr--antlr-runtime--3.3=BSD License diff --git a/duniter4j-es-core/src/main/assembly/plugin.xml b/duniter4j-es-core/src/main/assembly/plugin.xml deleted file mode 100644 index 39a21301ee04cc92204da80b2b72221d0965fee1..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/assembly/plugin.xml +++ /dev/null @@ -1,42 +0,0 @@ -<?xml version="1.0"?> -<assembly> - <id>plugin</id> - - - <formats> - <format>zip</format> - </formats> - - <includeBaseDirectory>false</includeBaseDirectory> - - <dependencySets> - <dependencySet> - <outputDirectory>/</outputDirectory> - <useProjectArtifact>true</useProjectArtifact> - <useTransitiveFiltering>true</useTransitiveFiltering> - <excludes> - <exclude>org.elasticsearch:elasticsearch</exclude> - <exclude>net.java.dev.jna:jna</exclude> - <exclude>com.fasterxml.jackson.core:jackson-core</exclude> - <exclude>log4j:log4j</exclude> - </excludes> - </dependencySet> - </dependencySets> - - <fileSets> - <fileSet> - <includes> - <include>LICENSE</include> - </includes> - </fileSet> - - <fileSet> - <directory>target/classes</directory> - <outputDirectory/> - <includes> - <include>plugin-descriptor.properties</include> - <include>plugin-security.policy</include> - </includes> - </fileSet> - </fileSets> -</assembly> \ No newline at end of file diff --git a/duniter4j-es-core/src/main/filtered-resources/duniter4j.config b/duniter4j-es-core/src/main/filtered-resources/duniter4j.config deleted file mode 100644 index 053ef9a31becf9a95e3ad3a352f61dde65b2da3e..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/filtered-resources/duniter4j.config +++ /dev/null @@ -1,6 +0,0 @@ -app.name=duniter4j -duniter4j.config.path=sqqs -duniter4j.version=${project.version} -duniter4j.site.url=${project.url} -duniter4j.inceptionYear=${project.inceptionYear} -duniter4j.organizationName=${license.organizationName} diff --git a/duniter4j-es-core/src/main/filtered-resources/log4j.properties b/duniter4j-es-core/src/main/filtered-resources/log4j.properties deleted file mode 100644 index a730face0905880dd12f4cc5a87e46970f35b182..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/filtered-resources/log4j.properties +++ /dev/null @@ -1,33 +0,0 @@ - -# Global logging configuration -log4j.rootLogger=ERROR, stdout, file -#log4j.rootLogger=ERROR, stdout - -# Console output -log4j.appender.stdout=org.apache.log4j.ConsoleAppender -log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p %m%n - -# Duniter4j levels -log4j.logger.org.duniter=INFO -#log4j.logger.org.duniter.core.client=DEBUG -#log4j.logger.org.duniter.core.client.service=DEBUG -log4j.logger.org.duniter.elasticsearch=DEBUG - -# Other frameworks levels -log4j.logger.org.apache.http=ERROR -log4j.logger.org.nuiton.util=WARN -log4j.logger.org.nuiton.config=WARN -log4j.logger.org.nuiton.converter=WARN -log4j.logger.org.nuiton.i18n=ERROR -log4j.logger.org.elasticsearch=WARN -#log4j.logger.org.elasticsearch=INFO - -log4j.appender.file=org.apache.log4j.RollingFileAppender -log4j.appender.file.file=${duniter4j.log.file} -log4j.appender.file.MaxFileSize=10MB -log4j.appender.file.MaxBackupIndex=4 - -log4j.appender.file.layout=org.apache.log4j.PatternLayout -log4j.appender.file.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m%n - diff --git a/duniter4j-es-core/src/main/filtered-resources/plugin-descriptor.properties b/duniter4j-es-core/src/main/filtered-resources/plugin-descriptor.properties deleted file mode 100644 index ed67f81482b9a575df6954720a4738ca534a76b7..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/filtered-resources/plugin-descriptor.properties +++ /dev/null @@ -1,9 +0,0 @@ -name=duniter4j-es-core -description=Plugin for Duniter -version=${project.version} -site=false -jvm=true -classname=org.duniter.elasticsearch.Plugin -java.version=1.8 -elasticsearch.version=2.4.6 -isolated=false diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java deleted file mode 100644 index dfdac128d3b8941336f124bc9745be77f7779755..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java +++ /dev/null @@ -1,104 +0,0 @@ -package org.duniter.elasticsearch; - -/* - * #%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 com.google.common.collect.Lists; -import org.duniter.elasticsearch.dao.DaoModule; -import org.duniter.elasticsearch.rest.RestModule; -import org.duniter.elasticsearch.script.BlockchainTxCountScriptFactory; -import org.duniter.elasticsearch.security.SecurityModule; -import org.duniter.elasticsearch.service.ServiceModule; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.websocket.WebSocketModule; -import org.elasticsearch.common.component.LifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Module; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; - -import java.util.Collection; - -public class Plugin extends org.elasticsearch.plugins.Plugin { - - private ESLogger logger; - - private boolean enable; - - @Inject public Plugin(Settings settings) { - this.enable = settings.getAsBoolean("duniter.enabled", true); - this.logger = Loggers.getLogger("duniter.core", settings, new String[0]); - } - - @Override - public String name() { - return "duniter4j-es-core"; - } - - @Override - public String description() { - return "Duniter Core Plugin"; - } - - @Inject - public void onModule(org.elasticsearch.script.ScriptModule scriptModule) { - // TODO: in ES v5+, see example here : - // https://github.com/imotov/elasticsearch-native-script-example/blob/60a390f77f2fb25cb89d76de5071c52207a57b5f/src/main/java/org/elasticsearch/examples/nativescript/plugin/NativeScriptExamplesPlugin.java - scriptModule.registerScript("txcount", BlockchainTxCountScriptFactory.class); - } - - @Override - public Collection<Module> nodeModules() { - Collection<Module> modules = Lists.newArrayList(); - if (!enable) { - logger.warn(description() + " has been disabled."); - return modules; - } - modules.add(new SecurityModule()); - - modules.add(new WebSocketModule()); - modules.add(new RestModule()); - - - modules.add(new DaoModule()); - modules.add(new ServiceModule()); - //modules.add(new ScriptModule()); - return modules; - } - - @Override - public Collection<Class<? extends LifecycleComponent>> nodeServices() { - Collection<Class<? extends LifecycleComponent>> components = Lists.newArrayList(); - if (!enable) { - return components; - } - components.add(PluginSettings.class); - components.add(ThreadPool.class); - components.add(PluginInit.class); - return components; - } - - /* -- protected methods -- */ - - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java deleted file mode 100644 index 1c279625a528d67c8fc79e6b117e10db50b95d14..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java +++ /dev/null @@ -1,307 +0,0 @@ -package org.duniter.elasticsearch; - -/* - * #%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.core.client.model.elasticsearch.Currency; -import org.duniter.core.client.model.local.Peer; -import org.duniter.elasticsearch.dao.*; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.service.BlockchainService; -import org.duniter.elasticsearch.service.CurrencyService; -import org.duniter.elasticsearch.service.DocStatService; -import org.duniter.elasticsearch.service.PeerService; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestRequest; - -/** - * Created by blavenie on 17/06/16. - */ -public class PluginInit extends AbstractLifecycleComponent<PluginInit> { - - private final PluginSettings pluginSettings; - private final ThreadPool threadPool; - private final Injector injector; - private final ESLogger logger; - - @Inject - public PluginInit(Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) { - super(settings); - this.logger = Loggers.getLogger("duniter.core", settings, new String[0]); - this.pluginSettings = pluginSettings; - this.threadPool = threadPool; - this.injector = injector; - } - - @Override - protected void doStart() { - threadPool.scheduleOnClusterReady(() -> { - createIndices(); - - // Waiting cluster back to GREEN or YELLOW state, before doAfterStart - threadPool.scheduleOnClusterReady(this::doAfterStart); - - }); - } - - @Override - protected void doStop() { - - } - - @Override - protected void doClose() { - - } - - protected void createIndices() { - - // Reload All indices - if (pluginSettings.reloadAllIndices()) { - if (logger.isWarnEnabled()) { - logger.warn("Reloading indices..."); - } - - injector.getInstance(CurrencyService.class) - .deleteIndex() - .createIndexIfNotExists(); - - if (pluginSettings.enableDocStats()) { - injector.getInstance(DocStatService.class) - .deleteIndex() - .createIndexIfNotExists(); - } - - if (logger.isInfoEnabled()) { - logger.info("Reloading indices [OK]"); - } - } - - else if (pluginSettings.enableBlockchainSync() && pluginSettings.reloadBlockchainIndices() && pluginSettings.reloadBlockchainIndicesFrom() <= 0) { - if (logger.isWarnEnabled()) { - logger.warn("/!\\ Reloading blockchain indices..."); - } - injector.getInstance(CurrencyService.class) - .deleteIndex() - .createIndexIfNotExists(); - - if (logger.isInfoEnabled()) { - logger.info("Reloading blockchain indices [OK]"); - } - } - - else { - - - if (logger.isDebugEnabled()) { - logger.debug("Checking indices..."); - } - - injector.getInstance(CurrencyService.class) - .createIndexIfNotExists(); - - if (pluginSettings.enableDocStats()) { - injector.getInstance(DocStatService.class) - .createIndexIfNotExists(); - } - - if (logger.isDebugEnabled()) { - logger.debug("Checking indices [OK]"); - } - } - } - - protected void doAfterStart() { - - // Synchronize blockchain - if (pluginSettings.enableBlockchainSync()) { - - Peer peer = pluginSettings.checkAndGetPeer(); - - Currency currency; - try { - // Index (or refresh) node's currency - currency = injector.getInstance(CurrencyService.class) - .indexCurrencyFromPeer(peer, true); - } catch(Throwable e){ - logger.error(String.format("Error while indexing currency. Skipping blockchain indexation.", e.getMessage()), e); - throw e; - } - - final String currencyName = currency.getCurrencyName(); - - // Add access security rules, for the currency indices - injector.getInstance(RestSecurityController.class) - - // Add access to <currency>/block index - .allowIndexType(RestRequest.Method.GET, - currencyName, - BlockDao.TYPE) - .allowPostSearchIndexType( - currencyName, - BlockDao.TYPE) - - // Add access to <currency>/blockStat index - .allowIndexType(RestRequest.Method.GET, - currencyName, - BlockStatDao.TYPE) - .allowPostSearchIndexType( - currencyName, - BlockStatDao.TYPE) - - // Add access to <currency>/peer index - .allowIndexType(RestRequest.Method.GET, - currencyName, - PeerDao.TYPE) - .allowPostSearchIndexType( - currencyName, - PeerDao.TYPE) - - // Add access to <currency>/movement index - .allowIndexType(RestRequest.Method.GET, - currencyName, - MovementDao.TYPE) - .allowPostSearchIndexType( - currencyName, - MovementDao.TYPE) - - // Add access to <currency>/synchro index - .allowIndexType(RestRequest.Method.GET, - currencyName, - SynchroExecutionDao.TYPE) - .allowPostSearchIndexType( - currencyName, - SynchroExecutionDao.TYPE); - - /* TODO à décommenter quand les pending seront sauvegardés - injector.getInstance(DocStatService.class) - .registerIndex(currencyName, - PendingRegistrationDao.TYPE); - */ - - // If partial reload (from a block) - if (pluginSettings.reloadBlockchainIndices() && pluginSettings.reloadBlockchainIndicesFrom() > 0) { - // Delete blocs range [from,to] - if (pluginSettings.reloadBlockchainIndicesTo() > pluginSettings.reloadBlockchainIndicesFrom()) { - if (logger.isWarnEnabled()) { - logger.warn(String.format("/!\\ Re-indexing blockchain range [%s-%s]...", - pluginSettings.reloadBlockchainIndicesFrom(), - pluginSettings.reloadBlockchainIndicesTo())); - } - - injector.getInstance(BlockchainService.class) - .deleteRange(currencyName, - pluginSettings.reloadBlockchainIndicesFrom(), - pluginSettings.reloadBlockchainIndicesTo()); - } - else { - if (logger.isWarnEnabled()) { - logger.warn(String.format("/!\\ Re-indexing blockchain from block #%s...", pluginSettings.reloadBlockchainIndicesFrom())); - } - - injector.getInstance(BlockchainService.class) - .deleteFrom(currencyName, pluginSettings.reloadBlockchainIndicesFrom()); - } - } - else { - if (logger.isInfoEnabled()) { - logger.info(String.format("[%s] Indexing blockchain...", currencyName)); - } - } - - - // Wait end of currency index creation, then index blocks - threadPool.scheduleOnClusterReady(() -> { - - // Reindex range - if (pluginSettings.reloadBlockchainIndices() - && pluginSettings.reloadBlockchainIndicesFrom() > 0 - && pluginSettings.reloadBlockchainIndicesTo() > pluginSettings.reloadBlockchainIndicesFrom()) { - injector.getInstance(BlockchainService.class) - .indexBlocksRange(peer, - pluginSettings.reloadBlockchainIndicesFrom(), - pluginSettings.reloadBlockchainIndicesTo()); - } - - try { - // Index blocks (and listen if new block appear) - injector.getInstance(BlockchainService.class) - .indexLastBlocks(peer) - .listenAndIndexNewBlock(peer); - - // Index peers (and listen if new peer appear) - injector.getInstance(PeerService.class) - .listenAndIndexPeers(peer); - - - // Start synchro - if (pluginSettings.enableSynchro()) { - injector.getInstance(SynchroService.class) - .startScheduling(); - } - - if (logger.isInfoEnabled()) { - logger.info(String.format("[%s] Indexing blockchain [OK]", currencyName)); - } - - } catch(Throwable e){ - logger.error(String.format("[%s] Indexing blockchain error: %s", currencyName, e.getMessage()), e); - throw e; - } - - }); - - } - - // If doc stats enable - if (pluginSettings.enableDocStats()) { - - // Add access to docstat index - injector.getInstance(RestSecurityController.class) - .allowIndexType(RestRequest.Method.GET, - DocStatDao.INDEX, - DocStatDao.TYPE) - .allowPostSearchIndexType( - DocStatDao.INDEX, - DocStatDao.TYPE); - - // Add index [currency/record] to stats - final DocStatService docStatService = injector - .getInstance(DocStatService.class) - .registerIndex(CurrencyService.INDEX, CurrencyService.RECORD_TYPE); - - // Wait end of currency index creation, then index blocks - threadPool.scheduleOnClusterReady(docStatService::startScheduling); - } - - // Allow scroll search - injector.getInstance(RestSecurityController.class) - .allow(RestRequest.Method.POST, "^/_search/scroll$"); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java deleted file mode 100644 index 775231f2890928e2bbd91b0e1dcdaf66876d7329..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java +++ /dev/null @@ -1,399 +0,0 @@ -package org.duniter.elasticsearch; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.ImmutableSet; -import org.apache.commons.io.FileUtils; -import org.duniter.core.client.config.Configuration; -import org.duniter.core.client.config.ConfigurationOption; -import org.duniter.core.client.config.ConfigurationProvider; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.i18n.I18nInitializer; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.nuiton.config.ApplicationConfig; -import org.nuiton.config.ApplicationConfigHelper; -import org.nuiton.config.ApplicationConfigProvider; -import org.nuiton.config.ArgumentsParserException; -import org.nuiton.i18n.I18n; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.List; -import java.util.Locale; -import java.util.Set; - -import static org.nuiton.i18n.I18n.t; - -/** - * Access to configuration options - * @author Benoit Lavenier <benoit.lavenier@e-is.pro> - * @since 1.0 - */ -public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { - - protected final Settings settings; - - private List<String> i18nBundleNames = new ArrayList<>(); // Default - - /** - * Delegate application config. - */ - protected final ApplicationConfig applicationConfig; - protected final org.duniter.core.client.config.Configuration clientConfig; - - @Inject - public PluginSettings(org.elasticsearch.common.settings.Settings settings) { - super(settings); - - this.settings = settings; - this.applicationConfig = new ApplicationConfig(); - - // Cascade the application config to the client module - clientConfig = new org.duniter.core.client.config.Configuration(this.applicationConfig); - Configuration.setInstance(clientConfig); - - // Set the default bundle name - addI18nBundleName(getI18nBundleName()); - } - - @Override - protected void doStart() { - - - // get all config providers - Set<ApplicationConfigProvider> providers = - ImmutableSet.of(new ConfigurationProvider()); - - // load all default options - ApplicationConfigHelper.loadAllDefaultOption(applicationConfig, - providers); - - // Ovverides defaults - String baseDir = settings.get("path.home"); - applicationConfig.setDefaultOption(ConfigurationOption.BASEDIR.getKey(), baseDir); - applicationConfig.setDefaultOption(ConfigurationOption.NODE_HOST.getKey(), getNodeBmaHost()); - applicationConfig.setDefaultOption(ConfigurationOption.NODE_PORT.getKey(), String.valueOf(getNodeBmaPort())); - applicationConfig.setDefaultOption(ConfigurationOption.NETWORK_TIMEOUT.getKey(), String.valueOf(getNetworkTimeout())); - applicationConfig.setDefaultOption(ConfigurationOption.NETWORK_MAX_CONNECTIONS.getKey(), String.valueOf(getNetworkMaxConnections())); - applicationConfig.setDefaultOption(ConfigurationOption.NETWORK_MAX_CONNECTIONS_PER_ROUTE.getKey(), String.valueOf(getNetworkMaxConnectionsPerRoute())); - - try { - applicationConfig.parse(new String[]{}); - - } catch (ArgumentsParserException e) { - throw new TechnicalException(t("duniter4j.config.parse.error"), e); - } - - File appBasedir = applicationConfig.getOptionAsFile( - ConfigurationOption.BASEDIR.getKey()); - - if (appBasedir == null) { - appBasedir = new File(""); - } - if (!appBasedir.isAbsolute()) { - appBasedir = new File(appBasedir.getAbsolutePath()); - } - if (appBasedir.getName().equals("..")) { - appBasedir = appBasedir.getParentFile().getParentFile(); - } - if (appBasedir.getName().equals(".")) { - appBasedir = appBasedir.getParentFile(); - } - applicationConfig.setOption( - ConfigurationOption.BASEDIR.getKey(), - appBasedir.getAbsolutePath()); - - // Init i18n - try { - initI18n(); - } - catch(IOException e) { - logger.error(String.format("Could not init i18n: %s", e.getMessage()), e); - } - - initVersion(applicationConfig); - } - - @Override - protected void doStop() { - - } - - @Override - protected void doClose() { - - } - - public Settings getSettings() { - return settings; - } - - public String getClusterName() { - return settings.get("cluster.name", "?"); - } - - public String getNodeBmaHost() { - return settings.get("duniter.host", "g1.duniter.org"); - } - - public int getNodeBmaPort() { - return settings.getAsInt("duniter.port", 10901); - } - - public boolean getNodeBmaUseSsl() { - return settings.getAsBoolean("duniter.useSsl", getNodeBmaPort() == 443); - } - - public boolean isIndexBulkEnable() { - return settings.getAsBoolean("duniter.bulk.enable", true); - } - - public int getIndexBulkSize() { - return settings.getAsInt("duniter.bulk.size", 1000); - } - - public int getNodeForkResyncWindow() { - return settings.getAsInt("duniter.fork.resync.window", 100); - } - - public String getDefaultStringAnalyzer() { - return settings.get("duniter.string.analyzer", "english"); - } - - public boolean reloadAllIndices() { - return settings.getAsBoolean("duniter.indices.reload", false); - } - - public boolean enableBlockchainSync() { - return settings.getAsBoolean("duniter.blockchain.enable", false); - } - - public boolean reloadBlockchainIndices() { - return settings.getAsBoolean("duniter.blockchain.reload", false); - } - - public int reloadBlockchainIndicesFrom() { - return settings.getAsInt("duniter.blockchain.reload.from", 0); - } - public int reloadBlockchainIndicesTo() { - return settings.getAsInt("duniter.blockchain.reload.to", -1); - } - - public File getTempDirectory() { - return Configuration.instance().getTempDirectory(); - } - - public int getNetworkTimeout() { - return settings.getAsInt("duniter.network.timeout", 30000 /*30s*/); - } - - public int getNetworkMaxConnections() { - return settings.getAsInt("duniter.network.maxConnections", 100); - } - - public int getNetworkMaxConnectionsPerRoute() { - return settings.getAsInt("duniter.network.maxConnectionsPerRoute", 5); - } - - public boolean enableSynchro() { - return settings.getAsBoolean("duniter.p2p.enable", true); - } - - public boolean enableSynchroWebsocket() { - return settings.getAsBoolean("duniter.p2p.ws.enable", true); - } - - public boolean fullResyncAtStartup() { - return settings.getAsBoolean("duniter.p2p.fullResyncAtStartup", false); - } - - public int getSynchroTimeOffset() { - return settings.getAsInt("duniter.p2p.peerTimeOffset", 60*60/*=1hour*/); - } - - public String[] getSynchroIncludesEndpoints() { - return settings.getAsArray("duniter.p2p.includes.endpoints"); - } - - public String[] getSynchroIncludesPubkeys() { - return settings.getAsArray("duniter.p2p.includes.pubkeys"); - } - - public boolean enableSynchroDiscovery() { - return settings.getAsBoolean("duniter.p2p.discovery.enable", true); - } - - public boolean isDevMode() { - return settings.getAsBoolean("duniter.dev.enable", false); - } - - public int getNodeRetryCount() { - return settings.getAsInt("duniter.retry.count", 5); - } - - public int getNodeRetryWaitDuration() { - return settings.getAsInt("duniter.retry.waitDuration", 5000); - } - - public String getShareBaseUrl() { - return settings.get("duniter.share.base.url"); - } - - public Peer checkAndGetPeer() { - if (StringUtils.isBlank(getNodeBmaHost())) { - logger.error("ERROR: node host is required"); - System.exit(-1); - return null; - } - if (getNodeBmaPort() <= 0) { - logger.error("ERROR: node port is required"); - System.exit(-1); - return null; - } - - Peer peer = Peer.newBuilder().setHost(getNodeBmaHost()).setPort(getNodeBmaPort()).setUseSsl(getNodeBmaUseSsl()).build(); - return peer; - } - - public String getKeyringSalt() { - return settings.get("duniter.keyring.salt"); - } - - public String getKeyringPassword() { - return settings.get("duniter.keyring.password"); - } - - public String getKeyringPublicKey() { - return settings.get("duniter.keyring.pub"); - } - - public String getKeyringSecretKey() { - return settings.get("duniter.keyring.sec"); - } - - public boolean enableSecurity() { - return settings.getAsBoolean("duniter.security.enable", true); - } - - public int getDocumentTimeMaxPastDelta() { - return settings.getAsInt("duniter.document.time.maxPastDelta", 7200); // in seconds = 2h - } - - public int getDocumentTimeMaxFutureDelta() { - return settings.getAsInt("duniter.document.time.maxFutureDelta", 600); // in seconds = 10min - } - - public boolean allowDocumentDeletionByAdmin() { - return settings.getAsBoolean("duniter.document.allowAdminDeletion", true); // - } - - public String getWebSocketHost() { - return settings.get("network.host", "localhost"); - } - - public String getWebSocketPort() { - return settings.get("duniter.ws.port", "9400"); - } - - public boolean getWebSocketEnable() { - return settings.getAsBoolean("duniter.ws.enable", Boolean.TRUE); - } - - public String[] getWebSocketChangesListenSource() { - return settings.getAsArray("duniter.ws.changes.listenSource", new String[]{"*"}); - } - - public boolean enableDocStats() { - return settings.getAsBoolean("duniter.stats.enable", true); - } - - /* protected methods */ - - protected void initI18n() throws IOException { - //if (I18n.getDefaultLocale() != null) return; // already init - - // --------------------------------------------------------------------// - // init i18n - // --------------------------------------------------------------------// - - File i18nDirectory = clientConfig.getI18nDirectory(); - if (i18nDirectory.exists()) { - // clean i18n cache - FileUtils.cleanDirectory(i18nDirectory); - } - - FileUtils.forceMkdir(i18nDirectory); - - if (logger.isDebugEnabled()) { - logger.debug("I18N directory: " + i18nDirectory); - } - - Locale i18nLocale = clientConfig.getI18nLocale(); - - if (logger.isInfoEnabled()) { - logger.info(String.format("Starts i18n with locale [%s] at [%s]", - i18nLocale, i18nDirectory)); - } - - I18n.init(new I18nInitializer(i18nDirectory, getI18nBundleNames()), - i18nLocale); - } - - protected String getI18nBundleName() { - return "duniter4j-es-core-i18n"; - } - - protected String[] getI18nBundleNames() { - return i18nBundleNames.toArray(new String[i18nBundleNames.size()]); - } - - public void addI18nBundleName(String i18nBundleName) { - if (!this.i18nBundleNames.contains(i18nBundleName)) { - this.i18nBundleNames.add(i18nBundleName); - } - } - - public Locale getI18nLocale() { - return clientConfig.getI18nLocale(); - } - - /** - * Override the version default option, from the MANIFEST implementation version (if any) - * @param applicationConfig - */ - protected void initVersion(ApplicationConfig applicationConfig) { - // Override application version - String implementationVersion = this.getClass().getPackage().getSpecificationVersion(); - if (implementationVersion != null) { - applicationConfig.setDefaultOption( - ConfigurationOption.VERSION.getKey(), - implementationVersion); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/beans/ESBeanFactory.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/beans/ESBeanFactory.java deleted file mode 100644 index 7bc9af4cc448d6dc131b5a1e8a055524f3e48a85..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/beans/ESBeanFactory.java +++ /dev/null @@ -1,64 +0,0 @@ -package org.duniter.elasticsearch.beans; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.beans.Bean; -import org.duniter.core.beans.BeanCreationException; -import org.duniter.core.beans.BeanFactory; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; - -/** - * Created by blavenie on 31/03/17. - */ -public class ESBeanFactory extends BeanFactory { - - private Injector injector = null; - - @Inject - public void setInjector(Injector injector) { - this.injector = injector; - } - - @Override - protected <S extends Bean> void initBean(S bean) { - super.initBean(bean); - if (injector != null) { - injector.injectMembers(bean); - } - } - - @Override - protected <S extends Bean> S newBean(Class<S> clazz) { - try { - return super.newBean(clazz); - } - catch(BeanCreationException e) { - // try using injector, if exists - if (injector != null) { - return injector.getBinding(clazz).getProvider().get(); - } - throw e; - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java deleted file mode 100644 index d7afec69051ad3a6d0eeaab5b12add4dc4df8c80..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java +++ /dev/null @@ -1,121 +0,0 @@ -package org.duniter.elasticsearch.client; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.beans.Bean; -import org.duniter.core.client.model.local.LocalEntity; -import org.duniter.elasticsearch.dao.handler.StringReaderHandler; -import org.duniter.elasticsearch.threadpool.CompletableActionFuture; -import org.elasticsearch.ElasticsearchException; -import org.elasticsearch.action.ActionRequest; -import org.elasticsearch.action.ActionRequestBuilder; -import org.elasticsearch.action.ActionResponse; -import org.elasticsearch.action.ListenableActionFuture; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.client.Client; -import org.elasticsearch.search.SearchHit; - -import java.io.File; -import java.io.InputStream; -import java.util.Map; -import java.util.Set; -import java.util.concurrent.CompletableFuture; -import java.util.concurrent.ScheduledThreadPoolExecutor; - -/** - * Created by blavenie on 03/04/17. - */ -public interface Duniter4jClient extends Bean, Client { - - boolean existsIndex(String index); - - void deleteIndexIfExists(String indexName); - - Object getFieldById(String index, String type, String docId, String fieldName); - - Map<String, Object> getFieldByIds(String index, String type, Set<String> ids, String fieldName); - - Map<String, Object> getFieldsById(String index, String type, String docId, String... fieldNames); - - <T> T getTypedFieldById(String index, String type, String docId, String fieldName); - - Map<String, Object> getMandatoryFieldsById(String index, String type, String docId, String... fieldNames); - - <T> T getMandatoryTypedFieldById(String index, String type, String docId, String fieldName); - - String indexDocumentFromJson(String index, String type, String json); - - void updateDocumentFromJson(String index, String type, String id, String json); - - void checkSameDocumentField(String index, String type, String id, String fieldName, String expectedvalue) throws ElasticsearchException; - - void checkSameDocumentIssuer(String index, String type, String id, String expectedIssuer); - - boolean isDocumentExists(String index, String type, String id) throws ElasticsearchException; - - void checkDocumentExists(String index, String type, String id) throws ElasticsearchException; - - /** - * Retrieve a document by id (safe mode) - * @param docId - * @return - */ - <T extends Object> T getSourceByIdOrNull(String index, String type, String docId, Class<T> classOfT, String... fieldNames); - - /** - * Retrieve a document by id - * @param docId - * @return - */ - <T extends Object> T getSourceById(String index, String type, String docId, Class<T> classOfT, String... fieldNames); - - <C extends LocalEntity<String>> C readSourceOrNull(SearchHit searchHit, Class<? extends C> clazz); - - void bulkFromClasspathFile(String classpathFile, String indexName, String indexType); - - void bulkFromClasspathFile(String classpathFile, String indexName, String indexType, StringReaderHandler handler); - - void bulkFromFile(File file, String indexName, String indexType); - - void bulkFromFile(File file, String indexName, String indexType, StringReaderHandler handler); - - void bulkFromStream(InputStream is, String indexName, String indexType); - - void bulkFromStream(InputStream is, String indexName, String indexType, StringReaderHandler handler); - - void flushDeleteBulk(final String index, final String type, BulkRequestBuilder bulkRequest); - - void flushBulk(BulkRequestBuilder bulkRequest); - - BulkRequestBuilder bulkDeleteFromSearch(String index, - String type, - SearchRequestBuilder searchRequest, - BulkRequestBuilder bulkRequest, - int bulkSize, - boolean flushAll); - - void safeExecuteRequest(ActionRequestBuilder<?, ?, ?> request, boolean wait); - - ScheduledThreadPoolExecutor scheduler(); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java deleted file mode 100644 index 5bed8bd66e393df212855b15230a0a25a7ef6e7a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java +++ /dev/null @@ -1,1098 +0,0 @@ -package org.duniter.elasticsearch.client; - -/* - * #%L - * Duniter4j :: 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.databind.ObjectMapper; -import com.google.common.base.Joiner; -import com.google.common.collect.Lists; -import org.apache.commons.collections4.MapUtils; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.local.LocalEntity; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.ObjectUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.dao.handler.StringReaderHandler; -import org.duniter.elasticsearch.exception.AccessDeniedException; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.elasticsearch.ElasticsearchException; -import org.elasticsearch.action.*; -import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequestBuilder; -import org.elasticsearch.action.admin.indices.exists.indices.IndicesExistsRequestBuilder; -import org.elasticsearch.action.admin.indices.exists.indices.IndicesExistsResponse; -import org.elasticsearch.action.bulk.BulkItemResponse; -import org.elasticsearch.action.bulk.BulkRequest; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.bulk.BulkResponse; -import org.elasticsearch.action.count.CountRequest; -import org.elasticsearch.action.count.CountRequestBuilder; -import org.elasticsearch.action.count.CountResponse; -import org.elasticsearch.action.delete.DeleteRequest; -import org.elasticsearch.action.delete.DeleteRequestBuilder; -import org.elasticsearch.action.delete.DeleteResponse; -import org.elasticsearch.action.exists.ExistsRequest; -import org.elasticsearch.action.exists.ExistsRequestBuilder; -import org.elasticsearch.action.exists.ExistsResponse; -import org.elasticsearch.action.explain.ExplainRequest; -import org.elasticsearch.action.explain.ExplainRequestBuilder; -import org.elasticsearch.action.explain.ExplainResponse; -import org.elasticsearch.action.fieldstats.FieldStatsRequest; -import org.elasticsearch.action.fieldstats.FieldStatsRequestBuilder; -import org.elasticsearch.action.fieldstats.FieldStatsResponse; -import org.elasticsearch.action.get.*; -import org.elasticsearch.action.index.IndexRequest; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.index.IndexResponse; -import org.elasticsearch.action.indexedscripts.delete.DeleteIndexedScriptRequest; -import org.elasticsearch.action.indexedscripts.delete.DeleteIndexedScriptRequestBuilder; -import org.elasticsearch.action.indexedscripts.delete.DeleteIndexedScriptResponse; -import org.elasticsearch.action.indexedscripts.get.GetIndexedScriptRequest; -import org.elasticsearch.action.indexedscripts.get.GetIndexedScriptRequestBuilder; -import org.elasticsearch.action.indexedscripts.get.GetIndexedScriptResponse; -import org.elasticsearch.action.indexedscripts.put.PutIndexedScriptRequest; -import org.elasticsearch.action.indexedscripts.put.PutIndexedScriptRequestBuilder; -import org.elasticsearch.action.indexedscripts.put.PutIndexedScriptResponse; -import org.elasticsearch.action.percolate.*; -import org.elasticsearch.action.search.*; -import org.elasticsearch.action.suggest.SuggestRequest; -import org.elasticsearch.action.suggest.SuggestRequestBuilder; -import org.elasticsearch.action.suggest.SuggestResponse; -import org.elasticsearch.action.termvectors.*; -import org.elasticsearch.action.update.UpdateRequest; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.action.update.UpdateResponse; -import org.elasticsearch.client.AdminClient; -import org.elasticsearch.client.Client; -import org.elasticsearch.client.Requests; -import org.elasticsearch.client.support.Headers; -import org.elasticsearch.common.Nullable; -import org.elasticsearch.common.bytes.BytesArray; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.common.util.concurrent.EsRejectedExecutionException; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.SearchHit; -import org.elasticsearch.search.SearchHitField; -import org.elasticsearch.search.SearchHits; -import org.elasticsearch.threadpool.ThreadPool; - -import java.io.*; -import java.util.*; -import java.util.concurrent.ScheduledThreadPoolExecutor; - -/** - * Created by Benoit on 08/04/2015. - */ -public class Duniter4jClientImpl implements Duniter4jClient { - - private final ESLogger logger; - - private final Client client; - private final org.duniter.elasticsearch.threadpool.ThreadPool threadPool; - - @Inject - public Duniter4jClientImpl(Client client, Settings settings, org.duniter.elasticsearch.threadpool.ThreadPool threadPool) { - super(); - this.logger = Loggers.getLogger("duniter.client", settings, new String[0]); - this.client = client; - this.threadPool = threadPool; - } - - @Override - public boolean existsIndex(String indexes) { - IndicesExistsRequestBuilder requestBuilder = client.admin().indices().prepareExists(indexes); - IndicesExistsResponse response = requestBuilder.execute().actionGet(); - return response.isExists(); - } - - @Override - public void deleteIndexIfExists(String indexName){ - if (!existsIndex(indexName)) { - return; - } - if (logger.isInfoEnabled()) { - logger.info(String.format("Deleting index [%s]", indexName)); - } - - DeleteIndexRequestBuilder deleteIndexRequestBuilder = client.admin().indices().prepareDelete(indexName); - deleteIndexRequestBuilder.execute().actionGet(); - } - - @Override - public String indexDocumentFromJson(String index, String type, String json) { - IndexResponse response = client.prepareIndex(index, type) - .setSource(json) - .setRefresh(true) - .execute().actionGet(); - return response.getId(); - } - - @Override - public void updateDocumentFromJson(String index, String type, String id, String json) { - // Execute indexBlocksFromNode - safeExecuteRequest(client.prepareUpdate(index, type, id) - .setRefresh(true) - .setDoc(json), true); - } - - @Override - public void checkSameDocumentField(String index, String type, String id, String fieldName, String expectedvalue) throws ElasticsearchException { - - GetResponse response = client.prepareGet(index, type, id) - .setFields(fieldName) - .execute().actionGet(); - boolean failed = !response.isExists(); - if (failed) { - throw new NotFoundException(String.format("Document [%s/%s/%s] not exists.", index, type, id)); - } else { - String docValue = (String)response.getFields().get(fieldName).getValue(); - if (!Objects.equals(expectedvalue, docValue)) { - throw new AccessDeniedException(String.format("Could not delete this document: not same [%s].", fieldName)); - } - } - } - - @Override - public boolean isDocumentExists(String index, String type, String id) throws ElasticsearchException { - GetResponse response = client.prepareGet(index, type, id) - .setFetchSource(false) - .execute().actionGet(); - return response.isExists(); - } - - @Override - public void checkDocumentExists(String index, String type, String id) throws ElasticsearchException { - if (!isDocumentExists(index, type, id)) { - throw new NotFoundException(String.format("Document [%s/%s/%s] not exists.", index, type, id)); - } - } - - - @Override - public void checkSameDocumentIssuer(String index, String type, String id, String expectedIssuer) { - String issuer = getMandatoryFieldsById(index, type, id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); - if (!ObjectUtils.equals(expectedIssuer, issuer)) { - throw new AccessDeniedException("Not same issuer"); - } - } - - /** - * Retrieve some field from a document id, and check if all field not null - * @param index - * @param type - * @param docId - * @param fieldNames - * @return - */ - @Override - public Map<String, Object> getMandatoryFieldsById(String index, String type, String docId, String... fieldNames) { - 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 mandatory field [%s].", index, type, docId, fieldName)); - }); - return fields; - } - - /** - * Retrieve some field from a document id - * @param docId - * @return - */ - @Override - public Map<String, Object> getFieldsById(String index, String type, String docId, String... fieldNames) { - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(index) - .setTypes(type) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH); - - searchRequest.setQuery(QueryBuilders.idsQuery().ids(docId)); - searchRequest.addFields(fieldNames); - - // Execute query - try { - SearchResponse response = searchRequest.execute().actionGet(); - - if (response.getHits().getTotalHits() == 0) return null; - - Map<String, Object> result = new HashMap<>(); - // Read query result - SearchHit[] searchHits = response.getHits().getHits(); - for (SearchHit searchHit : searchHits) { - Map<String, SearchHitField> hitFields = searchHit.getFields(); - for(String fieldName: hitFields.keySet()) { - result.put(fieldName, hitFields.get(fieldName).getValue()); - } - break; - } - return result; - } - catch(SearchPhaseExecutionException e) { - // Failed or no item on index - throw new TechnicalException(String.format("[%s/%s] Unable to retrieve fields [%s] for id [%s]", - index, type, - Joiner.on(',').join(fieldNames).toString(), - docId), e); - } - } - - /** - * Retrieve some field from a document id - * @param index - * @param type - * @param ids - * @param fieldName - * @return - */ - @Override - public Map<String, Object> getFieldByIds(String index, String type, Set<String> ids, String fieldName) { - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(index) - .setTypes(type) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH); - - searchRequest.setQuery(QueryBuilders.idsQuery().ids(ids)); - searchRequest.addFields(fieldName); - - // Execute query - try { - SearchResponse response = searchRequest.execute().actionGet(); - - Map<String, Object> result = new HashMap<>(); - // Read query result - SearchHit[] searchHits = response.getHits().getHits(); - for (SearchHit searchHit : searchHits) { - Map<String, SearchHitField> hitFields = searchHit.getFields(); - if (hitFields.get(fieldName) != null) { - result.put(searchHit.getId(), hitFields.get(fieldName).getValue()); - } - } - return result; - } - catch(SearchPhaseExecutionException e) { - // Failed or no item on index - throw new TechnicalException(String.format("[%s/%s] Unable to retrieve field [%s] for ids [%s]", - index, type, fieldName, - Joiner.on(',').join(ids).toString()), e); - } - } - - /** - * Retrieve a field from a document id - * @param docId - * @return - */ - @Override - public Object getFieldById(String index, String type, String docId, String fieldName) { - - Map<String, Object> result = getFieldsById(index, type, docId, fieldName); - if (MapUtils.isEmpty(result)) { - return null; - } - return result.get(fieldName); - } - - @Override - public <T> T getTypedFieldById(String index, String type, String docId, String fieldName) { - return (T)getFieldById(index, type, docId, fieldName); - } - - @Override - public <T> T getMandatoryTypedFieldById(String index, String type, String docId, String fieldName) { - Object result = getFieldById(index, type, docId, fieldName); - if (result == null) { - throw new NotFoundException(String.format("Document [%s/%s/%s] missing value for mandatory field [%s].", index, type, docId, fieldName)); - } - return (T)result; - } - - /** - * Retrieve a document by id (safe mode) - * @param docId - * @return - */ - @Override - public <T extends Object> T getSourceByIdOrNull(String index, String type, String docId, Class<T> classOfT, String... fieldNames) { - try { - return getSourceById(index, type, docId, classOfT, fieldNames); - } - catch(TechnicalException e) { - return null; // not found - } - } - - /** - * Retrieve a document by id - * @param docId - * @return - */ - @Override - public <T extends Object> T getSourceById(String index, String type, String docId, Class<T> classOfT, String... fieldNames) { - - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(index) - .setSearchType(SearchType.QUERY_AND_FETCH); - - searchRequest.setQuery(QueryBuilders.idsQuery(type).ids(docId)); - if (CollectionUtils.isNotEmpty(fieldNames)) { - searchRequest.setFetchSource(fieldNames, null); - } - else { - searchRequest.setFetchSource(true); // full source - } - - // Execute query - try { - SearchResponse response = searchRequest.execute().actionGet(); - - if (response.getHits().getTotalHits() == 0) return null; - - // Read query result - SearchHit[] searchHits = response.getHits().getHits(); - ObjectMapper objectMapper = JacksonUtils.getThreadObjectMapper(); - - for (SearchHit searchHit : searchHits) { - if (searchHit.source() != null) { - return objectMapper.readValue(searchHit.source(), classOfT); - } - break; - } - return null; - } - catch(SearchPhaseExecutionException | IOException e) { - // Failed to get source - throw new TechnicalException(String.format("[%s/%s] Error while getting [%s]", - index, type, - docId), e); - } - } - - @Override - public <C extends LocalEntity<String>> C readSourceOrNull(SearchHit searchHit, Class<? extends C> clazz) { - try { - C value = JacksonUtils.getThreadObjectMapper().readValue(searchHit.getSourceRef().streamInput(), clazz); - value.setId(searchHit.getId()); - return value; - } - catch(IOException e) { - logger.warn(String.format("Unable to deserialize source [%s/%s/%s] into [%s]: %s", searchHit.getIndex(), searchHit.getType(), searchHit.getId(), clazz.getName(), e.getMessage())); - return null; - } - } - - - @Override - public void bulkFromClasspathFile(String classpathFile, String indexName, String indexType) { - bulkFromClasspathFile(classpathFile, indexName, indexType, null); - } - - @Override - public void bulkFromClasspathFile(String classpathFile, String indexName, String indexType, StringReaderHandler handler) { - InputStream is = null; - try { - is = getClass().getClassLoader().getResourceAsStream(classpathFile); - if (is == null) { - throw new TechnicalException(String.format("Could not retrieve data file [%s] need to fill index [%s]: ", classpathFile, indexName)); - } - - bulkFromStream(is, indexName, indexType, handler); - } - finally { - if (is != null) { - try { - is.close(); - } - catch(IOException e) { - // Silent is gold - } - } - } - } - - @Override - public void bulkFromFile(File file, String indexName, String indexType) { - bulkFromFile(file, indexName, indexType, null); - } - - @Override - public void bulkFromFile(File file, String indexName, String indexType, StringReaderHandler handler) { - Preconditions.checkNotNull(file); - Preconditions.checkArgument(file.exists()); - - InputStream is = null; - try { - is = new BufferedInputStream(new FileInputStream(file)); - bulkFromStream(is, indexName, indexType, handler); - } - catch(FileNotFoundException e) { - throw new TechnicalException(String.format("[%s] Could not find file %s", indexName, file.getPath()), e); - } - finally { - if (is != null) { - try { - is.close(); - } - catch(IOException e) { - // Silent is gold - } - } - } - } - - @Override - public void bulkFromStream(InputStream is, String indexName, String indexType) { - bulkFromStream(is, indexName, indexType, null); - } - - @Override - public void bulkFromStream(InputStream is, String indexName, String indexType, StringReaderHandler handler) { - Preconditions.checkNotNull(is); - BulkRequest bulkRequest = Requests.bulkRequest(); - - BufferedReader br = null; - - try { - br = new BufferedReader(new InputStreamReader(is)); - - String line = br.readLine(); - StringBuilder builder = new StringBuilder(); - while(line != null) { - line = line.trim(); - if (StringUtils.isNotBlank(line)) { - if (logger.isTraceEnabled()) { - logger.trace(String.format("[%s] Add to bulk: %s", indexName, line)); - } - if (handler != null) { - line = handler.onReadLine(line.trim()); - } - builder.append(line).append('\n'); - } - line = br.readLine(); - } - - byte[] data = builder.toString().getBytes(); - bulkRequest.add(new BytesArray(data), indexName, indexType, false); - - } catch(Exception e) { - throw new TechnicalException(String.format("[%s] Error while inserting rows into %s", indexName, indexType), e); - } - finally { - if (br != null) { - try { - br.close(); - } - catch(IOException e) { - // Silent is gold - } - } - } - - try { - client.bulk(bulkRequest).actionGet(); - } catch(Exception e) { - throw new TechnicalException(String.format("[%s] Error while inserting rows into %s", indexName, indexType), e); - } - } - - @Override - public void flushDeleteBulk(final String index, final String type, final BulkRequestBuilder bulkRequest) { - if (bulkRequest.numberOfActions() > 0) { - - BulkResponse bulkResponse = bulkRequest.execute().actionGet(); - // If failures, continue but save missing blocks - if (bulkResponse.hasFailures()) { - // process failures by iterating through each bulk response item - for (BulkItemResponse itemResponse : bulkResponse) { - boolean skip = !itemResponse.isFailed(); - if (!skip) { - logger.debug(String.format("[%s/%s] Error while deleting doc [%s]: %s. Skipping this deletion.", index, type, itemResponse.getId(), itemResponse.getFailureMessage())); - } - } - } - } - } - - @Override - public void flushBulk(final BulkRequestBuilder bulkRequest) { - if (bulkRequest.numberOfActions() > 0) { - - // Flush the bulk if not empty - BulkResponse bulkResponse = bulkRequest.get(); - - Set<String> missingDocIds = new LinkedHashSet<>(); - - // If failures, continue but save missing blocks - if (bulkResponse.hasFailures()) { - // process failures by iterating through each bulk response item - for (BulkItemResponse itemResponse : bulkResponse) { - boolean skip = !itemResponse.isFailed() - || missingDocIds.contains(itemResponse.getId()); - if (!skip) { - logger.error(String.format("[%s/%s] could not process _id=%s: %s. Skipping.", - itemResponse.getIndex(), itemResponse.getType(), itemResponse.getId(), itemResponse.getFailureMessage())); - missingDocIds.add(itemResponse.getId()); - } - } - } - } - } - - @Override - public BulkRequestBuilder bulkDeleteFromSearch(final String index, - final String type, - final SearchRequestBuilder searchRequest, - BulkRequestBuilder bulkRequest, - final int bulkSize, - final boolean flushAll) { - - // Execute query, while there is some data - try { - - int counter = 0; - boolean loop = true; - searchRequest.setSize(bulkSize); - SearchResponse response = searchRequest.execute().actionGet(); - - // Execute query, while there is some data - do { - - // Read response - SearchHit[] searchHits = response.getHits().getHits(); - for (SearchHit searchHit : searchHits) { - - // Add deletion to bulk - bulkRequest.add( - client.prepareDelete(index, type, searchHit.getId()) - ); - counter++; - - // Flush the bulk if not empty - if ((bulkRequest.numberOfActions() % bulkSize) == 0) { - flushDeleteBulk(index, type, bulkRequest); - bulkRequest = client.prepareBulk(); - } - } - - // Prepare next iteration - if (counter == 0 || counter >= response.getHits().getTotalHits()) { - loop = false; - } - // Prepare next iteration - else { - searchRequest.setFrom(counter); - response = searchRequest.execute().actionGet(); - } - } while(loop); - - // last flush - if (flushAll && (bulkRequest.numberOfActions() % bulkSize) != 0) { - flushDeleteBulk(index, type, bulkRequest); - } - - } catch (SearchPhaseExecutionException e) { - // Failed or no item on index - logger.error(String.format("Error while deleting by reference: %s. Skipping deletions.", e.getMessage()), e); - } - - return bulkRequest; - } - - /* delegate methods */ - - @Override - public AdminClient admin() { - return client.admin(); - } - - @Override - public ActionFuture<IndexResponse> index(IndexRequest request) { - return client.index(request); - } - - @Override - public void index(IndexRequest request, ActionListener<IndexResponse> listener) { - client.index(request, listener); - } - - @Override - public IndexRequestBuilder prepareIndex() { - return client.prepareIndex(); - } - - @Override - public ActionFuture<UpdateResponse> update(UpdateRequest request) { - return client.update(request); - } - - @Override - public void update(UpdateRequest request, ActionListener<UpdateResponse> listener) { - client.update(request, listener); - } - - @Override - public UpdateRequestBuilder prepareUpdate() { - return client.prepareUpdate(); - } - - @Override - public UpdateRequestBuilder prepareUpdate(String index, String type, String id) { - return client.prepareUpdate(index, type, id); - } - - @Override - public IndexRequestBuilder prepareIndex(String index, String type) { - return client.prepareIndex(index, type); - } - - @Override - public IndexRequestBuilder prepareIndex(String index, String type, @Nullable String id) { - return client.prepareIndex(index, type, id); - } - - @Override - public ActionFuture<DeleteResponse> delete(DeleteRequest request) { - return client.delete(request); - } - - @Override - public void delete(DeleteRequest request, ActionListener<DeleteResponse> listener) { - client.delete(request, listener); - } - - @Override - public DeleteRequestBuilder prepareDelete() { - return client.prepareDelete(); - } - - @Override - public DeleteRequestBuilder prepareDelete(String index, String type, String id) { - return client.prepareDelete(index, type, id); - } - - @Override - public ActionFuture<BulkResponse> bulk(BulkRequest request) { - return client.bulk(request); - } - - @Override - public void bulk(BulkRequest request, ActionListener<BulkResponse> listener) { - client.bulk(request, listener); - } - - @Override - public BulkRequestBuilder prepareBulk() { - return client.prepareBulk(); - } - - @Override - public ActionFuture<GetResponse> get(GetRequest request) { - return client.get(request); - } - - @Override - public void get(GetRequest request, ActionListener<GetResponse> listener) { - client.get(request, listener); - } - - @Override - public GetRequestBuilder prepareGet() { - return client.prepareGet(); - } - - @Override - public GetRequestBuilder prepareGet(String index, @Nullable String type, String id) { - return client.prepareGet(index, type, id); - } - - @Override - public PutIndexedScriptRequestBuilder preparePutIndexedScript() { - return client.preparePutIndexedScript(); - } - - @Override - public PutIndexedScriptRequestBuilder preparePutIndexedScript(@Nullable String scriptLang, String id, String source) { - return client.preparePutIndexedScript(scriptLang, id, source); - } - - @Override - public void deleteIndexedScript(DeleteIndexedScriptRequest request, ActionListener<DeleteIndexedScriptResponse> listener) { - client.deleteIndexedScript(request, listener); - } - - @Override - public ActionFuture<DeleteIndexedScriptResponse> deleteIndexedScript(DeleteIndexedScriptRequest request) { - return client.deleteIndexedScript(request); - } - - @Override - public DeleteIndexedScriptRequestBuilder prepareDeleteIndexedScript() { - return client.prepareDeleteIndexedScript(); - } - - @Override - public DeleteIndexedScriptRequestBuilder prepareDeleteIndexedScript(@Nullable String scriptLang, String id) { - return client.prepareDeleteIndexedScript(scriptLang, id); - } - - @Override - public void putIndexedScript(PutIndexedScriptRequest request, ActionListener<PutIndexedScriptResponse> listener) { - client.putIndexedScript(request, listener); - } - - @Override - public ActionFuture<PutIndexedScriptResponse> putIndexedScript(PutIndexedScriptRequest request) { - return client.putIndexedScript(request); - } - - @Override - public GetIndexedScriptRequestBuilder prepareGetIndexedScript() { - return client.prepareGetIndexedScript(); - } - - @Override - public GetIndexedScriptRequestBuilder prepareGetIndexedScript(@Nullable String scriptLang, String id) { - return client.prepareGetIndexedScript(scriptLang, id); - } - - @Override - public void getIndexedScript(GetIndexedScriptRequest request, ActionListener<GetIndexedScriptResponse> listener) { - client.getIndexedScript(request, listener); - } - - @Override - public ActionFuture<GetIndexedScriptResponse> getIndexedScript(GetIndexedScriptRequest request) { - return client.getIndexedScript(request); - } - - @Override - public ActionFuture<MultiGetResponse> multiGet(MultiGetRequest request) { - return client.multiGet(request); - } - - @Override - public void multiGet(MultiGetRequest request, ActionListener<MultiGetResponse> listener) { - client.multiGet(request, listener); - } - - @Override - public MultiGetRequestBuilder prepareMultiGet() { - return client.prepareMultiGet(); - } - - @Override - @Deprecated - public ActionFuture<CountResponse> count(CountRequest request) { - return client.count(request); - } - - @Override - @Deprecated - public void count(CountRequest request, ActionListener<CountResponse> listener) { - client.count(request, listener); - } - - @Override - @Deprecated - public CountRequestBuilder prepareCount(String... indices) { - return client.prepareCount(indices); - } - - @Override - @Deprecated - public ActionFuture<ExistsResponse> exists(ExistsRequest request) { - return client.exists(request); - } - - @Override - @Deprecated - public void exists(ExistsRequest request, ActionListener<ExistsResponse> listener) { - client.exists(request, listener); - } - - @Override - @Deprecated - public ExistsRequestBuilder prepareExists(String... indices) { - return client.prepareExists(indices); - } - - @Override - public ActionFuture<SuggestResponse> suggest(SuggestRequest request) { - return client.suggest(request); - } - - @Override - public void suggest(SuggestRequest request, ActionListener<SuggestResponse> listener) { - client.suggest(request, listener); - } - - @Override - public SuggestRequestBuilder prepareSuggest(String... indices) { - return client.prepareSuggest(indices); - } - - @Override - public ActionFuture<SearchResponse> search(SearchRequest request) { - return client.search(request); - } - - @Override - public void search(SearchRequest request, ActionListener<SearchResponse> listener) { - client.search(request, listener); - } - - @Override - public SearchRequestBuilder prepareSearch(String... indices) { - return client.prepareSearch(indices); - } - - @Override - public ActionFuture<SearchResponse> searchScroll(SearchScrollRequest request) { - return client.searchScroll(request); - } - - @Override - public void searchScroll(SearchScrollRequest request, ActionListener<SearchResponse> listener) { - client.searchScroll(request, listener); - } - - @Override - public SearchScrollRequestBuilder prepareSearchScroll(String scrollId) { - return client.prepareSearchScroll(scrollId); - } - - @Override - public ActionFuture<MultiSearchResponse> multiSearch(MultiSearchRequest request) { - return client.multiSearch(request); - } - - @Override - public void multiSearch(MultiSearchRequest request, ActionListener<MultiSearchResponse> listener) { - client.multiSearch(request, listener); - } - - @Override - public MultiSearchRequestBuilder prepareMultiSearch() { - return client.prepareMultiSearch(); - } - - @Override - public ActionFuture<TermVectorsResponse> termVectors(TermVectorsRequest request) { - return client.termVectors(request); - } - - @Override - public void termVectors(TermVectorsRequest request, ActionListener<TermVectorsResponse> listener) { - client.termVectors(request, listener); - } - - @Override - public TermVectorsRequestBuilder prepareTermVectors() { - return client.prepareTermVectors(); - } - - @Override - public TermVectorsRequestBuilder prepareTermVectors(String index, String type, String id) { - return client.prepareTermVectors(index, type, id); - } - - @Override - @Deprecated - public ActionFuture<TermVectorsResponse> termVector(TermVectorsRequest request) { - return client.termVector(request); - } - - @Override - @Deprecated - public void termVector(TermVectorsRequest request, ActionListener<TermVectorsResponse> listener) { - client.termVector(request, listener); - } - - @Override - @Deprecated - public TermVectorsRequestBuilder prepareTermVector() { - return client.prepareTermVector(); - } - - @Override - @Deprecated - public TermVectorsRequestBuilder prepareTermVector(String index, String type, String id) { - return client.prepareTermVector(index, type, id); - } - - @Override - public ActionFuture<MultiTermVectorsResponse> multiTermVectors(MultiTermVectorsRequest request) { - return client.multiTermVectors(request); - } - - @Override - public void multiTermVectors(MultiTermVectorsRequest request, ActionListener<MultiTermVectorsResponse> listener) { - client.multiTermVectors(request, listener); - } - - @Override - public MultiTermVectorsRequestBuilder prepareMultiTermVectors() { - return client.prepareMultiTermVectors(); - } - - @Override - public ActionFuture<PercolateResponse> percolate(PercolateRequest request) { - return client.percolate(request); - } - - @Override - public void percolate(PercolateRequest request, ActionListener<PercolateResponse> listener) { - client.percolate(request, listener); - } - - @Override - public PercolateRequestBuilder preparePercolate() { - return client.preparePercolate(); - } - - @Override - public ActionFuture<MultiPercolateResponse> multiPercolate(MultiPercolateRequest request) { - return client.multiPercolate(request); - } - - @Override - public void multiPercolate(MultiPercolateRequest request, ActionListener<MultiPercolateResponse> listener) { - client.multiPercolate(request, listener); - } - - @Override - public MultiPercolateRequestBuilder prepareMultiPercolate() { - return client.prepareMultiPercolate(); - } - - @Override - public ExplainRequestBuilder prepareExplain(String index, String type, String id) { - return client.prepareExplain(index, type, id); - } - - @Override - public ActionFuture<ExplainResponse> explain(ExplainRequest request) { - return client.explain(request); - } - - @Override - public void explain(ExplainRequest request, ActionListener<ExplainResponse> listener) { - client.explain(request, listener); - } - - @Override - public ClearScrollRequestBuilder prepareClearScroll() { - return client.prepareClearScroll(); - } - - @Override - public ActionFuture<ClearScrollResponse> clearScroll(ClearScrollRequest request) { - return client.clearScroll(request); - } - - @Override - public void clearScroll(ClearScrollRequest request, ActionListener<ClearScrollResponse> listener) { - client.clearScroll(request, listener); - } - - @Override - public FieldStatsRequestBuilder prepareFieldStats() { - return client.prepareFieldStats(); - } - - @Override - public ActionFuture<FieldStatsResponse> fieldStats(FieldStatsRequest request) { - return client.fieldStats(request); - } - - @Override - public void fieldStats(FieldStatsRequest request, ActionListener<FieldStatsResponse> listener) { - client.fieldStats(request, listener); - } - - @Override - public Settings settings() { - return client.settings(); - } - - @Override - public Headers headers() { - return client.headers(); - } - - public <Request extends ActionRequest, Response extends ActionResponse, RequestBuilder extends ActionRequestBuilder<Request, Response, RequestBuilder>> ActionFuture<Response> execute(Action<Request, Response, RequestBuilder> action, Request request) { - return client.execute(action, request); - } - - public <Request extends ActionRequest, Response extends ActionResponse, RequestBuilder extends ActionRequestBuilder<Request, Response, RequestBuilder>> void execute(Action<Request, Response, RequestBuilder> action, Request request, ActionListener<Response> listener) { - client.execute(action, request, listener); - } - - public <Request extends ActionRequest, Response extends ActionResponse, RequestBuilder extends ActionRequestBuilder<Request, Response, RequestBuilder>> RequestBuilder prepareExecute(Action<Request, Response, RequestBuilder> action) { - return client.prepareExecute(action); - } - - public ThreadPool threadPool() { - return client.threadPool(); - } - - public ScheduledThreadPoolExecutor scheduler() { - return (ScheduledThreadPoolExecutor)client.threadPool().scheduler(); - } - - public void close() { - client.close(); - } - - public void safeExecuteRequest(ActionRequestBuilder<?, ?, ?> request, boolean wait) { - // Execute in a pool - if (!wait) { - boolean acceptedInPool = false; - while(!acceptedInPool) - try { - request.execute(); - acceptedInPool = true; - } - catch(EsRejectedExecutionException e) { - // not accepted, so wait - try { - Thread.sleep(1000); // 1s - } - catch(InterruptedException e2) { - // silent - } - } - - } else { - request.execute().actionGet(); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java deleted file mode 100644 index 21f248515a7f601c4ef3cb35010ae54e1623fbfd..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractDao.java +++ /dev/null @@ -1,113 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/* - * #%L - * Duniter4j :: 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.databind.ObjectMapper; -import com.google.common.collect.Lists; -import org.duniter.core.beans.Bean; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.local.LocalEntity; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.search.SearchHit; - -import java.io.IOException; -import java.util.List; - -/** - * Created by Benoit on 08/04/2015. - */ -public abstract class AbstractDao implements Bean { - - - protected final String loggerName; - protected ESLogger logger; - - protected Duniter4jClient client; - protected CryptoService cryptoService; - protected PluginSettings pluginSettings; - - public AbstractDao(String loggerName) { - super(); - this.loggerName = loggerName; - } - - @Inject - public void setClient(Duniter4jClient client) { - this.client = client; - } - - @Inject - public void setCryptoService(CryptoService cryptoService) { - this.cryptoService = cryptoService; - } - - @Inject - public void setPluginSettings(PluginSettings pluginSettings) { - this.pluginSettings = pluginSettings; - this.logger = Loggers.getLogger(loggerName, pluginSettings.getSettings(), new String[0]); - } - - /* -- protected methods -- */ - - protected ObjectMapper getObjectMapper() { - return JacksonUtils.getThreadObjectMapper(); - } - - protected <C extends LocalEntity<String>> List<C> toList(SearchResponse response, Class<? extends C> clazz) { - ObjectMapper objectMapper = getObjectMapper(); - - if (response.getHits() == null || response.getHits().getTotalHits() == 0) return null; - - List<C> result = Lists.newArrayList(); - for (SearchHit hit: response.getHits().getHits()) { - - try { - C value = objectMapper.readValue(hit.getSourceRef().streamInput(), clazz); - value.setId(hit.getId()); - result.add(value); - } - catch(IOException e) { - logger.warn(String.format("Unable to deserialize source [%s/%s/%s] into [%s]: %s", hit.getIndex(), hit.getType(), hit.getId(), clazz.getName(), e.getMessage())); - } - } - return result; - } - - protected List<String> toListIds(SearchResponse response) { - if (response.getHits() == null || response.getHits().getTotalHits() == 0) return null; - - List<String> result = Lists.newArrayList(); - for (SearchHit hit: response.getHits().getHits()) { - result.add(hit.getId()); - } - return result; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexDao.java deleted file mode 100644 index f891cde09a2bb83c8a0696f249ea415c28c54ced..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexDao.java +++ /dev/null @@ -1,91 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.exception.TechnicalException; -import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequestBuilder; - -/** - * Created by Benoit on 08/04/2015. - */ -public abstract class AbstractIndexDao<T extends IndexDao> extends AbstractDao implements IndexDao<T> { - - private final String index; - - public AbstractIndexDao(String index) { - super("duniter.dao."+index); - this.index = index; - } - - /** - * Create index - * @throws JsonProcessingException - */ - protected abstract void createIndex() throws JsonProcessingException; - - @Override - public String getIndex() { - return index; - } - - @Override - public T createIndexIfNotExists() { - try { - if (!client.existsIndex(index)) { - createIndex(); - } - } - catch(JsonProcessingException e) { - throw new TechnicalException(String.format("Error while creating index [%s]", index)); - } - return (T)this; - } - - @Override - public T deleteIndex() { - client.deleteIndexIfExists(index); - return (T)this; - } - - @Override - public boolean existsIndex() { - return client.existsIndex(index); - } - - - /* -- protected methods -- */ - - protected void deleteIndexIfExists(){ - if (!client.existsIndex(index)) { - return; - } - if (logger.isInfoEnabled()) { - logger.info(String.format("Deleting index [%s]", index)); - } - - DeleteIndexRequestBuilder deleteIndexRequestBuilder = client.admin().indices().prepareDelete(index); - deleteIndexRequestBuilder.execute().actionGet(); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java deleted file mode 100644 index 1963a91d604c86fb13254bfc6e8832f7dfd4cb6c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/AbstractIndexTypeDao.java +++ /dev/null @@ -1,197 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.elasticsearch.dao.handler.StringReaderHandler; -import org.elasticsearch.action.bulk.BulkRequestBuilder; - -import java.io.File; -import java.io.InputStream; -import java.util.Map; - -/** - * Created by Benoit on 08/04/2015. - */ -public abstract class AbstractIndexTypeDao<T extends IndexTypeDao> extends AbstractDao implements IndexTypeDao<T> { - - private final String index; - private final String type; - - public AbstractIndexTypeDao(String index, String type) { - super("duniter.dao."+index); - this.index = index; - this.type = type; - } - - /** - * Create index - * @throws JsonProcessingException - */ - protected abstract void createIndex() throws JsonProcessingException; - - @Override - public String getIndex() { - return index; - } - - @Override - public String getType() { - return type; - } - - @Override - public T createIndexIfNotExists() { - try { - if (!client.existsIndex(index)) { - createIndex(); - } - } - catch(JsonProcessingException e) { - throw new TechnicalException(String.format("Error while creating index [%s]", index)); - } - return (T)this; - } - - @Override - public T deleteIndex() { - client.deleteIndexIfExists(index); - return (T)this; - } - - @Override - public boolean isExists(String docId) { - return client.isDocumentExists(index, type, docId); - } - - public String create(final String json) { - return client.indexDocumentFromJson(index, type, json); - } - - public void update(final String id, final String json) { - client.updateDocumentFromJson(index, type, id, json); - } - - public String indexDocumentFromJson(String json) { - return client.indexDocumentFromJson(index, type, json); - } - - public void updateDocumentFromJson(String id, String json) { - client.updateDocumentFromJson(index, type, id, json); - } - - /** - * Retrieve a field from a document id - * @param docId - * @return - */ - public Object getFieldById(String docId, String fieldName) { - return client.getFieldById(index, type, docId, fieldName); - } - - public <T> T getTypedFieldById(String docId, String fieldName) { - return client.getTypedFieldById(index, type, docId, fieldName); - } - - @Override - public Map<String, Object> getMandatoryFieldsById(String docId, String... fieldNames) { - return client.getMandatoryFieldsById(index, type, docId, fieldNames); - } - - @Override - public Map<String, Object> getFieldsById(String docId, String... fieldNames) { - return client.getFieldsById(index, type, docId, fieldNames); - } - - /** - * Retrieve a document by id (safe mode) - * @param docId - * @return - */ - public <T extends Object> T getSourceByIdOrNull(String docId, Class<T> classOfT, String... fieldNames) { - return client.getSourceByIdOrNull(index, type, docId, classOfT, fieldNames); - } - - /** - * Retrieve a document by id - * @param docId - * @return - */ - public <T extends Object> T getSourceById(String docId, Class<T> classOfT, String... fieldNames) { - return client.getSourceById(index, type, docId, classOfT, fieldNames); - } - - public void bulkFromClasspathFile(String classpathFile) { - client.bulkFromClasspathFile(classpathFile, index, type, null); - } - - public void bulkFromClasspathFile(String classpathFile, StringReaderHandler handler) { - client.bulkFromClasspathFile(classpathFile, index, type, handler); - } - - public void bulkFromFile(File file) { - client.bulkFromFile(file, index, type, null); - } - - public void bulkFromFile(File file, StringReaderHandler handler) { - client.bulkFromFile(file, index, type, handler); - } - - public void bulkFromStream(InputStream is) { - client.bulkFromStream(is, index, type, null); - } - - public void bulkFromStream(InputStream is, StringReaderHandler handler) { - client.bulkFromStream(is, index, type, handler); - } - - public void flushDeleteBulk(BulkRequestBuilder bulkRequest) { - client.flushDeleteBulk(index, type, bulkRequest); - } - - @Override - public boolean existsIndex() { - return client.existsIndex(index); - } - - public void create(String json, boolean wait) { - Preconditions.checkNotNull(json); - - // Execute - client.safeExecuteRequest(client.prepareIndex(getIndex(), getType()) - .setRefresh(false) // let's see if this works - .setSource(json), wait); - } - - public void update(String id, String json, boolean wait) { - Preconditions.checkNotNull(json); - - // Execute - client.safeExecuteRequest(client.prepareUpdate(getIndex(), getType(), id) - .setRefresh(false) // let's see if this works - .setDoc(json), wait); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/BlockDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/BlockDao.java deleted file mode 100644 index 9e4dca4cbb926421d691e15f2e76c030b49d3189..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/BlockDao.java +++ /dev/null @@ -1,73 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.beans.Bean; -import org.duniter.core.client.model.bma.BlockchainBlock; - -import java.util.Collection; -import java.util.List; -import java.util.Set; - -/** - * Created by blavenie on 03/04/17. - */ -public interface BlockDao extends Bean, TypeDao<BlockDao> { - - String TYPE = "block"; - - - void create(BlockchainBlock block, boolean wait); - - /** - * - * @param currencyName - * @param number the block number - * @param json block as JSON - */ - void create(String currencyName, String id, byte[] json, boolean wait); - - boolean isExists(String currencyName, String id); - - void update(BlockchainBlock block, boolean wait); - - /** - * - * @param currencyName - * @param number the block number, or -1 for current - * @param json block as JSON - */ - void update(String currencyName, String id, byte[] json, boolean wait); - - List<BlockchainBlock> findBlocksByHash(String currencyName, String query); - - int getMaxBlockNumber(String currencyName); - - BlockchainBlock getBlockById(String currencyName, String id); - - void deleteRange(final String currencyName, final int fromNumber, final int toNumber); - - List<BlockchainBlock> getBlocksByIds(String currencyName, Collection<String> ids); - - void deleteById(final String currencyName, String id); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/BlockStatDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/BlockStatDao.java deleted file mode 100644 index b0919aa987ba15044ba7e644e3408f14bbc7f55a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/BlockStatDao.java +++ /dev/null @@ -1,66 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.beans.Bean; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.elasticsearch.model.BlockchainBlockStat; - -import java.util.List; - -/** - * Created by blavenie on 03/04/17. - */ -public interface BlockStatDao extends Bean, TypeDao<BlockStatDao> { - - String TYPE = "blockstat"; - - void create(BlockchainBlockStat block, boolean wait); - - /** - * - * @param currencyName - * @param number the block number - * @param json block as JSON - */ - void create(String currencyName, String id, byte[] json, boolean wait); - - boolean isExists(String currencyName, String id); - - void update(BlockchainBlockStat block, boolean wait); - - /** - * - * @param currencyName - * @param number the block number, or -1 for current - * @param json block as JSON - */ - void update(String currencyName, String id, byte[] json, boolean wait); - - void delete(String currency, String id, boolean wait); - - void delete(String currency, String id, String hash, boolean wait); - - BlockchainBlockStat toBlockStat(BlockchainBlock block); - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/CurrencyExtendDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/CurrencyExtendDao.java deleted file mode 100644 index de4aae0ac0f93977c6b9baed4b051e4c9dcaf7cf..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/CurrencyExtendDao.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.dao.CurrencyDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface CurrencyExtendDao extends CurrencyDao, IndexTypeDao<CurrencyExtendDao> { - String INDEX = "currency"; - String RECORD_TYPE = "record"; - - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/DaoModule.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/DaoModule.java deleted file mode 100644 index acb7f105a2e853240b44d47af79dd7b95b89ef8b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/DaoModule.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/* - * #%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.core.beans.Bean; -import org.duniter.core.client.dao.CurrencyDao; -import org.duniter.core.client.dao.PeerDao; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.client.Duniter4jClientImpl; -import org.duniter.elasticsearch.dao.impl.BlockStatDaoImpl; -import org.duniter.elasticsearch.dao.impl.DocStatDaoImpl; -import org.duniter.elasticsearch.dao.impl.MovementDaoImpl; -import org.duniter.elasticsearch.dao.impl.SynchroExecutionDaoImpl; -import org.duniter.elasticsearch.service.ServiceLocator; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class DaoModule extends AbstractModule implements Module { - - @Override protected void configure() { - - requestInjection(ServiceLocator.getESBeanFactory()); - - // Common instance - bind(Duniter4jClient.class).to(Duniter4jClientImpl.class).asEagerSingleton(); - bind(DocStatDao.class).to(DocStatDaoImpl.class).asEagerSingleton(); - - // Dao defined in module es-core - bind(BlockStatDao.class).to(BlockStatDaoImpl.class).asEagerSingleton(); - bind(MovementDao.class).to(MovementDaoImpl.class).asEagerSingleton(); - bind(SynchroExecutionDao.class).to(SynchroExecutionDaoImpl.class).asEagerSingleton(); - - // Dao defined in module core-client - bindWithLocator(BlockDao.class); - bindWithLocator(PeerDao.class); - bindWithLocator(CurrencyDao.class); - - - } - - /* protected methods */ - - protected <T extends Bean> void bindWithLocator(Class<T> clazz) { - bind(clazz).toProvider(new ServiceLocator.Provider<>(clazz)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/DocStatDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/DocStatDao.java deleted file mode 100644 index 12dc18d2b966844daac76fd1d38180dd1b153c30..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/DocStatDao.java +++ /dev/null @@ -1,41 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.model.DocStat; -import org.elasticsearch.action.index.IndexRequestBuilder; - -import javax.annotation.Nullable; - -/** - * Created by blavenie on 13/09/17. - */ -public interface DocStatDao extends IndexTypeDao<DocStatDao>{ - String INDEX = "docstat"; - String TYPE = "record"; - - long countDoc(String index, @Nullable String type); - - IndexRequestBuilder prepareIndex(DocStat stat); - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexDao.java deleted file mode 100644 index 1a2b803d390007d4f1ca530549a0d817a328351d..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexDao.java +++ /dev/null @@ -1,40 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.dao.handler.StringReaderHandler; - -/** - * Created by blavenie on 30/03/17. - */ - -public interface IndexDao<T extends IndexDao> { - - T createIndexIfNotExists(); - - T deleteIndex(); - - String getIndex(); - - boolean existsIndex(); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java deleted file mode 100644 index 79244d80c7056c7603b368fd63f6b02f6d2cca6b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java +++ /dev/null @@ -1,60 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.dao.handler.StringReaderHandler; -import org.elasticsearch.common.xcontent.XContentBuilder; - -import java.util.Map; - -/** - * Created by blavenie on 03/04/17. - */ -public interface IndexTypeDao<T extends IndexTypeDao> extends IndexDao<T> { - - T createIndexIfNotExists(); - - T deleteIndex(); - - String getIndex(); - - boolean existsIndex(); - - XContentBuilder createTypeMapping(); - - String getType(); - - boolean isExists(String docId); - - Object getFieldById(String docId, String fieldName); - - Map<String, Object> getFieldsById(String docId, String... fieldNames); - - <B> B getTypedFieldById(String docId, String fieldName); - - Map<String, Object> getMandatoryFieldsById(String docId, String... fieldNames); - - void bulkFromClasspathFile(String classpathFile); - - void bulkFromClasspathFile(String classpathFile, StringReaderHandler handler); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/MovementDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/MovementDao.java deleted file mode 100644 index fdb6be061d128bce43adb855d1c87312581ecec2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/MovementDao.java +++ /dev/null @@ -1,53 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.beans.Bean; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.elasticsearch.model.Movement; -import org.elasticsearch.action.bulk.BulkRequestBuilder; - -import java.util.List; - -/** - * Created by blavenie on 03/04/17. - */ -public interface MovementDao extends Bean, TypeDao<MovementDao> { - - String TYPE = "movement"; - - void create(Movement block, boolean wait); - - boolean isExists(String currencyName, String id); - - void update(Movement operation, boolean wait); - - void delete(String currency, String id, boolean wait); - - BulkRequestBuilder bulkDeleteByBlock(String currency, - String number, - String hash, - BulkRequestBuilder bulkRequest, - int bulkSize, - boolean flushAll); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/PeerDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/PeerDao.java deleted file mode 100644 index 30dd30260be60e3e466573cdba0d7a503f6cf09c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/PeerDao.java +++ /dev/null @@ -1,33 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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% - */ - -/** - * Created by blavenie on 26/04/17. - */ -public interface PeerDao extends org.duniter.core.client.dao.PeerDao, TypeDao<PeerDao>{ - - String TYPE = "peer"; - - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/SynchroExecutionDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/SynchroExecutionDao.java deleted file mode 100644 index c6b3071a734f579c8c6bcf2bde287e51108b19ed..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/SynchroExecutionDao.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.local.Peer; -import org.duniter.elasticsearch.model.SynchroExecution; - -/** - * Created by blavenie on 26/04/17. - */ -public interface SynchroExecutionDao extends TypeDao<SynchroExecutionDao>{ - - String TYPE = "synchro"; - - void save(SynchroExecution execution); - - SynchroExecution getLastExecution(Peer peer); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/TypeDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/TypeDao.java deleted file mode 100644 index fe3e10af00ab96dc1d962c386d57ec4aa541deeb..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/TypeDao.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.dao; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.common.xcontent.XContentBuilder; - -import java.util.Map; - -/** - * Created by blavenie on 30/03/17. - */ - -public interface TypeDao<T extends TypeDao> { - - XContentBuilder createTypeMapping(); - - String getType(); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/handler/AddSequenceAttributeHandler.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/handler/AddSequenceAttributeHandler.java deleted file mode 100644 index a2c9824e62739804590d63fd1ea9269015765fee..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/handler/AddSequenceAttributeHandler.java +++ /dev/null @@ -1,48 +0,0 @@ -package org.duniter.elasticsearch.dao.handler; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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 java.util.regex.Pattern; - -public class AddSequenceAttributeHandler implements StringReaderHandler { - private int order; - private final String attributeName; - private final Pattern filterPattern; - public AddSequenceAttributeHandler(String attributeName, String filterRegex, int startValue) { - this.order = startValue; - this.attributeName = attributeName; - this.filterPattern = Pattern.compile(filterRegex); - } - - @Override - public String onReadLine(String line) { - // add 'order' field into - if (filterPattern.matcher(line).matches()) { - return String.format("%s, \"%s\": %d}", - line.substring(0, line.length()-1), - attributeName, - order++); - } - return line; - } - } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/handler/StringReaderHandler.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/handler/StringReaderHandler.java deleted file mode 100644 index 2aea68ed34b7692430f4d3445d3f20f977620d40..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/handler/StringReaderHandler.java +++ /dev/null @@ -1,28 +0,0 @@ -package org.duniter.elasticsearch.dao.handler; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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% - */ - -public interface StringReaderHandler { - - String onReadLine(String line); - } diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java deleted file mode 100644 index c3bf183539b63b69cd3eb4979755982d735e0ba2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockDaoImpl.java +++ /dev/null @@ -1,414 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.Lists; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.core.util.json.JsonSyntaxException; -import org.duniter.elasticsearch.dao.AbstractDao; -import org.duniter.elasticsearch.dao.BlockDao; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.SearchHitField; -import org.elasticsearch.search.aggregations.AggregationBuilders; -import org.elasticsearch.search.aggregations.metrics.max.Max; -import org.elasticsearch.search.highlight.HighlightField; -import org.elasticsearch.search.sort.SortOrder; - -import java.io.IOException; -import java.util.Collection; -import java.util.List; -import java.util.Map; -import java.util.Set; - -/** - * Created by Benoit on 30/03/2015. - */ -public class BlockDaoImpl extends AbstractDao implements BlockDao { - - - public BlockDaoImpl(){ - super("duniter.dao.block"); - } - - @Override - public String getType() { - return TYPE; - } - - public void create(BlockchainBlock block, boolean wait) { - Preconditions.checkNotNull(block); - Preconditions.checkArgument(StringUtils.isNotBlank(block.getCurrency())); - Preconditions.checkNotNull(block.getHash()); - Preconditions.checkNotNull(block.getNumber()); - - // Serialize into JSON - try { - String json = getObjectMapper().writeValueAsString(block); - - // Preparing - IndexRequestBuilder request = client.prepareIndex(block.getCurrency(), TYPE) - .setId(block.getNumber().toString()) - .setSource(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - /** - * - * @param currencyName - * @param id the block id - * @param json block as JSON - */ - public void create(String currencyName, String id, byte[] json, boolean wait) { - Preconditions.checkNotNull(currencyName); - Preconditions.checkNotNull(id); - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length > 0); - - // Preparing indexBlocksFromNode - IndexRequestBuilder request = client.prepareIndex(currencyName, TYPE) - .setId(id) - .setRefresh(true) - .setSource(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - - public boolean isExists(String currencyName, String id) { - return client.isDocumentExists(currencyName, TYPE, id); - } - - public void update(BlockchainBlock block, boolean wait) { - Preconditions.checkNotNull(block); - Preconditions.checkArgument(StringUtils.isNotBlank(block.getCurrency())); - Preconditions.checkNotNull(block.getHash()); - Preconditions.checkNotNull(block.getNumber()); - - // Serialize into JSON - // WARN: must use GSON, to have same JSON result (e.g identities and joiners field must be converted into String) - try { - String json = getObjectMapper().writeValueAsString(block); - - // Preparing - UpdateRequestBuilder request = client.prepareUpdate(block.getCurrency(), TYPE, block.getNumber().toString()) - .setRefresh(true) - .setDoc(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - /** - * - * @param currencyName - * @param id the block id - * @param json block as JSON - */ - public void update(String currencyName, String id, byte[] json, boolean wait) { - Preconditions.checkNotNull(currencyName); - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length > 0); - - // Preparing indexBlocksFromNode - UpdateRequestBuilder request = client.prepareUpdate(currencyName, TYPE, id) - .setRefresh(true) - .setDoc(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - - public List<BlockchainBlock> findBlocksByHash(String currencyName, String query) { - String[] queryParts = query.split("[\\t ]+"); - - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(currencyName) - .setTypes(TYPE) - .setFetchSource(true) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH); - - // If only one term, search as prefix - if (queryParts.length == 1) { - searchRequest.setQuery(QueryBuilders.prefixQuery("hash", query)); - } - - // If more than a word, search on terms match - else { - searchRequest.setQuery(QueryBuilders.matchQuery("hash", query)); - } - - // Sort as score/memberCount - searchRequest.addSort("_score", SortOrder.DESC) - .addSort("number", SortOrder.DESC); - - // Highlight matched words - searchRequest.setHighlighterTagsSchema("styled") - .addHighlightedField("hash") - .addFields("hash") - .addFields("*", "_source"); - - // Execute query - SearchResponse searchResponse = searchRequest.execute().actionGet(); - - // Read query result - return toBlocks(searchResponse, true); - } - - public List<BlockchainBlock> getBlocksByIds(String currencyName, Collection<String> ids) { - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(currencyName) - .setTypes(TYPE) - .setSize(ids.size()) - .setFetchSource(true) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH); - - // If only one term, search as prefix - searchRequest.setQuery(QueryBuilders.idsQuery(TYPE).addIds(ids)); - - // Sort as id - searchRequest.addSort("_id", SortOrder.ASC); - - // Execute query - SearchResponse searchResponse = searchRequest.execute().actionGet(); - - // Read query result - return toBlocks(searchResponse, false); - } - - public int getMaxBlockNumber(String currencyName) { - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(currencyName) - .setTypes(TYPE) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH); - - // Get max(number) - searchRequest.addAggregation(AggregationBuilders.max("max_number").field("number")); - - // Execute query - SearchResponse searchResponse = searchRequest.execute().actionGet(); - - // Read query result - Max result = searchResponse.getAggregations().get("max_number"); - if (result == null) { - return -1; - } - - return (result.getValue() == Double.NEGATIVE_INFINITY) - ? -1 - : (int)result.getValue(); - } - - - public BlockchainBlock getBlockById(String currencyName, String id) { - return client.getSourceById(currencyName, TYPE, id, BlockchainBlock.class); - } - - /** - * Delete blocks from a start number (using bulk) - * @param currencyName - * @param fromNumber - */ - public void deleteRange(final String currencyName, final int fromNumber, final int toNumber) { - - int bulkSize = pluginSettings.getIndexBulkSize(); - - BulkRequestBuilder bulkRequest = client.prepareBulk(); - for (int number=fromNumber; number<=toNumber; number++) { - - bulkRequest.add( - client.prepareDelete(currencyName, TYPE, String.valueOf(number)) - ); - - // Flush the bulk if not empty - if ((fromNumber - number % bulkSize) == 0) { - client.flushDeleteBulk(currencyName, TYPE, bulkRequest); - bulkRequest = client.prepareBulk(); - } - } - - // last flush - client.flushDeleteBulk(currencyName, TYPE, bulkRequest); - } - - @Override - public void deleteById(String currencyName, String number) { - client.prepareDelete(currencyName, TYPE, number).execute().actionGet(); - } - - - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder() - .startObject() - .startObject(TYPE) - .startObject("properties") - - // currency - .startObject("currency") - .field("type", "string") - .endObject() - - // version - .startObject("version") - .field("type", "integer") - .endObject() - - // time - .startObject("time") - .field("type", "long") - .endObject() - - // medianTime - .startObject("medianTime") - .field("type", "long") - .endObject() - - // number - .startObject("number") - .field("type", "integer") - .endObject() - - // nonce - .startObject("nonce") - .field("type", "long") - .endObject() - - // hash - .startObject("hash") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // issuer - .startObject("issuer") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // previous hash - .startObject("previousHash") - .field("type", "string") - .endObject() - - // membersCount - .startObject("membersCount") - .field("type", "integer") - .endObject() - - // unitbase - .startObject("unitbase") - .field("type", "integer") - .endObject() - - // monetaryMass - .startObject("monetaryMass") - .field("type", "long") - .endObject() - - // dividend - .startObject("dividend") - .field("type", "integer") - .endObject() - - // identities: - //.startObject("identities") - //.endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException("Error while getting mapping for block index: " + ioe.getMessage(), ioe); - } - } - - - /* -- Internal methods -- */ - - protected List<BlockchainBlock> toBlocks(SearchResponse response, boolean withHighlight) { - // Read query result - List<BlockchainBlock> result = Lists.newArrayList(); - - response.getHits().forEach(searchHit -> { - BlockchainBlock block; - if (searchHit.source() != null) { - String jsonString = new String(searchHit.source()); - try { - block = getObjectMapper().readValue(jsonString, BlockchainBlock.class); - } catch(Exception e) { - if (logger.isDebugEnabled()) { - logger.debug("Error while parsing block from JSON:\n" + jsonString); - } - throw new JsonSyntaxException("Error while read block from JSON: " + e.getMessage(), e); - } - } - else { - block = new BlockchainBlock(); - SearchHitField field = searchHit.getFields().get("hash"); - block.setHash(field.getValue()); - } - result.add(block); - - // If possible, use highlights - if (withHighlight) { - Map<String, HighlightField> fields = searchHit.getHighlightFields(); - for (HighlightField field : fields.values()) { - String blockNameHighLight = field.getFragments()[0].string(); - block.setHash(blockNameHighLight); - } - } - }); - - return result; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockStatDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockStatDaoImpl.java deleted file mode 100644 index d0ac965980494abb05e04a6384065c4b237d4318..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/BlockStatDaoImpl.java +++ /dev/null @@ -1,327 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.bma.BlockchainBlocks; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.dao.AbstractDao; -import org.duniter.elasticsearch.dao.BlockStatDao; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.model.BlockchainBlockStat; -import org.elasticsearch.action.delete.DeleteRequestBuilder; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.metrics.CounterMetric; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; -import java.math.BigInteger; -import java.util.Arrays; - -/** - * Created by Benoit on 30/03/2015. - */ -public class BlockStatDaoImpl extends AbstractDao implements BlockStatDao { - - public BlockStatDaoImpl(){ - super("duniter.dao.block.stat"); - } - - @Override - public String getType() { - return TYPE; - } - - public void create(BlockchainBlockStat block, boolean wait) { - Preconditions.checkNotNull(block); - Preconditions.checkArgument(StringUtils.isNotBlank(block.getCurrency())); - Preconditions.checkNotNull(block.getHash()); - Preconditions.checkNotNull(block.getNumber()); - - // Serialize into JSON - try { - String json = getObjectMapper().writeValueAsString(block); - - // Preparing - IndexRequestBuilder request = client.prepareIndex(block.getCurrency(), TYPE) - .setId(String.valueOf(block.getNumber())) - .setRefresh(false) - .setSource(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - @Override - public void create(String currencyName, String id, byte[] json, boolean wait) { - Preconditions.checkNotNull(currencyName); - Preconditions.checkNotNull(id); - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length > 0); - - // Preparing indexBlocksFromNode - IndexRequestBuilder request = client.prepareIndex(currencyName, TYPE) - .setId(id) - .setRefresh(false) - .setSource(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - - public boolean isExists(String currencyName, String id) { - return client.isDocumentExists(currencyName, TYPE, id); - } - - public void update(BlockchainBlockStat block, boolean wait) { - Preconditions.checkNotNull(block); - Preconditions.checkArgument(StringUtils.isNotBlank(block.getCurrency())); - Preconditions.checkNotNull(block.getNumber()); - - // Serialize into JSON - // WARN: must use GSON, to have same JSON result (e.g identities and joiners field must be converted into String) - try { - String json = getObjectMapper().writeValueAsString(block); - - // Preparing - UpdateRequestBuilder request = client.prepareUpdate(block.getCurrency(), TYPE, block.getNumber().toString()) - .setRefresh(true) - .setDoc(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - /** - * - * @param currencyName - * @param id the block id - * @param json block as JSON - */ - public void update(String currencyName, String id, byte[] json, boolean wait) { - Preconditions.checkNotNull(currencyName); - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length > 0); - - // Preparing index - UpdateRequestBuilder request = client.prepareUpdate(currencyName, TYPE, id) - .setRefresh(true) - .setDoc(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - - @Override - public void delete(String currency, String id, boolean wait) { - Preconditions.checkNotNull(currency); - Preconditions.checkNotNull(id); - - // Preparing request - DeleteRequestBuilder request = client.prepareDelete(currency, TYPE, id); - - // Execute - client.safeExecuteRequest(request, wait); - } - - @Override - public void delete(String currency, String id, String hash, boolean wait) { - Preconditions.checkNotNull(currency); - Preconditions.checkNotNull(id); - Preconditions.checkNotNull(hash); - - try { - // get the current hash - String existingHash = client.getTypedFieldById(currency, TYPE, id, BlockchainBlockStat.PROPERTY_HASH); - - // Execute the delete, only if same hash - if (hash.equals(existingHash)) { - DeleteRequestBuilder request = client.prepareDelete(currency, TYPE, id); - client.safeExecuteRequest(request, wait); - } - } catch(NotFoundException e) { - // Not exists: do not delete - } - - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder() - .startObject() - .startObject(TYPE) - .startObject("properties") - - // currency - .startObject(BlockchainBlockStat.PROPERTY_CURRENCY) - .field("type", "string") - .endObject() - - // version - .startObject(BlockchainBlockStat.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // block number - .startObject(BlockchainBlockStat.PROPERTY_NUMBER) - .field("type", "integer") - .endObject() - - // medianTime - .startObject(BlockchainBlockStat.PROPERTY_MEDIAN_TIME) - .field("type", "long") - .endObject() - - // issuer - .startObject(BlockchainBlockStat.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // hash - .startObject(BlockchainBlockStat.PROPERTY_HASH) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // membersCount - .startObject(BlockchainBlockStat.PROPERTY_MEMBERS_COUNT) - .field("type", "integer") - .endObject() - - // unitbase - .startObject(BlockchainBlockStat.PROPERTY_UNITBASE) - .field("type", "integer") - .endObject() - - // monetaryMass - .startObject(BlockchainBlockStat.PROPERTY_MONETARY_MASS) - .field("type", "long") - .endObject() - - // dividend - .startObject(BlockchainBlockStat.PROPERTY_DIVIDEND) - .field("type", "integer") - .endObject() - - // --- STATS properties --- - - // txCount - .startObject(BlockchainBlockStat.PROPERTY_TX_COUNT) - .field("type", "integer") - .endObject() - - // txAmount - .startObject(BlockchainBlockStat.PROPERTY_TX_AMOUNT) - .field("type", "long") - .endObject() - - // txChangeCount - .startObject(BlockchainBlockStat.PROPERTY_TX_CHANGE_COUNT) - .field("type", "integer") - .endObject() - - // certCount - .startObject(BlockchainBlockStat.PROPERTY_CERT_COUNT) - .field("type", "integer") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException("Error while getting mapping for block stat index: " + ioe.getMessage(), ioe); - } - } - - public BlockchainBlockStat toBlockStat(BlockchainBlock block) { - - BlockchainBlockStat result = newBlockStat(block); - - // Tx - if (CollectionUtils.isNotEmpty(block.getTransactions())) { - CounterMetric txChangeCounter = new CounterMetric(); - CounterMetric txAmountCounter = new CounterMetric(); - Arrays.stream(block.getTransactions()) - .forEach(tx -> { - long txAmount = BlockchainBlocks.getTxAmount(tx); - if (txAmount == 0l) { - txChangeCounter.inc(); - } - else { - txAmountCounter.inc(txAmount); - } - }); - result.setTxAmount(BigInteger.valueOf(txAmountCounter.count())); - result.setTxChangeCount((int)txChangeCounter.count()); - result.setTxCount(block.getTransactions().length); - } - else { - result.setTxAmount(BigInteger.valueOf(0)); - result.setTxChangeCount(0); - result.setTxCount(0); - } - - // Cert count - result.setCertCount(CollectionUtils.size(block.getCertifications())); - - return result; - } - - /* -- Internal methods -- */ - - private BlockchainBlockStat newBlockStat(BlockchainBlock block) { - BlockchainBlockStat stat = new BlockchainBlockStat(); - - stat.setNumber(block.getNumber()); - stat.setCurrency(block.getCurrency()); - stat.setHash(block.getHash()); - stat.setIssuer(block.getIssuer()); - stat.setMedianTime(block.getMedianTime()); - stat.setMembersCount(block.getMembersCount()); - stat.setMonetaryMass(block.getMonetaryMass()); - stat.setUnitbase(block.getUnitbase()); - stat.setVersion(block.getVersion()); - stat.setDividend(block.getDividend()); - - return stat; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/CurrencyDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/CurrencyDaoImpl.java deleted file mode 100644 index be4033f9783db1cbc170ebbb394a4d5c1feabe86..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/CurrencyDaoImpl.java +++ /dev/null @@ -1,247 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * UCoin Java :: 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.core.JsonProcessingException; -import com.google.common.collect.Lists; -import org.duniter.core.client.model.local.Currency; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.dao.CurrencyExtendDao; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; -import java.util.List; -import java.util.Map; - -/** - * Created by blavenie on 29/12/15. - */ -public class CurrencyDaoImpl extends AbstractIndexTypeDao<CurrencyExtendDao> implements CurrencyExtendDao { - - protected static final String REGEX_WORD_SEPARATOR = "[-\\t@# _]+"; - - public CurrencyDaoImpl(){ - super(INDEX, RECORD_TYPE); - } - - @Override - public org.duniter.core.client.model.local.Currency create(final org.duniter.core.client.model.local.Currency currency) { - - try { - - if (currency instanceof org.duniter.core.client.model.elasticsearch.Currency) { - fillTags((org.duniter.core.client.model.elasticsearch.Currency)currency); - } - - // Serialize into JSON - byte[] json = getObjectMapper().writeValueAsBytes(currency); - - // Preparing indexBlocksFromNode - IndexRequestBuilder indexRequest = client.prepareIndex(INDEX, RECORD_TYPE) - .setId(currency.getId()) - .setSource(json); - - // Execute indexBlocksFromNode - indexRequest - .setRefresh(true) - .execute().actionGet(); - - } catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - - return currency; - } - - @Override - public org.duniter.core.client.model.local.Currency update(final org.duniter.core.client.model.local.Currency currency) { - try { - - if (currency instanceof org.duniter.core.client.model.elasticsearch.Currency) { - fillTags((org.duniter.core.client.model.elasticsearch.Currency)currency); - } - - // Serialize into JSON - byte[] json = getObjectMapper().writeValueAsBytes(currency); - - UpdateRequestBuilder updateRequest = client.prepareUpdate(INDEX, RECORD_TYPE, currency.getId()) - .setDoc(json); - - // Execute indexBlocksFromNode - updateRequest - .setRefresh(true) - .execute(); - - } catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - - - return currency; - } - - @Override - public void remove(final org.duniter.core.client.model.local.Currency currency) { - Preconditions.checkNotNull(currency); - Preconditions.checkArgument(StringUtils.isNotBlank(currency.getId())); - - // Delete the document - client.prepareDelete(INDEX, RECORD_TYPE, currency.getId()).execute().actionGet(); - } - - @Override - public org.duniter.core.client.model.local.Currency getById(String currencyId) { - return client.getSourceByIdOrNull(INDEX, RECORD_TYPE, currencyId, org.duniter.core.client.model.elasticsearch.Currency.class); - } - - @Override - public List<Currency> getCurrencies(long accountId) { - throw new TechnicalException("Not implemented yet"); - } - - @Override - public List<String> getCurrencyIds() { - SearchRequestBuilder request = client.prepareSearch(INDEX) - .setTypes(RECORD_TYPE) - .setSize(pluginSettings.getIndexBulkSize()) - .setFetchSource(false); - - return toListIds(request.execute().actionGet()); - } - - @Override - public long getLastUD(String currencyId) { - org.duniter.core.client.model.local.Currency currency = getById(currencyId); - if (currency == null) { - return -1; - } - return currency.getLastUD(); - } - - @Override - public Map<Integer, Long> getAllUD(String currencyId) { - - throw new TechnicalException("Not implemented yet"); - } - - @Override - public void insertUDs(String currencyId, Map<Integer, Long> newUDs) { - throw new TechnicalException("Not implemented yet"); - } - - public boolean existsIndex() { - return client.existsIndex(INDEX); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(RECORD_TYPE) - .startObject("properties") - - // currency - .startObject("currency") - .field("type", "string") - .endObject() - - // firstBlockSignature - .startObject("firstBlockSignature") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // member count - .startObject("membersCount") - .field("type", "long") - .endObject() - - // lastUD - .startObject("lastUD") - .field("type", "long") - .endObject() - - // unitbase - .startObject("unitbase") - .field("type", "integer") - .endObject() - - // tags - .startObject("tags") - .field("type", "completion") - .field("search_analyzer", "simple") - .field("analyzer", "simple") - .field("preserve_separators", "false") - - .endObject() - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", INDEX, RECORD_TYPE, ioe.getMessage()), ioe); - } - } - - /* -- internal methods -- */ - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", INDEX)); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(RECORD_TYPE, createTypeMapping()); - createIndexRequestBuilder.execute().actionGet(); - } - - protected void fillTags(org.duniter.core.client.model.elasticsearch.Currency currency) { - String currencyName = currency.getCurrencyName(); - String[] tags = currencyName.split(REGEX_WORD_SEPARATOR); - List<String> tagsList = Lists.newArrayList(tags); - - // Convert as a sentence (replace separator with a space) - String sentence = currencyName.replaceAll(REGEX_WORD_SEPARATOR, " "); - if (!tagsList.contains(sentence)) { - tagsList.add(sentence); - } - - currency.setTags(tagsList.toArray(new String[tagsList.size()])); - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/DocStatDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/DocStatDaoImpl.java deleted file mode 100644 index 5f3f7656ae3a7d9d14789b3f93cddcfc31b271de..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/DocStatDaoImpl.java +++ /dev/null @@ -1,163 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.dao.*; -import org.duniter.elasticsearch.model.DocStat; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -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 java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class DocStatDaoImpl extends AbstractIndexTypeDao<DocStatDao> implements DocStatDao { - - private PluginSettings pluginSettings; - - @Inject - public DocStatDaoImpl(PluginSettings pluginSettings) { - super(DocStatDao.INDEX, DocStatDao.TYPE); - this.pluginSettings = pluginSettings; - } - - @Override - public String getType() { - return TYPE; - } - - @Override - public long countDoc(String index, String type) { - Preconditions.checkArgument(StringUtils.isNotBlank(index)); - - SearchRequestBuilder searchRequest = client.prepareSearch(index) - .setFetchSource(false) - .setSize(0); - - // Set type if present - if (StringUtils.isNotBlank(type)) { - searchRequest.setTypes(type); - } - - SearchResponse response = searchRequest.execute().actionGet(); - return response.getHits().getTotalHits(); - } - - @Override - public IndexRequestBuilder prepareIndex(DocStat stat) { - Preconditions.checkNotNull(stat); - Preconditions.checkArgument(StringUtils.isNotBlank(stat.getIndex())); - - // Make sure time has been set - if (stat.getTime() == 0) { - stat.setTime(System.currentTimeMillis()/1000); - } - - try { - return client.prepareIndex(INDEX, TYPE) - .setRefresh(false) - .setSource(getObjectMapper().writeValueAsBytes(stat)); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", INDEX)); - - client.admin().indices().prepareCreate(INDEX) - .setSettings(Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - .build()) - .addMapping(TYPE, createTypeMapping()) - .execute().actionGet(); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder() - .startObject() - .startObject(TYPE) - .startObject("properties") - - // index - .startObject(DocStat.PROPERTY_INDEX) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // indexType - .startObject(DocStat.PROPERTY_INDEX_TYPE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // type - .startObject(DocStat.PROPERTY_COUNT) - .field("type", "long") - .endObject() - - // time - .startObject(DocStat.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException("Error while getting mapping for doc stat index: " + ioe.getMessage(), ioe); - } - } - - /* -- protected method -- */ - - protected long countData(String index, String type) { - - SearchRequestBuilder searchRequest = client.prepareSearch(index) - .setTypes(type) - .setFetchSource(false) - .setSize(0); - - SearchResponse response = searchRequest.execute().actionGet(); - return response.getHits().getTotalHits(); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/MovementDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/MovementDaoImpl.java deleted file mode 100644 index 140aa0b4b4bfaa35e73d984da83998f19493a6b4..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/MovementDaoImpl.java +++ /dev/null @@ -1,274 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.dao.AbstractDao; -import org.duniter.elasticsearch.dao.BlockDao; -import org.duniter.elasticsearch.dao.MovementDao; -import org.duniter.elasticsearch.model.Movement; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.delete.DeleteRequestBuilder; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; - -import java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class MovementDaoImpl extends AbstractDao implements MovementDao { - - public MovementDaoImpl(){ - super("duniter.dao.movement"); - } - - @Override - public String getType() { - return TYPE; - } - - public void create(Movement operation, boolean wait) { - Preconditions.checkNotNull(operation); - Preconditions.checkArgument(StringUtils.isNotBlank(operation.getCurrency())); - Preconditions.checkNotNull(operation.getIssuer()); - Preconditions.checkNotNull(operation.getRecipient()); - Preconditions.checkNotNull(operation.getAmount()); - - // Serialize into JSON - try { - String json = getObjectMapper().writeValueAsString(operation); - - // Preparing - IndexRequestBuilder request = client.prepareIndex(operation.getCurrency(), TYPE) - .setRefresh(false) - .setSource(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - public boolean isExists(String currencyName, String id) { - return client.isDocumentExists(currencyName, TYPE, id); - } - - public void update(Movement operation, boolean wait) { - Preconditions.checkNotNull(operation); - Preconditions.checkArgument(StringUtils.isNotBlank(operation.getCurrency())); - Preconditions.checkNotNull(operation.getIssuer()); - Preconditions.checkNotNull(operation.getRecipient()); - Preconditions.checkNotNull(operation.getAmount()); - - // Serialize into JSON - try { - String json = getObjectMapper().writeValueAsString(operation); - - // Preparing - UpdateRequestBuilder request = client.prepareUpdate(operation.getCurrency(), TYPE, operation.getId()) - .setRefresh(true) - .setDoc(json); - - // Execute - client.safeExecuteRequest(request, wait); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - @Override - public void delete(String currency, String id, boolean wait) { - Preconditions.checkNotNull(currency); - Preconditions.checkNotNull(id); - - // Preparing request - DeleteRequestBuilder request = client.prepareDelete(currency, TYPE, id); - - // Execute - client.safeExecuteRequest(request, wait); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder() - .startObject() - .startObject(TYPE) - .startObject("properties") - - // --- BLOCK properties --- - - // currency - .startObject(Movement.PROPERTY_CURRENCY) - .field("type", "string") - .endObject() - - // medianTime - .startObject(Movement.PROPERTY_MEDIAN_TIME) - .field("type", "long") - .endObject() - - // --- TX properties --- - - // version - .startObject(Movement.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // issuer - .startObject(Movement.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // recipient - .startObject(Movement.PROPERTY_RECIPIENT) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // amount - .startObject(Movement.PROPERTY_AMOUNT) - .field("type", "long") - .endObject() - - // unitbase - .startObject(Movement.PROPERTY_UNITBASE) - .field("type", "integer") - .endObject() - - // comment - .startObject(Movement.PROPERTY_COMMENT) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // --- OTHER properties --- - - // is UD ? - .startObject(Movement.PROPERTY_IS_UD) - .field("type", "boolean") - .field("index", "not_analyzed") - .endObject() - - // reference - .startObject(Movement.PROPERTY_REFERENCE) - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - // reference.index - .startObject(Movement.Reference.PROPERTY_INDEX) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - // reference.index - .startObject(Movement.Reference.PROPERTY_TYPE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject(Movement.Reference.PROPERTY_ID) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject(Movement.Reference.PROPERTY_HASH) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject(Movement.Reference.PROPERTY_ANCHOR) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .endObject() - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException("Error while getting mapping for block operation index: " + ioe.getMessage(), ioe); - } - } - - public BulkRequestBuilder bulkDeleteByBlock(final String currency, - final String number, - final String hash, - BulkRequestBuilder bulkRequest, - final int bulkSize, - final boolean flushAll) { - - Preconditions.checkNotNull(currency); - Preconditions.checkNotNull(number); - Preconditions.checkNotNull(bulkRequest); - Preconditions.checkArgument(bulkSize > 0); - - // Prepare search request - SearchRequestBuilder searchRequest = client - .prepareSearch(currency) - .setTypes(TYPE) - .setFetchSource(false) - .setSearchType(SearchType.QUERY_AND_FETCH); - - // Query = filter on reference - BoolQueryBuilder boolQuery = QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(Movement.PROPERTY_REFERENCE + "." + Movement.Reference.PROPERTY_INDEX, currency)) - .filter(QueryBuilders.termQuery(Movement.PROPERTY_REFERENCE + "." + Movement.Reference.PROPERTY_TYPE, BlockDao.TYPE)) - .filter(QueryBuilders.termQuery(Movement.PROPERTY_REFERENCE + "." + Movement.Reference.PROPERTY_ID, number)); - if (StringUtils.isNotBlank(hash)) { - boolQuery.filter(QueryBuilders.termQuery(Movement.PROPERTY_REFERENCE + "." + Movement.Reference.PROPERTY_HASH, hash)); - } - - searchRequest.setQuery(QueryBuilders.nestedQuery(Movement.PROPERTY_REFERENCE, QueryBuilders.constantScoreQuery(boolQuery))); - - // Execute query, while there is some data - return client.bulkDeleteFromSearch(currency, TYPE, searchRequest, bulkRequest, bulkSize, flushAll); - } - - public BulkRequestBuilder bulkDeleteByBlock(final BlockchainBlock block, - BulkRequestBuilder bulkRequest, - final int bulkSize, - final boolean flushAll) { - Preconditions.checkNotNull(block); - - return bulkDeleteByBlock(block.getCurrency(), String.valueOf(block.getNumber()), block.getHash(), bulkRequest, bulkSize, flushAll); - } - - /* -- Internal methods -- */ - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/PeerDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/PeerDaoImpl.java deleted file mode 100644 index dcf605ec017c139946c16e388eb867ab1d945335..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/PeerDaoImpl.java +++ /dev/null @@ -1,451 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * UCoin Java :: 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.core.JsonProcessingException; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.dao.AbstractDao; -import org.duniter.elasticsearch.dao.PeerDao; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchPhaseExecutionException; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.NestedQueryBuilder; -import org.elasticsearch.index.query.QueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.SearchHit; -import org.elasticsearch.search.aggregations.AggregationBuilders; -import org.elasticsearch.search.aggregations.bucket.SingleBucketAggregation; -import org.elasticsearch.search.aggregations.metrics.max.Max; - -import java.io.IOException; -import java.util.Date; -import java.util.List; -import java.util.Set; - -/** - * Created by blavenie on 29/12/15. - */ -public class PeerDaoImpl extends AbstractDao implements PeerDao { - - public PeerDaoImpl(){ - super("duniter.dao.peer"); - } - - @Override - public String getType() { - return TYPE; - } - - @Override - public Peer create(Peer peer) { - Preconditions.checkNotNull(peer); - Preconditions.checkArgument(StringUtils.isNotBlank(peer.getId())); - Preconditions.checkArgument(StringUtils.isNotBlank(peer.getCurrency())); - //Preconditions.checkNotNull(peer.getHash()); - Preconditions.checkNotNull(peer.getHost()); - Preconditions.checkNotNull(peer.getApi()); - - // Serialize into JSON - // WARN: must use GSON, to have same JSON result (e.g identities and joiners field must be converted into String) - try { - String json = getObjectMapper().writeValueAsString(peer); - - // Preparing indexBlocksFromNode - IndexRequestBuilder indexRequest = client.prepareIndex(peer.getCurrency(), TYPE) - .setId(peer.getId()) - .setSource(json); - - // Execute indexBlocksFromNode - indexRequest - .setRefresh(true) - .execute(); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - return peer; - } - - @Override - public Peer update(Peer peer) { - Preconditions.checkNotNull(peer); - Preconditions.checkArgument(StringUtils.isNotBlank(peer.getId())); - Preconditions.checkArgument(StringUtils.isNotBlank(peer.getCurrency())); - //Preconditions.checkNotNull(peer.getHash()); - Preconditions.checkNotNull(peer.getHost()); - Preconditions.checkNotNull(peer.getApi()); - - // Serialize into JSON - try { - String json = getObjectMapper().writeValueAsString(peer); - - // Preparing indexBlocksFromNode - UpdateRequestBuilder updateRequest = client.prepareUpdate(peer.getCurrency(), TYPE, peer.getId()) - .setDoc(json); - - // Execute indexBlocksFromNode - updateRequest - .setRefresh(true) - .execute(); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - return peer; - } - - @Override - public Peer getById(String id) { - throw new TechnicalException("not implemented"); - } - - @Override - public void remove(Peer peer) { - Preconditions.checkNotNull(peer); - Preconditions.checkArgument(StringUtils.isNotBlank(peer.getId())); - Preconditions.checkArgument(StringUtils.isNotBlank(peer.getCurrency())); - - // Delete the document - client.prepareDelete(peer.getCurrency(), TYPE, peer.getId()).execute().actionGet(); - } - - @Override - public List<Peer> getPeersByCurrencyId(String currencyId) { - throw new TechnicalException("no implemented: loading all peers may be unsafe for memory..."); - } - - @Override - public List<Peer> getPeersByCurrencyIdAndApi(String currencyId, String endpointApi) { - return getPeersByCurrencyIdAndApiAndPubkeys(currencyId, endpointApi, null); - } - - @Override - public List<Peer> getPeersByCurrencyIdAndApiAndPubkeys(String currencyId, String endpointApi, String[] pubkeys) { - Preconditions.checkNotNull(currencyId); - Preconditions.checkNotNull(endpointApi); - - SearchRequestBuilder request = client.prepareSearch(currencyId) - .setTypes(TYPE) - .setSize(1000); - - // Query = filter on lastUpTime - NestedQueryBuilder statusQuery = QueryBuilders.nestedQuery(Peer.PROPERTY_STATS, - QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(Peer.PROPERTY_STATS + "." + Peer.Stats.PROPERTY_STATUS, Peer.PeerStatus.UP.name()))); - - BoolQueryBuilder boolQuery = QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(Peer.PROPERTY_API, endpointApi)); - if (CollectionUtils.isNotEmpty(pubkeys)) { - boolQuery.filter(QueryBuilders.termsQuery(Peer.PROPERTY_PUBKEY, pubkeys)); - } - - request.setQuery(QueryBuilders.constantScoreQuery(QueryBuilders.boolQuery().must(boolQuery).must(statusQuery))); - - SearchResponse response = request.execute().actionGet(); - return toList(response, Peer.class); - } - - @Override - public boolean isExists(String currencyId, String peerId) { - return client.isDocumentExists(currencyId, TYPE, peerId); - } - - @Override - public Long getMaxLastUpTime(String currencyName) { - - // Prepare request - SearchRequestBuilder searchRequest = client - .prepareSearch(currencyName) - .setTypes(TYPE) - .setFetchSource(false) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH); - - // Get max(number) - searchRequest.addAggregation(AggregationBuilders.nested(Peer.PROPERTY_STATS) - .path(Peer.PROPERTY_STATS) - .subAggregation( - AggregationBuilders.max(Peer.Stats.PROPERTY_LAST_UP_TIME) - .field(Peer.PROPERTY_STATS + "." + Peer.Stats.PROPERTY_LAST_UP_TIME) - .missing(0) - )); - - // Execute query - SearchResponse searchResponse = searchRequest.execute().actionGet(); - - // Read query result - SingleBucketAggregation stats = searchResponse.getAggregations().get(Peer.PROPERTY_STATS); - if (stats == null) return null; - - Max result = stats.getAggregations().get(Peer.Stats.PROPERTY_LAST_UP_TIME); - if (result == null) { - return null; - } - - return (result.getValue() == Double.NEGATIVE_INFINITY) - ? null - : (long)result.getValue(); - } - - @Override - public void updatePeersAsDown(String currencyName, long upTimeLimit) { - - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] Setting peers as DOWN, if older than [%s]...", currencyName, new Date(upTimeLimit*1000))); - } - - SearchRequestBuilder searchRequest = client.prepareSearch(currencyName) - .setFetchSource(false) - .setTypes(TYPE); - - // Query = filter on lastUpTime - BoolQueryBuilder boolQuery = QueryBuilders.boolQuery() - // where lastUpTime < upTimeLimit - .filter(QueryBuilders.rangeQuery(Peer.PROPERTY_STATS + "." + Peer.Stats.PROPERTY_LAST_UP_TIME).lte(upTimeLimit)) - // AND status = UP - .filter(QueryBuilders.termQuery(Peer.PROPERTY_STATS + "." + Peer.Stats.PROPERTY_STATUS, Peer.PeerStatus.UP.name())); - searchRequest.setQuery(QueryBuilders.nestedQuery(Peer.PROPERTY_STATS, QueryBuilders.constantScoreQuery(boolQuery))); - - BulkRequestBuilder bulkRequest = client.prepareBulk(); - - // Execute query, while there is some data - try { - - int counter = 0; - boolean loop = true; - int bulkSize = pluginSettings.getIndexBulkSize(); - searchRequest.setSize(bulkSize); - SearchResponse response = searchRequest.execute().actionGet(); - - // Execute query, while there is some data - do { - - // Read response - SearchHit[] searchHits = response.getHits().getHits(); - for (SearchHit searchHit : searchHits) { - - // Add deletion to bulk - bulkRequest.add( - client.prepareUpdate(currencyName, TYPE, searchHit.getId()) - .setDoc(String.format("{\"%s\": {\"%s\": \"%s\"}}", Peer.PROPERTY_STATS, Peer.Stats.PROPERTY_STATUS, Peer.PeerStatus.DOWN.name()).getBytes()) - ); - counter++; - - // Flush the bulk if not empty - if ((bulkRequest.numberOfActions() % bulkSize) == 0) { - client.flushBulk(bulkRequest); - bulkRequest = client.prepareBulk(); - } - } - - // Prepare next iteration - if (counter == 0 || counter >= response.getHits().getTotalHits()) { - loop = false; - } - // Prepare next iteration - else { - searchRequest.setFrom(counter); - response = searchRequest.execute().actionGet(); - } - } while(loop); - - // last flush - if ((bulkRequest.numberOfActions() % bulkSize) != 0) { - client.flushBulk(bulkRequest); - } - - if (counter > 0) { - logger.info(String.format("Mark %s peers as DOWN", counter)); - } - - } catch (SearchPhaseExecutionException e) { - // Failed or no item on index - logger.error(String.format("Error while update peer status to DOWN: %s.", e.getMessage()), e); - } - - - } - - @Override - public boolean hasPeersUpWithApi(String currencyId, Set<EndpointApi> api) { - SearchRequestBuilder searchRequest = client.prepareSearch(currencyId) - .setFetchSource(false) - .setTypes(TYPE) - .setSize(0); - - // Query = filter on lastUpTime - BoolQueryBuilder query = QueryBuilders.boolQuery(); - - if (CollectionUtils.isNotEmpty(api)) { - query.minimumNumberShouldMatch(api.size()); - api.forEach(a -> query.should(QueryBuilders.termQuery(Peer.PROPERTY_API, a.name()))); - } - - query.must(QueryBuilders.nestedQuery(Peer.PROPERTY_STATS, QueryBuilders.constantScoreQuery(QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(Peer.PROPERTY_STATS + "." + Peer.Stats.PROPERTY_STATUS, Peer.PeerStatus.UP.name()))))); - - searchRequest.setQuery(query); - SearchResponse response = searchRequest.execute().actionGet(); - return response.getHits() != null && response.getHits().getTotalHits() > 0; - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder() - .startObject() - .startObject(TYPE) - .startObject("properties") - - // currency - .startObject(Peer.PROPERTY_CURRENCY) - .field("type", "string") - .endObject() - - // pubkey - .startObject(Peer.PROPERTY_PUBKEY) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // api - .startObject(Peer.PROPERTY_API) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // dns - .startObject(Peer.PROPERTY_DNS) - .field("type", "string") - .endObject() - - // ipv4 - .startObject(Peer.PROPERTY_IPV4) - .field("type", "string") - .endObject() - - // ipv6 - .startObject(Peer.PROPERTY_IPV6) - .field("type", "string") - .endObject() - - // epId - .startObject(Peer.PROPERTY_EP_ID) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // stats - .startObject(Peer.PROPERTY_STATS) - .field("type", "nested") - //.field("dynamic", "false") - .startObject("properties") - - // stats.version - .startObject(Peer.Stats.PROPERTY_VERSION) - .field("type", "string") - .endObject() - - // stats.status - .startObject(Peer.Stats.PROPERTY_STATUS) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // stats.blockNumber - .startObject("blockNumber") - .field("type", "integer") - .endObject() - - // stats.blockHash - .startObject("blockHash") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // stats.error - .startObject("error") - .field("type", "string") - .endObject() - - // stats.medianTime - .startObject("medianTime") - .field("type", "integer") - .endObject() - - // stats.hardshipLevel - .startObject("hardshipLevel") - .field("type", "integer") - .endObject() - - // stats.consensusPct - .startObject("consensusPct") - .field("type", "integer") - .endObject() - - // stats.uid - .startObject(Peer.Stats.PROPERTY_UID) - .field("type", "string") - .endObject() - - // stats.mainConsensus - .startObject("mainConsensus") - .field("type", "boolean") - .field("index", "not_analyzed") - .endObject() - - // stats.forkConsensus - .startObject("forkConsensus") - .field("type", "boolean") - .field("index", "not_analyzed") - .endObject() - - // stats.lastUP - .startObject(Peer.Stats.PROPERTY_LAST_UP_TIME) - .field("type", "integer") - .endObject() - - .endObject() - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException("Error while getting mapping for peer index: " + ioe.getMessage(), ioe); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/SynchroExecutionDaoImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/SynchroExecutionDaoImpl.java deleted file mode 100644 index 45a8ae7559a0966cffaca25175f0bf7d5690dd3f..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/impl/SynchroExecutionDaoImpl.java +++ /dev/null @@ -1,203 +0,0 @@ -package org.duniter.elasticsearch.dao.impl; - -/* - * #%L - * UCoin Java :: 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.core.JsonProcessingException; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.dao.AbstractDao; -import org.duniter.elasticsearch.dao.SynchroExecutionDao; -import org.duniter.elasticsearch.model.SynchroExecution; -import org.duniter.elasticsearch.model.SynchroResult; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.SearchHit; -import org.elasticsearch.search.sort.SortOrder; - -import java.io.IOException; - -/** - * Created by blavenie on 29/12/15. - */ -public class SynchroExecutionDaoImpl extends AbstractDao implements SynchroExecutionDao { - - public SynchroExecutionDaoImpl(){ - super("duniter.dao.peer"); - } - - @Override - public String getType() { - return TYPE; - } - - @Override - public void save(SynchroExecution execution) { - Preconditions.checkNotNull(execution); - Preconditions.checkArgument(StringUtils.isNotBlank(execution.getCurrency())); - Preconditions.checkArgument(StringUtils.isNotBlank(execution.getPeer())); - Preconditions.checkNotNull(execution.getTime()); - Preconditions.checkArgument(execution.getTime() > 0); - - try { - // Serialize into JSON - String json = getObjectMapper().writeValueAsString(execution); - - // Preparing indexBlocksFromNode - IndexRequestBuilder indexRequest = client.prepareIndex(execution.getCurrency(), TYPE) - .setSource(json); - - // Execute indexBlocksFromNode - indexRequest - .setRefresh(true) - .execute().actionGet(); - } - catch(JsonProcessingException e) { - throw new TechnicalException(e); - } - } - - @Override - public SynchroExecution getLastExecution(Peer peer) { - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(peer.getCurrency()); - Preconditions.checkNotNull(peer.getId()); - Preconditions.checkNotNull(peer.getApi()); - - BoolQueryBuilder query = QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(SynchroExecution.PROPERTY_PEER, peer.getId())) - .filter(QueryBuilders.termQuery(SynchroExecution.PROPERTY_API, peer.getApi())); - - SearchResponse response = client.prepareSearch(peer.getCurrency()) - .setTypes(TYPE) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) - .setQuery(query) - .setFetchSource(true) - .setSize(1) - .addSort(SynchroExecution.PROPERTY_TIME, SortOrder.DESC) - .get(); - - if (response.getHits().getTotalHits() == 0) return null; - - SearchHit hit = response.getHits().getHits()[0]; - return client.readSourceOrNull(hit, SynchroExecution.class); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder() - .startObject() - .startObject(TYPE) - .startObject("properties") - - // currency - .startObject(SynchroExecution.PROPERTY_CURRENCY) - .field("type", "string") - .endObject() - - // peer - .startObject(SynchroExecution.PROPERTY_PEER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // peer - .startObject(SynchroExecution.PROPERTY_API) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // issuer - .startObject(SynchroExecution.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // time - .startObject(SynchroExecution.PROPERTY_TIME) - .field("type", "long") - .endObject() - - // hash - .startObject(SynchroExecution.PROPERTY_HASH) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // signature - .startObject(SynchroExecution.PROPERTY_SIGNATURE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // execution time - .startObject(SynchroExecution.PROPERTY_EXECUTION_TIME) - .field("type", "long") - .endObject() - - // result - .startObject(SynchroExecution.PROPERTY_RESULT) - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - - // inserts - .startObject(SynchroResult.PROPERTY_INSERTS) - .field("type", "long") - .endObject() - - // updates - .startObject(SynchroResult.PROPERTY_UPDATES) - .field("type", "long") - .endObject() - - // deletes - .startObject(SynchroResult.PROPERTY_DELETES) - .field("type", "long") - .endObject() - - // deletes - .startObject(SynchroResult.PROPERTY_INVALID_SIGNATURES) - .field("type", "long") - .endObject() - - .endObject() - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException("Error while getting mapping for synchro index: " + ioe.getMessage(), ioe); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/AccessDeniedException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/AccessDeniedException.java deleted file mode 100644 index 21ff39d0ff6038e9ef11bf6f26fd180306a4634d..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/AccessDeniedException.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%L - * Duniter4j :: 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 org.elasticsearch.rest.RestStatus; - -/** - * Created by Benoit on 03/04/2015. - */ -public class AccessDeniedException extends DuniterElasticsearchException{ - - public AccessDeniedException(Throwable cause) { - super(cause); - } - - public AccessDeniedException(String msg, Object... args) { - super(msg, args); - } - - public AccessDeniedException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - @Override - public RestStatus status() { - return RestStatus.FORBIDDEN; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DocumentNotFoundException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DocumentNotFoundException.java deleted file mode 100644 index 338f7012b9aa3f42bfbfa7524ece3785d9801cab..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DocumentNotFoundException.java +++ /dev/null @@ -1,47 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%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.elasticsearch.rest.RestStatus; - -/** - * Created by blavenie on 01/03/16. - */ -public class DocumentNotFoundException extends DuniterElasticsearchException { - public DocumentNotFoundException(Throwable cause) { - super(cause); - } - - public DocumentNotFoundException(String msg, Object... args) { - super(msg, args); - } - - public DocumentNotFoundException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - @Override - public RestStatus status() { - return RestStatus.BAD_REQUEST; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DuniterElasticsearchException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DuniterElasticsearchException.java deleted file mode 100644 index e728d7ee4b2f914b004e4980684a6cca5a55cf10..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DuniterElasticsearchException.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%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.elasticsearch.ElasticsearchException; - -/** - * Created by blavenie on 28/07/16. - */ -public abstract class DuniterElasticsearchException extends ElasticsearchException { - - - public DuniterElasticsearchException(Throwable cause) { - super(cause); - } - - public DuniterElasticsearchException(String msg, Object... args) { - super(msg, args); - } - - public DuniterElasticsearchException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DuplicateIndexIdException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DuplicateIndexIdException.java deleted file mode 100644 index 9845cb0236827247b314845db774c1becdcb93df..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/DuplicateIndexIdException.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.exception.BusinessException; -import org.elasticsearch.rest.RestStatus; - -/** - * - * Created by Benoit on 03/04/2015. - */ -public class DuplicateIndexIdException extends DuniterElasticsearchException{ - - public DuplicateIndexIdException(Throwable cause) { - super(cause); - } - - public DuplicateIndexIdException(String msg, Object... args) { - super(msg, args); - } - - public DuplicateIndexIdException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - - @Override - public RestStatus status() { - return RestStatus.BAD_REQUEST; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidFormatException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidFormatException.java deleted file mode 100644 index f55ff99d9432fe907d50531b23bf972d901c9cd2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidFormatException.java +++ /dev/null @@ -1,47 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%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.elasticsearch.rest.RestStatus; - -/** - * Created by blavenie on 01/03/16. - */ -public class InvalidFormatException extends DuniterElasticsearchException { - public InvalidFormatException(Throwable cause) { - super(cause); - } - - public InvalidFormatException(String msg, Object... args) { - super(msg, args); - } - - public InvalidFormatException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - @Override - public RestStatus status() { - return RestStatus.BAD_REQUEST; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidSignatureException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidSignatureException.java deleted file mode 100644 index d2af9bee3c2371dc35a49504e71a0e3b006eb30d..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidSignatureException.java +++ /dev/null @@ -1,51 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%L - * Duniter4j :: Web - * %% - * 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 org.elasticsearch.rest.RestStatus; - -/** - * Created by Benoit on 03/04/2015. - */ -public class InvalidSignatureException extends DuniterElasticsearchException { - - - public InvalidSignatureException(Throwable cause) { - super(cause); - } - - public InvalidSignatureException(String msg, Object... args) { - super(msg, args); - } - - public InvalidSignatureException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - - @Override - public RestStatus status() { - return RestStatus.BAD_REQUEST; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidTimeException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidTimeException.java deleted file mode 100644 index 388e3dac1a7b1c043502fb14f67816ba07cdc2a6..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/InvalidTimeException.java +++ /dev/null @@ -1,47 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%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.elasticsearch.rest.RestStatus; - -/** - * Created by blavenie on 01/03/16. - */ -public class InvalidTimeException extends DuniterElasticsearchException { - public InvalidTimeException(Throwable cause) { - super(cause); - } - - public InvalidTimeException(String msg, Object... args) { - super(msg, args); - } - - public InvalidTimeException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - @Override - public RestStatus status() { - return RestStatus.BAD_REQUEST; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/NodeConfigException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/NodeConfigException.java deleted file mode 100644 index 74f603f16eef6f75888fc052b5708898fd805fe1..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/NodeConfigException.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%L - * Duniter4j :: 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 org.elasticsearch.rest.RestStatus; - -/** - * Created by Benoit on 03/04/2015. - */ -public class NodeConfigException extends DuniterElasticsearchException{ - - public NodeConfigException(Throwable cause) { - super(cause); - } - - public NodeConfigException(String msg, Object... args) { - super(msg, args); - } - - public NodeConfigException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - @Override - public RestStatus status() { - return RestStatus.INTERNAL_SERVER_ERROR; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/NotFoundException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/NotFoundException.java deleted file mode 100644 index 6ceca19ccdec1b61456f89eaceaf81ade410a15a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/exception/NotFoundException.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.exception; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.exception.BusinessException; -import org.elasticsearch.rest.RestStatus; - -/** - * Created by Benoit on 03/04/2015. - */ -public class NotFoundException extends DuniterElasticsearchException{ - - public NotFoundException(Throwable cause) { - super(cause); - } - - public NotFoundException(String msg, Object... args) { - super(msg, args); - } - - public NotFoundException(String msg, Throwable cause, Object... args) { - super(msg, args, cause); - } - - @Override - public RestStatus status() { - return RestStatus.NOT_FOUND; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/i18n/I18nInitializer.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/i18n/I18nInitializer.java deleted file mode 100644 index 55acfe7c94e29792228ff3fd22ddd7d663727601..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/i18n/I18nInitializer.java +++ /dev/null @@ -1,91 +0,0 @@ -package org.duniter.elasticsearch.i18n; - -/* - * #%L - * Duniter4j :: ElasticSearch Core 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.nuiton.i18n.bundle.I18nBundle; -import org.nuiton.i18n.init.DefaultI18nInitializer; -import org.nuiton.i18n.init.UserI18nInitializer; - -import java.io.File; -import java.util.ArrayList; -import java.util.List; - -/** - * Created by blavenie on 10/01/17. - */ -public class I18nInitializer extends org.nuiton.i18n.init.I18nInitializer{ - protected final File userDirectory; - - private String[] bundleNames; - private String i18nPath; - private List<UserI18nInitializer> delegates; - - - public I18nInitializer(File userDirectory, String[] bundleNames) throws NullPointerException { - this((String)null, userDirectory, bundleNames); - } - - public I18nInitializer(String i18nPath, File userDirectory, String[] bundleNames) throws NullPointerException { - super(); - - this.i18nPath = i18nPath; - this.bundleNames = bundleNames; - this.userDirectory = userDirectory; - this.delegates = createDelegates(userDirectory, bundleNames); - - if(userDirectory == null) { - throw new NullPointerException("parameter \'userDirectory\' can not be null"); - } - } - - public File getUserDirectory() { - return this.userDirectory; - } - - - @Override - public I18nBundle[] resolvBundles() throws Exception { - - List<I18nBundle> result = new ArrayList<>(); - for(DefaultI18nInitializer delegate: delegates) { - I18nBundle[] bundles = delegate.resolvBundles(); - for(I18nBundle bundle: bundles) { - result.add(bundle); - } - } - - return result.toArray(new I18nBundle[result.size()]); - } - - /* -- private methods -- */ - - private List<UserI18nInitializer> createDelegates(File userDirectory, String[] bundleNames) { - List<UserI18nInitializer> result = new ArrayList<>(); - for(String bundleName: bundleNames) { - UserI18nInitializer delegate = new UserI18nInitializer(userDirectory, new DefaultI18nInitializer(bundleName)); - result.add(delegate); - } - return result; - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/BlockchainBlockStat.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/BlockchainBlockStat.java deleted file mode 100644 index add32d5ca052cd277827c7470721fa9c39385155..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/BlockchainBlockStat.java +++ /dev/null @@ -1,181 +0,0 @@ -package org.duniter.elasticsearch.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 java.io.Serializable; -import java.math.BigInteger; - -/** - * Created by blavenie on 29/11/16. - */ -public class BlockchainBlockStat implements Serializable { - - public static final String PROPERTY_VERSION = "version"; - public static final String PROPERTY_CURRENCY = "currency"; - public static final String PROPERTY_NUMBER = "number"; - public static final String PROPERTY_ISSUER = "issuer"; - public static final String PROPERTY_HASH = "hash"; - public static final String PROPERTY_MEDIAN_TIME = "medianTime"; - public static final String PROPERTY_MEMBERS_COUNT = "membersCount"; - public static final String PROPERTY_MONETARY_MASS = "monetaryMass"; - public static final String PROPERTY_UNITBASE= "unitbase"; - public static final String PROPERTY_DIVIDEND = "dividend"; - public static final String PROPERTY_TX_COUNT = "txCount"; - public static final String PROPERTY_TX_AMOUNT = "txAmount"; - public static final String PROPERTY_TX_CHANGE_COUNT = "txChangeCount"; - public static final String PROPERTY_CERT_COUNT = "certCount"; - - // Property copied from Block - private int version; - private String currency; - private Integer number; - private String issuer; - private String hash; - private Long medianTime; - private Integer membersCount; - private BigInteger monetaryMass; - private Integer unitbase; - private BigInteger dividend; - - // Statistics - private Integer txCount; - private BigInteger txAmount; - private Integer txChangeCount; - private Integer certCount; - - public BlockchainBlockStat() { - super(); - } - - public int getVersion() { - return version; - } - - public void setVersion(int version) { - this.version = version; - } - - public String getCurrency() { - return currency; - } - - public void setCurrency(String currency) { - this.currency = currency; - } - - public String getIssuer() { - return issuer; - } - - public void setIssuer(String issuer) { - this.issuer = issuer; - } - - public BigInteger getDividend() { - return dividend; - } - - public void setDividend(BigInteger dividend) { - this.dividend = dividend; - } - - public Integer getTxCount() { - return txCount; - } - - public void setTxCount(Integer txCount) { - this.txCount = txCount; - } - - public BigInteger getTxAmount() { - return txAmount; - } - - public void setTxAmount(BigInteger txAmount) { - this.txAmount = txAmount; - } - - public Integer getTxChangeCount() { - return txChangeCount; - } - - public void setTxChangeCount(Integer txChangeCount) { - this.txChangeCount = txChangeCount; - } - - public Integer getNumber() { - return number; - } - - public void setNumber(Integer number) { - this.number = number; - } - - public String getHash() { - return hash; - } - - public void setHash(String hash) { - this.hash = hash; - } - - public Long getMedianTime() { - return medianTime; - } - - public void setMedianTime(Long medianTime) { - this.medianTime = medianTime; - } - - public Integer getMembersCount() { - return membersCount; - } - - public void setMembersCount(Integer membersCount) { - this.membersCount = membersCount; - } - - public BigInteger getMonetaryMass() { - return monetaryMass; - } - - public void setMonetaryMass(BigInteger monetaryMass) { - this.monetaryMass = monetaryMass; - } - - public Integer getUnitbase() { - return unitbase; - } - - public void setUnitbase(Integer unitbase) { - this.unitbase = unitbase; - } - - public Integer getCertCount() { - return certCount; - } - - public void setCertCount(Integer certCount) { - this.certCount = certCount; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Currency.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Currency.java deleted file mode 100644 index 0e071272530dd31b2a0dcd62c7a2cd0c84979e95..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Currency.java +++ /dev/null @@ -1,101 +0,0 @@ -package org.duniter.elasticsearch.model; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.bma.BlockchainParameters; - -import java.io.Serializable; - -/** - * Created by eis on 05/02/15. - */ -public class Currency implements Serializable { - - public static final String PROPERTY_CURRENCY = "currency"; - - private String currency; - private Integer membersCount; - private String firstBlockSignature; - private Long lastUD; - private BlockchainParameters parameters; - - private String[] tags; - private String issuer; - - public String getCurrency() { - return currency; - } - - public void setCurrency(String currency) { - this.currency = currency; - } - - public Integer getMembersCount() { - return membersCount; - } - - public void setMembersCount(Integer membersCount) { - this.membersCount = membersCount; - } - - public String getFirstBlockSignature() { - return firstBlockSignature; - } - - public void setFirstBlockSignature(String firstBlockSignature) { - this.firstBlockSignature = firstBlockSignature; - } - - public Long getLastUD() { - return lastUD; - } - - public void setLastUD(Long lastUD) { - this.lastUD = lastUD; - } - - public BlockchainParameters getParameters() { - return parameters; - } - - public void setParameters(BlockchainParameters parameters) { - this.parameters = parameters; - } - - public String[] getTags() { - return tags; - } - - public void setTags(String[] tags) { - this.tags = tags; - } - - public String getIssuer() { - return issuer; - } - - public void setIssuer(String issuer) { - this.issuer = issuer; - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/DocStat.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/DocStat.java deleted file mode 100644 index 0b688b52b0e4159bd716a4d5d25475e13ec82aa7..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/DocStat.java +++ /dev/null @@ -1,79 +0,0 @@ -package org.duniter.elasticsearch.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 java.io.Serializable; - -/** - * Created by blavenie on 29/11/16. - */ -public class DocStat implements Serializable { - - public static final String PROPERTY_INDEX = "index"; - public static final String PROPERTY_INDEX_TYPE = "indexType"; - public static final String PROPERTY_COUNT = "docCount"; - public static final String PROPERTY_TIME = "type"; - - // Property copied from Block - private String index; - private String indexType; - private long count; - private long time = 0L; - - - public DocStat() { - super(); - } - - public String getIndex() { - return index; - } - - public void setIndex(String index) { - this.index = index; - } - - public String getIndexType() { - return indexType; - } - - public void setIndexType(String indexType) { - this.indexType = indexType; - } - - public long getCount() { - return count; - } - - public void setCount(long count) { - this.count = count; - } - - public long getTime() { - return time; - } - - public void setTime(long time) { - this.time = time; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Movement.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Movement.java deleted file mode 100644 index d6c30f54685d6208aedc00d4ede6d713998b740c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Movement.java +++ /dev/null @@ -1,314 +0,0 @@ -package org.duniter.elasticsearch.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.JsonIgnore; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.local.LocalEntity; -import static org.duniter.core.util.Preconditions.*; -import org.duniter.elasticsearch.dao.BlockDao; - -import java.io.Serializable; - -/** - * Created by blavenie on 29/11/16. - */ -public class Movement implements LocalEntity<String>, Serializable { - - - public static Builder newBuilder() { - return new Builder(); - } - - public static Builder newBuilder(BlockchainBlock block) { - return new Builder(block); - } - - public static final String PROPERTY_CURRENCY = "currency"; - public static final String PROPERTY_MEDIAN_TIME = "medianTime"; - - public static final String PROPERTY_VERSION = "version"; - public static final String PROPERTY_ISSUER = "issuer"; - public static final String PROPERTY_RECIPIENT = "recipient"; - public static final String PROPERTY_AMOUNT = "amount"; - public static final String PROPERTY_UNITBASE = "unitbase"; - public static final String PROPERTY_COMMENT = "comment"; - - public static final String PROPERTY_IS_UD = "isUD"; - public static final String PROPERTY_REFERENCE = "reference"; - - // ES identifier - private String id; - - // Property copied from Block - private String currency; - private Long medianTime; - - // Property copied from Tx - private int version; - private String issuer; - private String recipient; - private Long amount; - private Integer unitbase; - private String comment; - - // Specific properties - private boolean isUD; - private Reference reference; - - public Movement() { - super(); - } - - @Override - @JsonIgnore - public String getId() { - return id; - } - - @Override - @JsonIgnore - public void setId(String id) { - this.id = id; - } - - public String getCurrency() { - return currency; - } - - public void setCurrency(String currency) { - this.currency = currency; - } - - public int getVersion() { - return version; - } - - public void setVersion(int version) { - this.version = version; - } - - public String getIssuer() { - return issuer; - } - - public void setIssuer(String issuer) { - this.issuer = issuer; - } - - public Long getMedianTime() { - return medianTime; - } - - public void setMedianTime(Long medianTime) { - this.medianTime = medianTime; - } - - public String getRecipient() { - return recipient; - } - - public void setRecipient(String recipient) { - this.recipient = recipient; - } - - public Long getAmount() { - return amount; - } - - public void setAmount(Long amount) { - this.amount = amount; - } - - public Integer getUnitbase() { - return unitbase; - } - - public void setUnitbase(Integer unitbase) { - this.unitbase = unitbase; - } - - public String getComment() { - return comment; - } - - public void setComment(String comment) { - this.comment = comment; - } - - public boolean isUD() { - return isUD; - } - - public void setIsUD(boolean isUD) { - this.isUD = isUD; - } - - public Reference getReference() { - return reference; - } - - public void setReference(Reference reference) { - this.reference = reference; - } - - public static class Builder { - - private Movement result; - - private Builder() { - result = new Movement(); - } - - public Builder(BlockchainBlock block) { - this(); - setBlock(block); - } - - public Builder setBlock(BlockchainBlock block) { - result.setCurrency(block.getCurrency()); - result.setMedianTime(block.getMedianTime()); - result.setReference(new Reference(block.getCurrency(), BlockDao.TYPE, String.valueOf(block.getNumber()))); - setReferenceHash(block.getHash()); - return this; - } - - public Builder setReferenceHash(String hash) { - checkNotNull(result.getReference(), "No reference set. Please call setReference() first"); - result.getReference().setHash(hash); - return this; - } - - public Builder setRecipient(String recipient) { - result.setRecipient(recipient); - return this; - } - - public Builder setIssuer(String issuer) { - result.setIssuer(issuer); - return this; - } - - public Builder setVersion(int version) { - result.setVersion(version); - return this; - } - - public Builder setComment(String comment) { - result.setComment(comment); - return this; - } - - public Builder setAmount(long amount, int unitbase) { - result.setAmount(amount); - result.setUnitbase(unitbase); - return this; - } - - public Builder setIsUD(boolean isUD) { - result.setIsUD(isUD); - return this; - } - - public Movement build() { - checkNotNull(result); - checkNotNull(result.getAmount()); - checkNotNull(result.getUnitbase()); - checkNotNull(result.getRecipient()); - checkNotNull(result.getIssuer()); - checkNotNull(result.getCurrency()); - checkNotNull(result.getVersion()); - - return result; - } - } - - public static class Reference { - - public static final String PROPERTY_INDEX="index"; - public static final String PROPERTY_TYPE="type"; - public static final String PROPERTY_ID="id"; - public static final String PROPERTY_ANCHOR="anchor"; - public static final String PROPERTY_HASH="hash"; - - private String index; - - private String type; - - private String id; - - private String anchor; - - private String hash; - - public Reference() { - } - - public Reference(String index, String type, String id) { - this(index, type, id, null); - } - - public Reference(String index, String type, String id, String anchor) { - this.index = index; - this.type = type; - this.id = id; - this.anchor = anchor; - } - - public Reference(Reference another) { - this.index = another.getIndex(); - this.type = another.getType(); - this.id = another.getId(); - this.hash = another.getHash(); - this.anchor = another.getAnchor(); - } - - public String getIndex() { - return index; - } - - public String getType() { - return type; - } - - public String getId() { - return id; - } - - public String getAnchor() { - return anchor; - } - - public void setAnchor(String anchor) { - this.anchor = anchor; - } - - public String getHash() { - return hash; - } - - public void setHash(String hash) { - this.hash = hash; - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Movements.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Movements.java deleted file mode 100644 index 24cae9aebb714228e6c5784cb70dd0e75901f2d0..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Movements.java +++ /dev/null @@ -1,95 +0,0 @@ -package org.duniter.elasticsearch.model; - -/*- - * #%L - * Duniter4j :: Core Client API - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.google.common.base.Preconditions; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.util.CollectionUtils; - -import java.util.Arrays; -import java.util.List; -import java.util.Set; -import java.util.stream.Collectors; -import java.util.stream.Stream; - -import static org.duniter.core.client.model.bma.BlockchainBlocks.*; - -/** - * Created by blavenie on 26/04/17. - */ -public final class Movements { - - private Movements() { - // helper class - } - - public static Stream<Movement> stream(final BlockchainBlock block) { - Preconditions.checkNotNull(block); - - // No Tx - if (CollectionUtils.isEmpty(block.getTransactions())) { - return Stream.empty(); - } - - return Arrays.stream(block.getTransactions()) - .flatMap(tx -> Movements.streamFromTx(block, tx)); - } - - public static List<Movement> getMovements(final BlockchainBlock block) { - return stream(block).collect(Collectors.toList()); - } - - - /* -- Internal methods -- */ - - private static Stream<Movement> streamFromTx(final BlockchainBlock block, final BlockchainBlock.Transaction tx) { - - final List<TxInput> inputs = getTxInputs(tx); - final List<TxOutput> outputs = getTxOutputs(tx); - final Set<String> recipients = getTxRecipients(outputs); - - final long totalAmount = inputs.stream().mapToLong(input -> powBase(input.amount, input.unitbase)).sum(); - - return Arrays.stream(tx.getIssuers()) - .flatMap(issuer -> { - long issuerInputsAmount = getTxInputAmountByIssuer(inputs, issuer); - double issuerInputRatio = issuerInputsAmount / totalAmount; - - return recipients.stream() - // Compute the recipient amount - .map(recipient -> { - Double recipientAmount = getTxOutputAmountByIssuerAndRecipient(outputs, issuer, recipient) * issuerInputRatio; - return Movement.newBuilder(block) - .setAmount(recipientAmount.longValue(), 0/*unitbase*/) - .setIssuer(issuer) - .setRecipient(recipient) - .setVersion(tx.getVersion()) - .setComment(tx.getComment()) - .build(); - }) - // Exclude movements to itself (e.g. changes) - .filter(movement -> movement.getAmount() != 0); - }); - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Peer.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Peer.java deleted file mode 100644 index 917b08187f23f634a32efa296197d13f703c2cc0..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/Peer.java +++ /dev/null @@ -1,120 +0,0 @@ -package org.duniter.elasticsearch.model; - -/* - * #%L - * Duniter4j :: ElasticSearch Core 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 java.io.Serializable; - -public class Peer implements Serializable { - - private String currency; - private String host; - private int port; - private String path; - private String url; - - public Peer() { - // default constructor, need for de-serialization - } - - public Peer(String host, int port) { - this(host, port, null); - } - - public Peer(String host, int port, String path) { - this.host = host; - this.port = port; - this.url = initUrl(host, port, path); - this.path = path; - } - - public String getHost() { - return host; - } - - public int getPort() { - return port; - } - - public String getUrl() { - return url; - } - - public String getCurrency() { - return currency; - } - - public void setCurrency(String currency) { - this.currency = currency; - } - - public void setPort(int port) { - this.port = port; - this.url = initUrl(host, port, path); - } - - public void setHost(String host) { - this.host = host; - this.url = initUrl(host, port, path); - } - - public String getPath() { - return path; - } - - public void setPath(String path) { - this.path = path; - this.url = initUrl(host, port, path); - } - - public String toString() { - return new StringBuilder().append(host) - .append(":") - .append(port) - .toString(); - } - - @Override - public boolean equals(Object o) { - if (o == null) { - return false; - } - if (currency != null && o instanceof Peer) { - if (!currency.equals(((Peer) o).getCurrency())) { - return false; - } - if (!getUrl().equals(((Peer) o).getUrl())) { - return false; - } - } - return super.equals(o); - } - - /* -- Internal methods -- */ - - protected String initUrl(String host, int port, String path) { - return String.format("%s://%s:%s%s", - port == 443 ? "https" : "http", - host, port, - (path != null) ? path : ""); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SearchResponse.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SearchResponse.java deleted file mode 100644 index 1feb958eeff39b619cc909af1ef5fcd2e5225448..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SearchResponse.java +++ /dev/null @@ -1,91 +0,0 @@ -package org.duniter.elasticsearch.model; - -/* - * #%L - * Duniter4j :: 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.databind.JsonNode; -import org.apache.lucene.util.BytesRef; -import org.elasticsearch.common.bytes.BytesArray; -import org.elasticsearch.common.bytes.BytesReference; -import org.elasticsearch.common.io.stream.StreamInput; -import org.jboss.netty.buffer.ChannelBuffer; - -import java.io.IOException; -import java.io.OutputStream; -import java.io.Serializable; -import java.nio.channels.GatheringByteChannel; -import java.util.Iterator; - -/** - * Created by eis on 05/02/15. - */ -public class SearchResponse implements Serializable { - - protected JsonNode node; - - public SearchResponse(JsonNode response) { - this.node = response; - } - - public SearchHits getHits() { - return new SearchHits(node.get("hits")); - } - - public class SearchHits implements Iterator<SearchHit>{ - - protected JsonNode node; - private Iterator<JsonNode> hits; - SearchHits(JsonNode node) { - this.node = node; - this.hits = node == null ? null : node.get("hits").iterator(); - } - - public int getTotalHits() { - return node == null ? 0 : node.get("total").asInt(0); - } - - public boolean hasNext() { - return hits != null && hits.hasNext(); - } - public SearchHit next() { - return hits == null ? null : new SearchHit(hits.next()); - } - } - - public class SearchHit { - - private JsonNode node; - SearchHit(JsonNode node) { - this.node = node; - } - - public String getId() { - return node.get("_id").asText(); - } - - public JsonNode getSource() { - return node.get("_source"); - } - - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SearchScrollResponse.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SearchScrollResponse.java deleted file mode 100644 index 900813b2b3b386c205e876c5a4562544801b2ca2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SearchScrollResponse.java +++ /dev/null @@ -1,40 +0,0 @@ -package org.duniter.elasticsearch.model; - -/* - * #%L - * Duniter4j :: 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.databind.JsonNode; - -/** - * Created by eis on 05/02/15. - */ -public class SearchScrollResponse extends SearchResponse { - - public SearchScrollResponse(JsonNode response) { - super(response); - } - - public String getScrollId() { - return node.get("_scroll_id").asText(); - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SynchroExecution.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SynchroExecution.java deleted file mode 100644 index 29634a6891bf0547f7e45b029b95dbaa8b9bf919..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SynchroExecution.java +++ /dev/null @@ -1,81 +0,0 @@ -package org.duniter.elasticsearch.model; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.Record; - -public class SynchroExecution extends Record { - - public static final String PROPERTY_CURRENCY = "currency"; - public static final String PROPERTY_PEER = "peer"; - public static final String PROPERTY_API = "api"; - public static final String PROPERTY_RESULT = "result"; - public static final String PROPERTY_EXECUTION_TIME = "executionTime"; - - - private String currency; - private String peer; - private String api; - private Long executionTime; - private SynchroResult result; - - public String getCurrency() { - return currency; - } - - public void setCurrency(String currency) { - this.currency = currency; - } - - public String getPeer() { - return peer; - } - - public void setPeer(String peer) { - this.peer = peer; - } - - public SynchroResult getResult() { - return result; - } - - public void setResult(SynchroResult result) { - this.result = result; - } - - public String getApi() { - return api; - } - - public void setApi(String api) { - this.api = api; - } - - public Long getExecutionTime() { - return executionTime; - } - - public void setExecutionTime(Long executionTime) { - this.executionTime = executionTime; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SynchroResult.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SynchroResult.java deleted file mode 100644 index d36211c9d9cc051abc6a9c24c79396ea04f2f714..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/model/SynchroResult.java +++ /dev/null @@ -1,146 +0,0 @@ -package org.duniter.elasticsearch.model; - -/* - * #%L - * Duniter4j :: ElasticSearch Core plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.annotation.JsonIgnore; - -import java.io.Serializable; -import java.util.HashMap; -import java.util.Map; - -/** - * Created by blavenie on 30/12/16. - */ -public class SynchroResult implements Serializable { - - public static final String PROPERTY_INSERTS = "inserts"; - public static final String PROPERTY_UPDATES = "updates"; - public static final String PROPERTY_DELETES = "deletes"; - public static final String PROPERTY_INVALID_SIGNATURES = "invalidSignatures"; - - private long insertTotal = 0; - private long updateTotal = 0; - private long deleteTotal = 0; - private long invalidSignatureTotal = 0; - private long invalidTimeTotal = 0; - private Map<String, Long> insertHits = new HashMap<>(); - private Map<String, Long> updateHits = new HashMap<>(); - private Map<String, Long> deleteHits = new HashMap<>(); - private Map<String, Long> invalidSignatureHits = new HashMap<>(); - private Map<String, Long> invalidTimeHits = new HashMap<>(); - - public void addInserts(String index, String type, long nbHits) { - insertHits.put(index + "/" + type, getInserts(index, type) + nbHits); - insertTotal += nbHits; - } - - public void addUpdates(String index, String type, long nbHits) { - updateHits.put(index + "/" + type, getUpdates(index, type) + nbHits); - updateTotal += nbHits; - } - - public void addDeletes(String index, String type, long nbHits) { - deleteHits.put(index + "/" + type, getDeletes(index, type) + nbHits); - deleteTotal += nbHits; - } - - public void addInvalidSignatures(String index, String type, long nbHits) { - invalidSignatureHits.put(index + "/" + type, getDeletes(index, type) + nbHits); - invalidSignatureTotal += nbHits; - } - - - public void addInvalidTimes(String index, String type, long nbHits) { - invalidTimeHits.put(index + "/" + type, getDeletes(index, type) + nbHits); - invalidTimeTotal += nbHits; - } - - @JsonIgnore - public long getInserts(String index, String type) { - return insertHits.getOrDefault(index + "/" + type, 0l); - } - - @JsonIgnore - public long getUpdates(String index, String type) { - return updateHits.getOrDefault(index + "/" + type, 0l); - } - - @JsonIgnore - public long getInvalidSignatures(String index, String type) { - return invalidSignatureHits.getOrDefault(index + "/" + type, 0l); - } - - @JsonIgnore - public long getDeletes(String index, String type) { - return deleteHits.getOrDefault(index + "/" + type, 0l); - } - - public long getInserts() { - return insertTotal; - } - - public long getUpdates() { - return updateTotal; - } - - public long getDeletes() { - return deleteTotal; - } - - public long getInvalidSignatures() { - return invalidSignatureTotal; - } - - public long getInvalidTimes() { - return invalidTimeTotal; - } - - @JsonIgnore - public long getTotal() { - return insertTotal + updateTotal + deleteTotal; - } - - public void setInserts(long inserts) { - this.insertTotal = inserts; - } - public void setDeletes(long deletes) { - this.deleteTotal = deletes; - } - public void setUpdates(long updates) { - this.updateTotal = updates; - } - public void setInvalidSignatures(long invalidSignatures) { - this.invalidSignatureTotal = invalidSignatures; - } - public void setInvalidTimes(long invalidTimes) { - this.invalidTimeTotal = invalidTimes; - } - - public String toString() { - return new StringBuilder() - .append("inserts [").append(insertTotal).append("]") - .append(", updates [").append(updateTotal).append("]") - .append(", deletes [").append(deleteTotal).append("]") - .toString(); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostIndexAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostIndexAction.java deleted file mode 100644 index 9738482120cdca29cdf9ba8dc70cdba9088c4537..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostIndexAction.java +++ /dev/null @@ -1,83 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.core.exception.BusinessException; -import org.duniter.elasticsearch.exception.DuniterElasticsearchException; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -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 static org.elasticsearch.rest.RestRequest.Method.GET; -import static org.elasticsearch.rest.RestRequest.Method.POST; -import static org.elasticsearch.rest.RestStatus.OK; - -public abstract class AbstractRestPostIndexAction extends BaseRestHandler { - - private final ESLogger log; - - private final JsonIndexer indexer; - - - public AbstractRestPostIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - String indexName, - String typeName, - JsonIndexer indexer) { - super(settings, controller, client); - log = Loggers.getLogger("duniter.rest" + indexName, settings, String.format("[%s]", indexName)); - controller.registerHandler(POST, - String.format("/%s/%s", indexName, typeName), - this); - securityController.allowIndexType(POST, indexName, typeName); - securityController.allowIndexType(GET, indexName, typeName); - this.indexer = indexer; - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel channel, Client client) throws Exception { - - try { - String id = indexer.handleJson(request.content().toUtf8()); - channel.sendResponse(new BytesRestResponse(OK, id)); - } - catch(DuniterElasticsearchException | BusinessException e) { - log.error(e.getMessage(), e); - channel.sendResponse(new XContentThrowableRestResponse(request, e)); - } - catch(Exception e) { - log.error(e.getMessage(), e); - } - } - - - public interface JsonIndexer { - String handleJson(String json) throws DuniterElasticsearchException, BusinessException; - } - - - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostMarkAsReadAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostMarkAsReadAction.java deleted file mode 100644 index f42b2712b7beb942200d831a91f2afd819f3b1d3..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostMarkAsReadAction.java +++ /dev/null @@ -1,82 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.core.exception.BusinessException; -import org.duniter.elasticsearch.exception.DuniterElasticsearchException; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -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 static org.elasticsearch.rest.RestRequest.Method.POST; -import static org.elasticsearch.rest.RestStatus.OK; - -public abstract class AbstractRestPostMarkAsReadAction extends BaseRestHandler { - - private final ESLogger log; - - private final JsonReadUpdater updater; - - - public AbstractRestPostMarkAsReadAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - String indexName, - String typeName, - JsonReadUpdater updater) { - super(settings, controller, client); - log = Loggers.getLogger("duniter.rest" + indexName, settings, String.format("[%s]", indexName)); - controller.registerHandler(POST, - String.format("/%s/%s/{id}/_read", indexName, typeName), - this); - securityController.allow(POST, String.format("/%s/%s/[^/]+/_read", indexName, typeName)); - this.updater = updater; - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel restChannel, Client client) throws Exception { - String id = request.param("id"); - - try { - updater.handleSignature(id, request.content().toUtf8()); - restChannel.sendResponse(new BytesRestResponse(OK, id)); - } - catch(DuniterElasticsearchException | BusinessException e) { - log.error(e.getMessage(), e); - restChannel.sendResponse(new XContentThrowableRestResponse(request, e)); - } - catch(Exception e) { - log.error(e.getMessage(), e); - } - } - - - public interface JsonReadUpdater { - void handleSignature(String id, String signature) throws DuniterElasticsearchException, BusinessException; - } - - - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostUpdateAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostUpdateAction.java deleted file mode 100644 index 2daa5387110ac813a01e7a9b7b2bc2bcf88fb491..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/AbstractRestPostUpdateAction.java +++ /dev/null @@ -1,88 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.core.exception.BusinessException; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.exception.AccessDeniedException; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.exception.DuniterElasticsearchException; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.*; - -import static org.elasticsearch.rest.RestRequest.Method.POST; -import static org.elasticsearch.rest.RestStatus.OK; - -public abstract class AbstractRestPostUpdateAction extends BaseRestHandler { - - private final ESLogger log; - - private final JsonUpdater updater; - - - public AbstractRestPostUpdateAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - String indexName, - String typeName, - JsonUpdater updater) { - super(settings, controller, client); - log = Loggers.getLogger("duniter.rest." + indexName, settings, String.format("[%s]", indexName)); - controller.registerHandler(POST, - String.format("/%s/%s/{id}/_update", indexName, typeName), - this); - securityController.allowIndexType(POST, indexName, typeName); - this.updater = updater; - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel restChannel, Client client) throws Exception { - String id = request.param("id"); - - try { - if (StringUtils.isBlank(id)) { - throw new AccessDeniedException("Bad request (missing id in path)"); - } - updater.handleJson(id, request.content().toUtf8()); - restChannel.sendResponse(new BytesRestResponse(OK, id)); - } - catch(DuniterElasticsearchException | BusinessException e) { - log.error(e.getMessage(), e); - restChannel.sendResponse(new XContentThrowableRestResponse(request, e)); - } - catch(Exception e) { - log.error(e.getMessage(), e); - } - } - - - public interface JsonUpdater { - void handleJson(String id, String json) throws DuniterElasticsearchException, BusinessException; - } - - - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/RestModule.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/RestModule.java deleted file mode 100644 index fa523902afec82c88d27b34c4019b94cac7d168f..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/RestModule.java +++ /dev/null @@ -1,55 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.attachment.RestImageAttachmentAction; -import org.duniter.elasticsearch.rest.currency.RestCurrencyIndexAction; -import org.duniter.elasticsearch.rest.node.RestNodeSummaryGetAction; -import org.duniter.elasticsearch.rest.security.RestSecurityAuthAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.rest.security.RestSecurityFilter; -import org.duniter.elasticsearch.rest.security.RestSecurityGetChallengeAction; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class RestModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // Common - bind(RestNodeSummaryGetAction.class).asEagerSingleton(); - - // Authentication & Security - bind(RestSecurityGetChallengeAction.class).asEagerSingleton(); - bind(RestSecurityAuthAction.class).asEagerSingleton(); - bind(RestSecurityFilter.class).asEagerSingleton(); - bind(RestSecurityController.class).asEagerSingleton(); - - // Attachment as image - bind(RestImageAttachmentAction.class).asEagerSingleton(); - - // Currency - //bind(RestCurrencyIndexAction.class).asEagerSingleton(); - - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/RestXContentBuilder.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/RestXContentBuilder.java deleted file mode 100644 index 5aa0977d15f04b155df2c5fd1a9d0b3332b68c01..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/RestXContentBuilder.java +++ /dev/null @@ -1,61 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.elasticsearch.common.io.stream.BytesStreamOutput; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.common.xcontent.XContentType; -import org.elasticsearch.rest.RestRequest; - -import java.io.IOException; - -public class RestXContentBuilder { - - public static XContentBuilder restContentBuilder(RestRequest request) throws IOException { - XContentType contentType = XContentType.fromRestContentType(request.param("format", request.header("Content-Type"))); - if (contentType == null) { - // default to JSON - contentType = XContentType.JSON; - } - XContentBuilder builder = new XContentBuilder(XContentFactory.xContent(contentType), - new BytesStreamOutput()); - if (request.paramAsBoolean("pretty", false)) { - builder.prettyPrint().lfAtEnd(); - } - String casing = request.param("case"); - if (casing != null && "camelCase".equals(casing)) { - builder.fieldCaseConversion(XContentBuilder.FieldCaseConversion.CAMELCASE); - } else { - // we expect all REST interfaces to write results in underscore casing, so - // no need for double casing - builder.fieldCaseConversion(XContentBuilder.FieldCaseConversion.NONE); - } - return builder; - } - - public static XContentBuilder emptyBuilder(RestRequest request) throws IOException { - return restContentBuilder(request).startObject().endObject(); - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/XContentRestResponse.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/XContentRestResponse.java deleted file mode 100644 index 102573a69cc6ad73a7ea15a23b2847719e9424f9..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/XContentRestResponse.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.rest.BytesRestResponse; -import org.elasticsearch.rest.RestRequest; -import org.elasticsearch.rest.RestStatus; - -import java.io.IOException; - -public class XContentRestResponse extends BytesRestResponse { - - public XContentRestResponse(RestRequest request, RestStatus status, XContentBuilder builder) throws IOException { - super(status, builder); - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/XContentThrowableRestResponse.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/XContentThrowableRestResponse.java deleted file mode 100644 index 4f498f4040c007324b37408de064afad0a8c8ea7..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/XContentThrowableRestResponse.java +++ /dev/null @@ -1,84 +0,0 @@ -package org.duniter.elasticsearch.rest; - -/* - * #%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.exception.DuniterElasticsearchException; -import org.elasticsearch.ElasticsearchException; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.rest.RestRequest; -import org.elasticsearch.rest.RestStatus; - -import java.io.IOException; - -import static org.elasticsearch.ExceptionsHelper.detailedMessage; - -public class XContentThrowableRestResponse extends XContentRestResponse { - - public XContentThrowableRestResponse(RestRequest request, Throwable t) throws IOException { - this(request, ((t instanceof ElasticsearchException) ? ((ElasticsearchException) t).status() : ((t instanceof DuniterElasticsearchException) ? ((DuniterElasticsearchException) t).status() : RestStatus.INTERNAL_SERVER_ERROR)), t); - } - - public XContentThrowableRestResponse(RestRequest request, RestStatus status, Throwable t) throws IOException { - super(request, status, convert(request, status, t)); - } - - private static XContentBuilder convert(RestRequest request, RestStatus status, Throwable t) throws IOException { - XContentBuilder builder = RestXContentBuilder.restContentBuilder(request).startObject() - .field("error", detailedMessage(t)) - .field("status", status.getStatus()); - if (t != null && request.paramAsBoolean("error_trace", false)) { - builder.startObject("error_trace"); - boolean first = true; - while (t != null) { - if (!first) { - builder.startObject("cause"); - } - buildThrowable(t, builder); - if (!first) { - builder.endObject(); - } - t = t.getCause(); - first = false; - } - builder.endObject(); - } - builder.endObject(); - return builder; - } - - private static void buildThrowable(Throwable t, XContentBuilder builder) throws IOException { - builder.field("message", t.getMessage()); - for (StackTraceElement stElement : t.getStackTrace()) { - builder.startObject("at") - .field("class", stElement.getClassName()) - .field("method", stElement.getMethodName()); - if (stElement.getFileName() != null) { - builder.field("file", stElement.getFileName()); - } - if (stElement.getLineNumber() >= 0) { - builder.field("line", stElement.getLineNumber()); - } - builder.endObject(); - } - } -} \ No newline at end of file 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 deleted file mode 100644 index e46a5926a10e170d5e51eadbb46875330ee65794..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/attachment/RestImageAttachmentAction.java +++ /dev/null @@ -1,111 +0,0 @@ -package org.duniter.elasticsearch.rest.attachment; - -/* - * #%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.core.util.StringUtils; -import org.elasticsearch.action.get.GetRequest; -import org.elasticsearch.action.get.GetResponse; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.Base64; -import org.elasticsearch.common.bytes.BytesArray; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.internal.Join; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.*; -import org.elasticsearch.rest.action.support.RestResponseListener; -import org.elasticsearch.search.fetch.source.FetchSourceContext; - -import java.util.Arrays; -import java.util.Map; - -import static org.elasticsearch.rest.RestStatus.OK; - -public class RestImageAttachmentAction extends BaseRestHandler { - - @Inject - public RestImageAttachmentAction(Settings settings, RestController controller, Client client) { - super(settings, controller, client); - controller.registerHandler(RestRequest.Method.GET, "/{index}/{type}/{id}/_image/{field}", this); - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel channel, Client client) throws Exception { - String index = request.param("index"); - String type = request.param("type"); - String id = request.param("id"); - String paramField = request.param("field"); - String[] fieldParts = paramField.split("\\."); - String extension = null; - if (fieldParts.length >= 2) { - extension = fieldParts[fieldParts.length-1]; - paramField = Join.join(".", Arrays.copyOf(fieldParts, fieldParts.length-1)); - } - - final String field = paramField; - final String expectedContentType = "image/" + extension; - - GetRequest getRequest = new GetRequest(index, type, id) - .fields(field) - .fetchSourceContext(FetchSourceContext.FETCH_SOURCE) - .realtime(true); - - client.get(getRequest, new RestResponseListener<GetResponse>(channel) { - @Override - public RestResponse buildResponse(GetResponse response) throws Exception { - if (response.getSource() == null || !response.getSource().containsKey(field)) { - return new BytesRestResponse(RestStatus.BAD_REQUEST, String.format("Field [%s] not exists.", field)); - } - Object value = response.getSource().get(field); - if (!(value instanceof Map)) { - return new BytesRestResponse(RestStatus.BAD_REQUEST, String.format("Field [%s] is not an attachment type.", field)); - } - Map<String, String> attachment = (Map<String, String>)value; - String contentType = attachment.get("_content_type"); - if (StringUtils.isBlank(contentType)) { - return new BytesRestResponse(RestStatus.BAD_REQUEST, String.format("Field [%s] not contains key [_content_type].", field)); - } - - if (!expectedContentType.equals(contentType)) { - return new BytesRestResponse(RestStatus.BAD_REQUEST, String.format("File extension not compatible with attachment content type [%s]", contentType)); - } - - return new BytesRestResponse(OK, - contentType, - new BytesArray(Base64.decode(attachment.get("_content")))); - } - }); - } - - - 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/currency/RestCurrencyIndexAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/currency/RestCurrencyIndexAction.java deleted file mode 100644 index 674b7e720af4139b95600cb7f05b832b30ed3e3e..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/currency/RestCurrencyIndexAction.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.rest.currency; - -/* - * #%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.core.exception.TechnicalException; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.service.CurrencyService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -/** - * A rest to post a request to process a new currency/peer. - * - */ -public class RestCurrencyIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestCurrencyIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, CurrencyService currencyService) { - super(settings, controller, client, securityController, - CurrencyService.INDEX, CurrencyService.RECORD_TYPE, - (json) -> { - throw new TechnicalException("Not implemented yet");/*currencyService.indexCurrencyFromJson(json)*/ - }); - } - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/node/RestNodeSummaryGetAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/node/RestNodeSummaryGetAction.java deleted file mode 100644 index d90a550103ab0a7baa0e57269bdf6cbcedaf6011..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/node/RestNodeSummaryGetAction.java +++ /dev/null @@ -1,83 +0,0 @@ -package org.duniter.elasticsearch.rest.node; - -/* - * #%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.core.client.config.Configuration; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.XContentRestResponse; -import org.duniter.elasticsearch.rest.XContentThrowableRestResponse; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.service.CurrencyService; -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.elasticsearch.rest.*; - -import java.io.IOException; - -/** - * A rest to post a request to process a new currency/peer. - * - */ -public class RestNodeSummaryGetAction extends BaseRestHandler { - - @Inject - public RestNodeSummaryGetAction(Settings settings, RestController controller, Client client, RestSecurityController securityController) { - super(settings, controller, client); - securityController.allow(RestRequest.Method.GET, "/node/summary"); - controller.registerHandler(RestRequest.Method.GET, "/node/summary", this); - } - - @Override - protected void handleRequest(RestRequest request, RestChannel channel, Client client) throws Exception { - XContentBuilder content = createSummary(); - channel.sendResponse(new XContentRestResponse(request, RestStatus.OK, content)); - } - - - public XContentBuilder createSummary() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject("duniter") - - // software - .field("software", "duniter4j-elasticsearch") - - // version - .field("version", Configuration.instance().getVersion().toString()) - - // status - .field("status", RestStatus.OK.getStatus()) - - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while generating JSON for [/node/summary]: %s", ioe.getMessage()), ioe); - } - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RedirectionRestRequest.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RedirectionRestRequest.java deleted file mode 100644 index 263c0cd613b44d4be208c6fb31e1e855dcfbce64..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RedirectionRestRequest.java +++ /dev/null @@ -1,159 +0,0 @@ -package org.duniter.elasticsearch.rest.security; - -/* - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.common.Nullable; -import org.elasticsearch.common.bytes.BytesReference; -import org.elasticsearch.common.unit.ByteSizeValue; -import org.elasticsearch.common.unit.TimeValue; -import org.elasticsearch.rest.RestRequest; - -import java.net.SocketAddress; -import java.util.Map; - -/** - * Created by blavenie on 30/12/16. - */ -public class RedirectionRestRequest extends RestRequest { - - private final RestRequest delegate; - private String path; - - public RedirectionRestRequest(RestRequest delegate, String path) { - super(); - this.delegate = delegate; - this.path = path; - } - - @Override - public Method method() { - return delegate.method(); - } - - @Override - public String uri() { - return delegate.uri(); - } - - @Override - public String rawPath() { - return delegate.rawPath(); - } - - @Override - public boolean hasContent() { - return delegate.hasContent(); - } - - @Override - public BytesReference content() { - return delegate.content(); - } - - @Override - public String header(String name) { - return delegate.header(name); - } - - @Override - public Iterable<Map.Entry<String, String>> headers() { - return delegate.headers(); - } - - @Override - @Nullable - public SocketAddress getRemoteAddress() { - return delegate.getRemoteAddress(); - } - - @Override - @Nullable - public SocketAddress getLocalAddress() { - return delegate.getLocalAddress(); - } - - @Override - public boolean hasParam(String key) { - return delegate.hasParam(key); - } - - @Override - public String param(String key) { - return delegate.param(key); - } - - @Override - public Map<String, String> params() { - return delegate.params(); - } - - @Override - public float paramAsFloat(String key, float defaultValue) { - return delegate.paramAsFloat(key, defaultValue); - } - - @Override - public int paramAsInt(String key, int defaultValue) { - return delegate.paramAsInt(key, defaultValue); - } - - @Override - public long paramAsLong(String key, long defaultValue) { - return delegate.paramAsLong(key, defaultValue); - } - - @Override - public boolean paramAsBoolean(String key, boolean defaultValue) { - return delegate.paramAsBoolean(key, defaultValue); - } - - @Override - public Boolean paramAsBoolean(String key, Boolean defaultValue) { - return delegate.paramAsBoolean(key, defaultValue); - } - - @Override - public TimeValue paramAsTime(String key, TimeValue defaultValue) { - return delegate.paramAsTime(key, defaultValue); - } - - @Override - public ByteSizeValue paramAsSize(String key, ByteSizeValue defaultValue) { - return delegate.paramAsSize(key, defaultValue); - } - - @Override - public String[] paramAsStringArray(String key, String[] defaultValue) { - return delegate.paramAsStringArray(key, defaultValue); - } - - @Override - public String[] paramAsStringArrayOrEmptyIfAll(String key) { - return delegate.paramAsStringArrayOrEmptyIfAll(key); - } - - @Override - public String param(String key, String defaultValue) { - return delegate.param(key, defaultValue); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityAuthAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityAuthAction.java deleted file mode 100644 index ad1e92ae6ea1cdd62f50d08c2aa036fb8f7815d0..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityAuthAction.java +++ /dev/null @@ -1,88 +0,0 @@ -package org.duniter.elasticsearch.rest.security; - -/* - * #%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 com.fasterxml.jackson.databind.ObjectMapper; -import org.duniter.core.client.service.ServiceLocator; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.security.challenge.ChallengeMessageStore; -import org.duniter.elasticsearch.security.token.SecurityTokenStore; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.*; - -import static org.elasticsearch.rest.RestRequest.Method.POST; -import static org.elasticsearch.rest.RestStatus.FORBIDDEN; -import static org.elasticsearch.rest.RestStatus.OK; - -public class RestSecurityAuthAction extends BaseRestHandler { - - private static final ESLogger log = ESLoggerFactory.getLogger(RestSecurityAuthAction.class.getName()); - - private ChallengeMessageStore challengeMessageStore; - private SecurityTokenStore securityTokenStore; - private ObjectMapper objectMapper; - - @Inject - public RestSecurityAuthAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - ChallengeMessageStore challengeMessageStore, - SecurityTokenStore securityTokenStore) { - super(settings, controller, client); - this.challengeMessageStore = challengeMessageStore; - this.securityTokenStore = securityTokenStore; - this.objectMapper = new ObjectMapper(); - controller.registerHandler(POST, "/auth", this); - securityController.allow(POST, "/auth"); - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel restChannel, Client client) throws Exception { - - AuthData authData = objectMapper.readValue(request.content().toUtf8(), AuthData.class); - - // TODO Authorization: Basic instead ? - - if (StringUtils.isNotBlank(authData.pubkey)) { - if (challengeMessageStore.validateChallenge(authData.challenge)) { - boolean signatureOK = ServiceLocator.instance().getCryptoService().verify(authData.challenge, authData.signature, authData.pubkey); - if (signatureOK) { - String token = securityTokenStore.createNewToken(authData.challenge, authData.signature, authData.pubkey); - restChannel.sendResponse(new BytesRestResponse(OK, token)); - return; - } - } - } - - restChannel.sendResponse(new BytesRestResponse(FORBIDDEN, Boolean.FALSE.toString())); - } - - class AuthData { - public String pubkey; - public String challenge; - public String signature; - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityController.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityController.java deleted file mode 100644 index f7817c552b1e96db767f4e48a053eac5f39563bc..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityController.java +++ /dev/null @@ -1,145 +0,0 @@ -package org.duniter.elasticsearch.rest.security; - -/* - * #%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.PluginSettings; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestRequest; - -import java.util.HashMap; -import java.util.Map; -import java.util.Set; -import java.util.TreeSet; - -/** - * Created by blavenie on 11/10/16. - */ -public class RestSecurityController extends AbstractLifecycleComponent<RestSecurityController> { - - private final ESLogger log; - - private boolean enable; - private boolean trace; - - private Map<RestRequest.Method, Set<String>> allowRulesByMethod; - - @Inject - public RestSecurityController(Settings settings, PluginSettings pluginSettings) { - super(settings); - this.log = Loggers.getLogger("duniter.security", settings, new String[0]); - this.trace = log.isTraceEnabled(); - this.enable = pluginSettings.enableSecurity(); - this.allowRulesByMethod = new HashMap<>(); - if (!enable) { - log.warn("/!\\ Security has been disable using option [duniter.security.enable]. This is NOT recommended in production !"); - } - } - - public RestSecurityController allowIndexType(RestRequest.Method method, String index, String type) { - allow(method, String.format("/%s/%s(/.*)?", index, type)); - return this; - } - - public RestSecurityController allowPostSearchIndexType(String index, String type) { - allow(RestRequest.Method.POST, String.format("/%s/%s/_search", index, type)); - return this; - } - - public RestSecurityController allowImageAttachment(String index, String type, String field) { - allow(RestRequest.Method.GET, String.format("/%s/%s/[^/]+/_image/%s.*", index, type, field)); - return this; - } - - public RestSecurityController allow(RestRequest.Method method, String regexPath) { - Set<String> allowRules = allowRulesByMethod.computeIfAbsent(method, k -> new TreeSet<>()); - - if (!allowRules.contains(regexPath)) { - allowRules.add(regexPath); - } - return this; - } - - public boolean isAllow(RestRequest request) { - if (!this.enable) return true; - - RestRequest.Method method = request.method(); - String path = request.path(); - - Set<String> allowRules = allowRulesByMethod.get(request.method()); - - // Trace mode - if (trace) { - log.trace(String.format("Checking rules for %s request [%s]...", method, path)); - if (allowRules == null) { - log.trace(String.format("No matching rules for %s request [%s]: reject", method, path)); - } - else { - boolean found = false; - for (String allowRule : allowRules) { - log.trace(String.format(" - Trying against rule [%s] for %s requests: not match", allowRule, method)); - if (path.matches(allowRule)) { - log.trace(String.format("Find matching rule [%s] for %s request [%s]: allow", allowRule, method, path)); - found = true; - break; - } - } - if (!found) { - log.trace(String.format("No matching rules for %s request [%s]: reject", method, path)); - } - } - } - - // Check if allow - if (allowRules != null) { - for (String allowRule : allowRules) { - if (path.matches(allowRule)) { - return true; - } - } - } - return false; - } - - @Override - protected void doStart() { - - } - - @Override - protected void doStop() { - - } - - @Override - protected void doClose() { - - } - - /* -- Internal method -- */ - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityFilter.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityFilter.java deleted file mode 100644 index c38ed56e8efe701e88a2a78a4690a6b3c24d1166..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityFilter.java +++ /dev/null @@ -1,76 +0,0 @@ -package org.duniter.elasticsearch.rest.security; - -/* - * #%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.PluginSettings; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.rest.*; - -import static org.elasticsearch.rest.RestStatus.FORBIDDEN; - -public class RestSecurityFilter extends RestFilter { - - private final ESLogger logger; - - private RestSecurityController securityController; - private final boolean debug; - - @Inject - public RestSecurityFilter(PluginSettings pluginSettings, RestController controller, RestSecurityController securityController) { - super(); - logger = Loggers.getLogger("duniter.security", pluginSettings.getSettings(), new String[0]); - if (pluginSettings.enableSecurity()) { - logger.info("Enable security on all duniter4j indices"); - controller.registerFilter(this); - } - this.securityController = securityController; - this.debug = logger.isDebugEnabled(); - } - - @Override - public void process(RestRequest request, RestChannel channel, RestFilterChain filterChain) throws Exception { - - if (request.path().contains("message/record")) { - logger.debug("---------------- Redirection ?!"); - - filterChain.continueProcessing(new RedirectionRestRequest(request, "message/inbox"), channel); - return; - } - - if (securityController.isAllow(request)) { - if (debug) { - logger.debug(String.format("Allow %s request [%s]", request.method().name(), request.path())); - } - - filterChain.continueProcessing(request, channel); - } - - else { - logger.warn(String.format("Refused %s request to [%s]", request.method().name(), request.path())); - channel.sendResponse(new BytesRestResponse(FORBIDDEN)); - } - } - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityGetChallengeAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityGetChallengeAction.java deleted file mode 100644 index 9b27b92a95e39ba4b9bf4baacf33be58bae52bb4..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/security/RestSecurityGetChallengeAction.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.rest.security; - -/* - * #%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.security.challenge.ChallengeMessageStore; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.*; - -import static org.elasticsearch.rest.RestRequest.Method.GET; -import static org.elasticsearch.rest.RestStatus.OK; - -public class RestSecurityGetChallengeAction extends BaseRestHandler { - - private ChallengeMessageStore challengeMessageStore; - - @Inject - public RestSecurityGetChallengeAction(Settings settings, RestController controller, Client client, ChallengeMessageStore challengeMessageStore) { - super(settings, controller, client); - this.challengeMessageStore = challengeMessageStore; - controller.registerHandler(GET, "/auth", this); - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel restChannel, Client client) throws Exception { - restChannel.sendResponse(new BytesRestResponse(OK, challengeMessageStore.createNewChallenge())); - } - -} \ 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 deleted file mode 100644 index 6dd3dd5a8d41c80a6aa9b6733b45de31e1c575e9..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/rest/share/AbstractRestShareLinkAction.java +++ /dev/null @@ -1,99 +0,0 @@ -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.ST; -import org.stringtemplate.v4.STGroup; - -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; - private String urlPattern; - - public AbstractRestShareLinkAction(Settings settings, RestController controller, Client client, - String indexName, - String typeName, - String shareBaseUrl, - OGDataResolver resolver - ) { - super(settings, controller, client); - log = Loggers.getLogger("duniter.rest." + indexName, settings, String.format("[%s]", indexName)); - - String pathPattern = String.format("/%s/%s/%s/_share", indexName, typeName, "%s"); - controller.registerHandler(GET, - String.format(pathPattern, "{id}"), - this); - this.urlPattern = (shareBaseUrl != null ? shareBaseUrl : "") + pathPattern; - 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", String.format(urlPattern, id)); - template.add("redirectUrl", data.url); - template.add("locale", data.locale); - template.add("imageHeight", data.imageHeight); - template.add("imageWidth", data.imageWidth); - 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/script/BlockchainTxCountScriptFactory.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/script/BlockchainTxCountScriptFactory.java deleted file mode 100644 index 31176d21d5f97006c2265a312843b2cf5ca68420..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/script/BlockchainTxCountScriptFactory.java +++ /dev/null @@ -1,53 +0,0 @@ -package org.duniter.elasticsearch.script; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.common.Nullable; -import org.elasticsearch.script.AbstractFloatSearchScript; -import org.elasticsearch.script.ExecutableScript; -import org.elasticsearch.script.NativeScriptFactory; - -import java.util.List; -import java.util.Map; - -public class BlockchainTxCountScriptFactory implements NativeScriptFactory { - - @Override - public ExecutableScript newScript(@Nullable Map<String, Object> params) { - return new BlockchainTxCountScript(); - } - - @Override - public boolean needsScores() { - return false; - } - - public class BlockchainTxCountScript extends AbstractFloatSearchScript { - - @Override - public float runAsFloat() { - Object a = source().get("transactions"); - return a != null ? ((List)a).size() : 0; - } - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/SecurityModule.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/SecurityModule.java deleted file mode 100644 index ed497aeca2374342cace3ab640cfffc31431ad8d..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/SecurityModule.java +++ /dev/null @@ -1,36 +0,0 @@ -package org.duniter.elasticsearch.security; - -/* - * #%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.security.challenge.ChallengeMessageStore; -import org.duniter.elasticsearch.security.token.SecurityTokenStore; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class SecurityModule extends AbstractModule implements Module { - - @Override protected void configure() { - bind(ChallengeMessageStore.class).asEagerSingleton(); - bind(SecurityTokenStore.class).asEagerSingleton(); - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/challenge/ChallengeMessageStore.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/challenge/ChallengeMessageStore.java deleted file mode 100644 index 2847036da6e6a25457ee4e560598be482de817e0..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/challenge/ChallengeMessageStore.java +++ /dev/null @@ -1,90 +0,0 @@ -package org.duniter.elasticsearch.security.challenge; - -/* - * #%L - * duniter4j :: UI Wicket - * %% - * 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.core.util.Preconditions; -import com.google.common.cache.CacheBuilder; -import com.google.common.cache.CacheLoader; -import com.google.common.cache.LoadingCache; -import org.duniter.core.util.ObjectUtils; -import org.duniter.core.util.StringUtils; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.settings.Settings; - -import java.util.concurrent.TimeUnit; - -/** - * Created by blavenie on 06/01/16. - */ -public class ChallengeMessageStore { - - private static final ESLogger log = ESLoggerFactory.getLogger(ChallengeMessageStore.class.getName()); - - - private String prefix; - private long validityDurationInSeconds; - private LoadingCache<String, String> chalengeMessageCache; - - @Inject - public ChallengeMessageStore(Settings settings) { - this.prefix = settings.get("duniter4j.auth.challenge.prefix", "duniter4j-challenge-"); - this.validityDurationInSeconds = settings.getAsInt("duniter4j.auth.challengeValidityDuration", 10); - this.chalengeMessageCache = initGeneratedMessageCache(); - } - - public boolean validateChallenge(String challenge) { - Preconditions.checkArgument(StringUtils.isNotBlank(challenge)); - - String storedChallenge = chalengeMessageCache.getIfPresent(challenge); - - // if no value in cache => maybe challenge expired - return ObjectUtils.equals(storedChallenge, challenge); - } - - public String createNewChallenge() { - String challenge = newChallenge(); - chalengeMessageCache.put(challenge, challenge); - return newChallenge(); - } - - /* -- internal methods -- */ - - protected String newChallenge() { - return String.valueOf(prefix + System.currentTimeMillis() * System.currentTimeMillis()); - } - - - protected LoadingCache<String, String> initGeneratedMessageCache() { - return CacheBuilder.newBuilder() - .expireAfterWrite(validityDurationInSeconds, TimeUnit.SECONDS) - .build(new CacheLoader<String, String>() { - @Override - public String load(String challenge) throws Exception { - // not used. Filled manually - return null; - } - }); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/token/SecurityTokenStore.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/token/SecurityTokenStore.java deleted file mode 100644 index 65bd3da5b04ffd5e32301dfa200223ffcaed55be..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/security/token/SecurityTokenStore.java +++ /dev/null @@ -1,88 +0,0 @@ -package org.duniter.elasticsearch.security.token; - -/* - * #%L - * duniter4j :: UI Wicket - * %% - * 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.core.util.Preconditions; -import com.google.common.cache.CacheBuilder; -import com.google.common.cache.CacheLoader; -import com.google.common.cache.LoadingCache; -import org.duniter.core.util.ObjectUtils; -import org.duniter.core.util.StringUtils; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.settings.Settings; - -import java.util.concurrent.TimeUnit; - -/** - * Created by blavenie on 06/01/16. - */ -public class SecurityTokenStore { - - private static final ESLogger log = ESLoggerFactory.getLogger(SecurityTokenStore.class.getName()); - - private String prefix; - private long validityDurationInSeconds; - private LoadingCache<String, String> tokenCache; - - @Inject - public SecurityTokenStore(Settings settings) { - this.prefix = settings.get("duniter.auth.token.prefix", "duniter-"); - this.validityDurationInSeconds = settings.getAsInt("duniter.auth.tokenValidityDuration", 600 /*= 10min*/ ); - this.tokenCache = initGeneratedMessageCache(); - } - - public boolean validateToken(String token) { - Preconditions.checkArgument(StringUtils.isNotBlank(token)); - - String storedToken = tokenCache.getIfPresent(token); - - // if no value in cache => maybe token expired - return ObjectUtils.equals(storedToken, token); - } - - public String createNewToken(String challenge, String signature, String pubkey) { - String token = newToken(challenge, signature, pubkey); - tokenCache.put(challenge, challenge); - return token; - } - - /* -- internal methods -- */ - - protected String newToken(String challenge, String signature, String pubkey) { - return String.valueOf(pubkey + ":" + challenge + "|" + signature); - } - - protected LoadingCache<String, String> initGeneratedMessageCache() { - return CacheBuilder.newBuilder() - .expireAfterWrite(validityDurationInSeconds, TimeUnit.SECONDS) - .build(new CacheLoader<String, String>() { - @Override - public String load(String challenge) throws Exception { - // not used. Filled manually - return null; - } - }); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractBlockchainListenerService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractBlockchainListenerService.java deleted file mode 100644 index ec9b0dddc073b0b408d2df5dfd28fc24c2740f11..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractBlockchainListenerService.java +++ /dev/null @@ -1,170 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.ImmutableList; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.Preconditions; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeService; -import org.duniter.elasticsearch.service.changes.ChangeSource; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.unit.TimeValue; - -import java.io.IOException; -import java.util.Collection; -import java.util.List; -import java.util.concurrent.TimeUnit; - -/** - * Created by Benoit on 26/04/2017. - */ -public abstract class AbstractBlockchainListenerService extends AbstractService implements ChangeService.ChangeListener { - - private static final List<ChangeSource> CHANGE_LISTEN_SOURCES = ImmutableList.of(new ChangeSource("*", BlockchainService.BLOCK_TYPE)); - - protected final boolean enable; - protected final String listenerId; - protected final ThreadPool threadPool; - protected final int bulkSize; - - private final TimeValue flushInterval; - protected final Object threadLock = Boolean.TRUE; - protected BulkRequestBuilder bulkRequest; - protected boolean flushing; - - @Inject - public AbstractBlockchainListenerService(String loggerName, - Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - ThreadPool threadPool, - TimeValue processingInterval) { - super(loggerName, client, settings, cryptoService); - this.listenerId = loggerName; - this.enable = pluginSettings.enableBlockchainSync(); - this.threadPool = threadPool; - - this.bulkSize = pluginSettings.getIndexBulkSize(); - this.bulkRequest = client.prepareBulk(); - - this.flushInterval = processingInterval; - this.flushing = false; - - if (this.enable) { - ChangeService.registerListener(this); - } - } - - - @Override - public String getId() { - return listenerId; - } - - @Override - public void onChange(ChangeEvent change) { - - // Skip _id=current - if("current".equals(change.getId())) return; - - switch (change.getOperation()) { - // on INDEX - case CREATE: - if (change.getSource() != null) { - synchronized (threadLock) { - processBlockIndex(change); - } - } - break; - - // on INDEX - case INDEX: - if (change.getSource() != null) { - synchronized (threadLock) { - processBlockIndex(change); - } - } - break; - - // on DELETE - case DELETE: - synchronized (threadLock) { - processBlockDelete(change); - } - break; - } - - } - - @Override - public Collection<ChangeSource> getChangeSources() { - return CHANGE_LISTEN_SOURCES; - } - - /* -- internal method -- */ - - - protected abstract void processBlockIndex(ChangeEvent change); - - protected abstract void processBlockDelete(ChangeEvent change); - - protected void flushBulkRequestOrSchedule() { - if (flushing || bulkRequest.numberOfActions() == 0) return; - - // Flush now, if need or later - if (bulkRequest.numberOfActions() % bulkSize == 0) { - client.flushBulk(bulkRequest); - bulkRequest = client.prepareBulk(); - } - else { - flushing = true; - threadPool.schedule(() -> { - synchronized (threadLock) { - client.flushBulk(bulkRequest); - bulkRequest = client.prepareBulk(); - flushing = false; - } - }, new TimeValue(500, TimeUnit.MILLISECONDS)); - } - } - - protected BlockchainBlock readBlock(ChangeEvent change) { - Preconditions.checkNotNull(change); - Preconditions.checkNotNull(change.getSource()); - - try { - return getObjectMapper().readValue(change.getSource().streamInput(), BlockchainBlock.class); - } catch (IOException e) { - throw new TechnicalException(String.format("Unable to parse received block %s", change.getId()), e); - } - } - -} 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 deleted file mode 100644 index aa5d0fb04fe250efdb8959c66ec037be9dcca88a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/AbstractService.java +++ /dev/null @@ -1,298 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 com.fasterxml.jackson.databind.ObjectMapper; -import com.google.common.collect.ImmutableSet; -import org.duniter.core.beans.Bean; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.elasticsearch.Records; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.json.JsonAttributeParser; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.InvalidFormatException; -import org.duniter.elasticsearch.exception.InvalidSignatureException; -import org.duniter.elasticsearch.exception.InvalidTimeException; -import org.elasticsearch.ElasticsearchException; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.nuiton.i18n.I18n; - -import java.io.IOException; -import java.util.Objects; -import java.util.Set; - -/** - * Created by Benoit on 08/04/2015. - */ -public abstract class AbstractService implements Bean { - - protected static JsonAttributeParser<String> PARSER_HASH = new JsonAttributeParser<>(Record.PROPERTY_HASH, String.class); - protected static JsonAttributeParser<String> PARSER_SIGNATURE = new JsonAttributeParser<>(Record.PROPERTY_SIGNATURE, String.class); - protected static JsonAttributeParser<String> PARSER_READ_SIGNATURE = new JsonAttributeParser<>(Records.PROPERTY_READ_SIGNATURE, String.class); - - protected final ESLogger logger; - protected Duniter4jClient client; - protected PluginSettings pluginSettings; - protected CryptoService cryptoService; - - private boolean ready = false; - private final int retryCount; - private final int retryWaitDuration; - private final int documentTimeMaxPastDelta; - private final int documentTimeMaxFutureDelta; - - public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings) { - this(loggerName, client, pluginSettings, null); - } - - public AbstractService(Duniter4jClient client, PluginSettings pluginSettings) { - this(client, pluginSettings, null); - } - - public AbstractService(Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - this("duniter", client, pluginSettings, cryptoService); - } - - public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - super(); - this.logger = Loggers.getLogger(loggerName, pluginSettings.getSettings(), new String[0]); - this.client = client; - this.pluginSettings = pluginSettings; - this.cryptoService = cryptoService; - this.retryCount = pluginSettings.getNodeRetryCount(); - this.retryWaitDuration = pluginSettings.getNodeRetryWaitDuration(); - this.documentTimeMaxPastDelta = pluginSettings.getDocumentTimeMaxPastDelta(); - this.documentTimeMaxFutureDelta = pluginSettings.getDocumentTimeMaxFutureDelta(); - } - - /* -- protected methods --*/ - - protected void setIsReady(boolean ready) { - this.ready = ready; - } - public boolean isReady() { - return this.ready; - } - - protected void waitReady() { - try { - while (!ready) { - Thread.sleep(500); - } - } catch (InterruptedException e){ - // Silent - } - } - - protected ObjectMapper getObjectMapper() { - return JacksonUtils.getThreadObjectMapper(); - } - - protected <T> T executeWithRetry(RetryFunction<T> retryFunction) throws TechnicalException{ - int retry = 0; - while (retry < retryCount) { - try { - return retryFunction.execute(); - } catch (TechnicalException e) { - retry++; - - if (retry == retryCount) { - throw e; - } - - if (logger.isDebugEnabled()) { - logger.debug(I18n.t("duniter4j.service.waitThenRetry", e.getMessage(), retry, retryCount)); - } - - try { - Thread.sleep(retryWaitDuration); // waiting - } catch (InterruptedException e2) { - throw new TechnicalException(e2); - } - } - } - - throw new TechnicalException("Error while trying to execute a function with retry"); - } - - protected JsonNode readAndVerifyIssuerSignature(String recordJson) throws ElasticsearchException { - return readAndVerifyIssuerSignature(recordJson, Records.PROPERTY_ISSUER); - } - - protected JsonNode readAndVerifyIssuerSignature(String recordJson, String issuerFieldName) throws ElasticsearchException { - - try { - JsonNode recordObj = getObjectMapper().readTree(recordJson); - readAndVerifyIssuerSignature(recordJson, recordObj, issuerFieldName); - return recordObj; - } - catch(IOException e) { - throw new InvalidFormatException("Invalid record JSON: " + e.getMessage(), e); - } - } - - - protected void readAndVerifyIssuerSignature(JsonNode actualObj, String issuerFieldName) throws ElasticsearchException, JsonProcessingException { - // Remove hash and signature - String recordJson = getObjectMapper().writeValueAsString(actualObj); - readAndVerifyIssuerSignature(recordJson, actualObj, issuerFieldName); - } - - protected void verifyTimeForUpdate(String index, String type, String id, JsonNode actualObj) { - verifyTimeForUpdate(index, type, id, actualObj, Record.PROPERTY_TIME); - } - - protected void verifyTimeForUpdate(String index, String type, String id, JsonNode actualObj, String timeFieldName) { - verifyTimeForUpdate(index, type, id, actualObj, false, timeFieldName); - } - - protected void verifyTimeForUpdate(String index, String type, String id, JsonNode actualObj, boolean allowOldDocuments, String timeFieldName) { - // Check time has been increase - fix #27 - int actualTime = getMandatoryField(actualObj, timeFieldName).asInt(); - int existingTime = client.getMandatoryTypedFieldById(index, type, id, timeFieldName); - if (actualTime <= existingTime) { - throw new InvalidTimeException(String.format("Invalid '%s' value: can not be less or equal to the previous value.", timeFieldName, timeFieldName)); - } - - verifyTime(actualTime, allowOldDocuments, timeFieldName); - } - - protected void verifyTimeForInsert(JsonNode actualObj) { - verifyTimeForInsert(actualObj, Record.PROPERTY_TIME); - } - - protected void verifyTimeForInsert(JsonNode actualObj, String timeFieldName) { - verifyTime(actualObj, false, timeFieldName); - } - - protected void verifyTime(JsonNode actualObj, boolean allowOldDocuments, String timeFieldName) { - int actualTime = getMandatoryField(actualObj, timeFieldName).asInt(); - verifyTime(actualTime, allowOldDocuments, timeFieldName); - } - - protected void verifyTime(int actualTime, - boolean allowOldDocuments, - String timeFieldName) { - // Check time has been increase - fix #27 - long deltaTime = System.currentTimeMillis()/1000 - actualTime; - - // Past time - if (!allowOldDocuments && (deltaTime > 0 && Math.abs(deltaTime) > documentTimeMaxPastDelta)) { - throw new InvalidTimeException(String.format("Invalid '%s' value: too far (in the past) from the UTC server time. Check your device's clock.", timeFieldName)); - } - - // Future time - if (deltaTime < 0 && Math.abs(deltaTime) > documentTimeMaxFutureDelta) { - throw new InvalidTimeException(String.format("Invalid '%s' value: too far (in the future) from the UTC server time. Check your device's clock.", timeFieldName)); - } - } - - protected String getIssuer(JsonNode actualObj) { - return getMandatoryField(actualObj, Records.PROPERTY_ISSUER).asText(); - } - - protected int getVersion(JsonNode actualObj) { - JsonNode value = actualObj.get(Records.PROPERTY_VERSION); - if (value == null || value.isMissingNode()) { - return 1; // first version - } - return value.asInt(); - } - - protected JsonNode getMandatoryField(JsonNode actualObj, String fieldName) { - JsonNode value = actualObj.get(fieldName); - if (value.isMissingNode()) { - throw new InvalidFormatException(String.format("Invalid format. Expected field '%s'", fieldName)); - } - return value; - } - - public interface RetryFunction<T> { - - T execute() throws TechnicalException; - } - - /* -- internal methods -- */ - - private void readAndVerifyIssuerSignature(String recordJson, JsonNode recordObj, String issuerFieldName) throws ElasticsearchException { - - Set<String> fieldNames = ImmutableSet.copyOf(recordObj.fieldNames()); - if (!fieldNames.contains(issuerFieldName) - || !fieldNames.contains(Records.PROPERTY_SIGNATURE)) { - throw new InvalidFormatException(String.format("Invalid record JSON format. Required fields [%s,%s]", Records.PROPERTY_ISSUER, Records.PROPERTY_SIGNATURE)); - } - String issuer = getMandatoryField(recordObj, issuerFieldName).asText(); - String signature = getMandatoryField(recordObj, Records.PROPERTY_SIGNATURE).asText(); - String hash = getMandatoryField(recordObj, Records.PROPERTY_HASH).asText(); - int version = getVersion(recordObj); - - boolean validSignature = false; - - // Remove hash and signature - recordJson = PARSER_SIGNATURE.removeFromJson(recordJson); - recordJson = PARSER_HASH.removeFromJson(recordJson); - - // Remove 'read_signature' attribute if exists (added AFTER signature) - String readSignature = null; - if (fieldNames.contains(Records.PROPERTY_READ_SIGNATURE)) { - readSignature = getMandatoryField(recordObj, Records.PROPERTY_READ_SIGNATURE).asText(); - recordJson = PARSER_READ_SIGNATURE.removeFromJson(recordJson); - } - - // Doc version == 1 - if (version == 1) { - validSignature = cryptoService.verify(recordJson, signature, issuer); - } - - // Doc version > 1 - else { - // Remove hash and signature - boolean validHash = Objects.equals(cryptoService.hash(recordJson), hash); - if (!validHash) { - throw new InvalidSignatureException("Invalid hash of JSON document"); - } - - // Validate signature on hash - validSignature = cryptoService.verify(hash, signature, issuer); - } - - if (!validSignature) { - - throw new InvalidSignatureException("Invalid signature of JSON string"); - } - - // Validate read signature on hash - if (readSignature != null) { - // TODO: validate read signature / recipient ? - } - - // TODO: check issuer is in the WOT ? - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainListenerService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainListenerService.java deleted file mode 100644 index d22758b8e48092a6bbcce16d755ac6eefd2b6c01..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainListenerService.java +++ /dev/null @@ -1,134 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.dao.BlockStatDao; -import org.duniter.elasticsearch.dao.MovementDao; -import org.duniter.elasticsearch.model.Movement; -import org.duniter.elasticsearch.model.BlockchainBlockStat; -import org.duniter.elasticsearch.model.Movements; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.unit.TimeValue; - -import java.util.concurrent.TimeUnit; - -/** - * Created by Benoit on 26/04/2017. - */ -public class BlockchainListenerService extends AbstractBlockchainListenerService { - - private final BlockStatDao blockStatDao; - private final MovementDao movementDao; - - @Inject - public BlockchainListenerService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - ThreadPool threadPool, - BlockStatDao blockStatDao, - MovementDao movementDao) { - super("duniter.blockchain.listener", client, settings, cryptoService, threadPool, - new TimeValue(500, TimeUnit.MILLISECONDS)); - this.blockStatDao = blockStatDao; - this.movementDao = movementDao; - } - - @Override - protected void processBlockIndex(ChangeEvent change) { - - BlockchainBlock block = readBlock(change); - - // Block stat - { - BlockchainBlockStat stat = blockStatDao.toBlockStat(block); - - // Add a delete to bulk - bulkRequest.add(client.prepareDelete(block.getCurrency(), BlockStatDao.TYPE, String.valueOf(block.getNumber())) - .setRefresh(false)); - flushBulkRequestOrSchedule(); - - // Add a insert to bulk - try { - bulkRequest.add(client.prepareIndex(block.getCurrency(), BlockStatDao.TYPE, String.valueOf(block.getNumber())) - .setRefresh(false) // recommended for heavy indexing - .setSource(getObjectMapper().writeValueAsBytes(stat))); - flushBulkRequestOrSchedule(); - } catch (JsonProcessingException e) { - logger.error("Could not serialize BlockStat into JSON: " + e.getMessage(), e); - } - } - - // Movements - { - // Delete previous indexation - bulkRequest = movementDao.bulkDeleteByBlock(block.getCurrency(), - String.valueOf(block.getNumber()), - null, /*do NOT filter on hash = delete by block number*/ - bulkRequest, bulkSize, false); - - // Add a insert to bulk - Movements.stream(block) - .forEach(movement -> { - try { - bulkRequest.add(client.prepareIndex(block.getCurrency(), MovementDao.TYPE) - .setRefresh(false) // recommended for heavy indexing - .setSource(getObjectMapper().writeValueAsBytes(movement))); - flushBulkRequestOrSchedule(); - } catch (JsonProcessingException e) { - logger.error("Could not serialize BlockOperation into JSON: " + e.getMessage(), e); - } - }); - } - } - - protected void processBlockDelete(ChangeEvent change) { - // blockStat - { - // Add delete to bulk - bulkRequest.add(client.prepareDelete(change.getIndex(), BlockStatDao.TYPE, change.getId()) - .setRefresh(false)); - } - - // Operation - { - // Add delete to bulk - bulkRequest = movementDao.bulkDeleteByBlock( - change.getIndex(), - change.getId(), - null/*do kwown the hash*/, - bulkRequest, bulkSize, false); - flushBulkRequestOrSchedule(); - } - } - - /* -- internal method -- */ - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java deleted file mode 100644 index 59f7baeb50c3bf89d6aa501b7b724353cd1d2bae..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/BlockchainService.java +++ /dev/null @@ -1,837 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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.google.common.base.Objects; -import com.google.common.collect.ImmutableSet; -import com.google.common.collect.Lists; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.bma.BlockchainParameters; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.bma.BlockchainRemoteService; -import org.duniter.core.client.service.bma.NetworkRemoteService; -import org.duniter.core.client.service.exception.BlockNotFoundException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.model.NullProgressionModel; -import org.duniter.core.model.ProgressionModel; -import org.duniter.core.model.ProgressionModelImpl; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.ObjectUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.core.util.json.JsonAttributeParser; -import org.duniter.core.util.websocket.WebsocketClientEndpoint; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.dao.BlockDao; -import org.duniter.elasticsearch.exception.DuplicateIndexIdException; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.action.bulk.BulkItemResponse; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.bulk.BulkResponse; -import org.elasticsearch.common.inject.Inject; -import org.nuiton.i18n.I18n; - -import java.io.IOException; -import java.util.*; - -/** - * Created by Benoit on 30/03/2015. - */ -public class BlockchainService extends AbstractService { - - public static final String BLOCK_TYPE = BlockDao.TYPE; - public static final String CURRENT_BLOCK_ID = "current"; - - private static final int SYNC_MISSING_BLOCK_MAX_RETRY = 5; - - private final ProgressionModel nullProgressionModel = new NullProgressionModel(); - - private BlockchainRemoteService blockchainRemoteService; - private List<WebsocketClientEndpoint.ConnectionListener> connectionListeners = new ArrayList<>(); - private final WebsocketClientEndpoint.ConnectionListener dispatchConnectionListener; - - private final JsonAttributeParser<Integer> blockNumberParser = new JsonAttributeParser<>("number", Integer.class); - private final JsonAttributeParser<String> blockCurrencyParser = new JsonAttributeParser<>("currency", String.class); - private final JsonAttributeParser<String> blockHashParser = new JsonAttributeParser<>("hash", String.class); - private final JsonAttributeParser<String> blockPreviousHashParser = new JsonAttributeParser<>("previousHash", String.class); - - private BlockDao blockDao; - - @Inject - public BlockchainService(Duniter4jClient client, - PluginSettings settings, - ThreadPool threadPool, - BlockDao blockDao, - final ServiceLocator serviceLocator){ - super("duniter.blockchain", client, settings); - this.client = client; - this.blockDao = blockDao; - threadPool.scheduleOnStarted(() -> { - blockchainRemoteService = serviceLocator.getBlockchainRemoteService(); - setIsReady(true); - }); - dispatchConnectionListener = new WebsocketClientEndpoint.ConnectionListener() { - @Override - public void onSuccess() { - synchronized (connectionListeners) { - connectionListeners.stream().forEach(connectionListener -> connectionListener.onSuccess()); - } - } - @Override - public void onError(Exception e, long lastTimeUp) { - synchronized (connectionListeners) { - connectionListeners.stream().forEach(connectionListener -> connectionListener.onError(e, lastTimeUp)); - } - } - }; - } - - - public void registerConnectionListener(WebsocketClientEndpoint.ConnectionListener listener) { - synchronized (connectionListeners) { - connectionListeners.add(listener); - } - } - - public BlockchainService listenAndIndexNewBlock(final Peer peer){ - WebsocketClientEndpoint wsEndPoint = blockchainRemoteService.addBlockListener(peer, message -> indexLastBlockFromJson(peer, message), true /*autoreconnect*/); - wsEndPoint.registerListener(dispatchConnectionListener); - return this; - } - - public BlockchainService indexLastBlocks(Peer peer) { - indexLastBlocks(peer, nullProgressionModel); - return this; - } - - public BlockchainService indexLastBlocks(Peer peer, ProgressionModel progressionModel) { - boolean bulkIndex = pluginSettings.isIndexBulkEnable(); - - progressionModel.setStatus(ProgressionModel.Status.RUNNING); - progressionModel.setTotal(100); - long timeStart = System.currentTimeMillis(); - - try { - // Get the blockchain name from node - BlockchainParameters parameter = blockchainRemoteService.getParameters(peer); - if (parameter == null) { - progressionModel.setStatus(ProgressionModel.Status.FAILED); - logger.error(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.remoteParametersError",peer)); - return this; - } - String currencyName = parameter.getCurrency(); - - progressionModel.setTask(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.task", currencyName, peer)); - logger.info(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.task", currencyName, peer)); - - // Then index allOfToList blocks - BlockchainBlock peerCurrentBlock = blockchainRemoteService.getCurrentBlock(peer); - - if (peerCurrentBlock != null) { - final int peerCurrentBlockNumber = peerCurrentBlock.getNumber(); - - // Get the last indexed block number - int startNumber = 0; - - // Check if a previous sync has been done - BlockchainBlock indexedCurrentBlock = getCurrentBlock(currencyName); - if (indexedCurrentBlock != null && indexedCurrentBlock.getNumber() != null) { - int indexedCurrentBlockNumber = indexedCurrentBlock.getNumber(); - - // Make sure this block has been indexed by its number (not only with _id='current') - indexedCurrentBlock = getBlockById(currencyName, indexedCurrentBlockNumber); - - // If current block exists on index, by _id=number AND _id=current - // then keep it and sync only next blocks - if (indexedCurrentBlock != null) { - startNumber = indexedCurrentBlockNumber + 1; - } - } - - // When current block not found, - // try to use the max(number), because block with _id='current' may not has been indexed - if (startNumber <= 1 ){ - startNumber = blockDao.getMaxBlockNumber(currencyName) + 1; - } - - // If some block has been already indexed: detect and resolve fork - if (startNumber > 0) { - String peerStartPreviousHash; - try { - BlockchainBlock peerStartBlock = blockchainRemoteService.getBlock(peer, startNumber - 1); - peerStartPreviousHash = peerStartBlock.getHash(); - } - catch(BlockNotFoundException e) { - // block not exists: use a fake hash for fork detection (will force to compare previous blocks) - peerStartPreviousHash = "--"; - } - boolean resolved = detectAndResolveFork(peer, currencyName, peerStartPreviousHash, startNumber - 1); - if (!resolved) { - // Bad blockchain ! skipping sync - logger.error(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.invalidBlockchain", currencyName, peer)); - return this; - } - } - - if (startNumber <= peerCurrentBlockNumber) { - Collection<String> missingBlocks = bulkIndex - ? indexBlocksUsingBulk(peer, currencyName, startNumber, peerCurrentBlockNumber, progressionModel, true) - : indexBlocksNoBulk(peer, currencyName, startNumber, peerCurrentBlockNumber, progressionModel, true); - - // If some blocks are missing, try to get it using other peers - if (CollectionUtils.isNotEmpty(missingBlocks)) { - progressionModel.setTask(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.otherPeers.task", currencyName)); - missingBlocks = indexMissingBlocksFromOtherPeers(peer, peerCurrentBlock, missingBlocks, 1); - } - - if (CollectionUtils.isEmpty(missingBlocks)) { - logger.info(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.succeed", currencyName, peer, (System.currentTimeMillis() - timeStart))); - progressionModel.setStatus(ProgressionModel.Status.SUCCESS); - } - else { - logger.warn(String.format("[%s] [%s] Could not indexed some blocks. Missing %s blocks.", currencyName, peer, missingBlocks.size())); - progressionModel.setStatus(ProgressionModel.Status.FAILED); - } - } - else { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] [%s] Already up to date at block #%s.", currencyName, peer, peerCurrentBlockNumber)); - } - progressionModel.setStatus(ProgressionModel.Status.SUCCESS); - } - } - } catch(Exception e) { - logger.error("Error during indexLastBlocks: " + e.getMessage(), e); - progressionModel.setStatus(ProgressionModel.Status.FAILED); - } - - return this; - } - - public BlockchainService indexBlocksRange(Peer peer, int firstNumber, int lastNumber) { - indexBlocksRange(peer, nullProgressionModel, firstNumber, lastNumber); - return this; - } - - public BlockchainService indexBlocksRange(Peer peer, ProgressionModel progressionModel, int firstNumber, int lastNumber) { - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(progressionModel); - Preconditions.checkArgument(firstNumber < lastNumber); - - boolean bulkIndex = pluginSettings.isIndexBulkEnable(); - - progressionModel.setStatus(ProgressionModel.Status.RUNNING); - progressionModel.setTotal(100); - long timeStart = System.currentTimeMillis(); - - try { - // Get the blockchain name from node - BlockchainParameters parameter = blockchainRemoteService.getParameters(peer); - if (parameter == null) { - progressionModel.setStatus(ProgressionModel.Status.FAILED); - logger.error(I18n.t("duniter4j.blockIndexerService.indexBlocksRange.remoteParametersError",peer)); - return this; - } - String currencyName = parameter.getCurrency(); - - progressionModel.setTask(I18n.t("duniter4j.blockIndexerService.indexBlocksRange.task", currencyName, peer, firstNumber, lastNumber)); - logger.info(I18n.t("duniter4j.blockIndexerService.indexBlocksRange.task", currencyName, peer, firstNumber, lastNumber)); - - // Then index allOfToList blocks - BlockchainBlock peerCurrentBlock = blockchainRemoteService.getCurrentBlock(peer); - - if (peerCurrentBlock != null) { - final int peerCurrentBlockNumber = peerCurrentBlock.getNumber(); - - - boolean isLastCurrent = lastNumber >= peerCurrentBlockNumber; - if (lastNumber > peerCurrentBlockNumber) { - lastNumber = peerCurrentBlockNumber; - } - - if (firstNumber <= peerCurrentBlockNumber) { - Collection<String> missingBlocks = bulkIndex - ? indexBlocksUsingBulk(peer, currencyName, firstNumber, lastNumber, progressionModel, isLastCurrent) - : indexBlocksNoBulk(peer, currencyName, firstNumber, lastNumber, progressionModel, isLastCurrent); - - // If some blocks are missing, try to get it using other peers - if (CollectionUtils.isNotEmpty(missingBlocks)) { - progressionModel.setTask(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.otherPeers.task", currencyName)); - missingBlocks = indexMissingBlocksFromOtherPeers(peer, peerCurrentBlock, missingBlocks, 1); - } - - if (CollectionUtils.isEmpty(missingBlocks)) { - logger.info(I18n.t("duniter4j.blockIndexerService.indexBlocksRange.succeed", currencyName, peer, firstNumber, lastNumber, (System.currentTimeMillis() - timeStart))); - progressionModel.setStatus(ProgressionModel.Status.SUCCESS); - } - else { - logger.warn(String.format("[%s] [%s] Could not indexed some blocks from range [%s-%s]. Missing %s blocks.", currencyName, peer, firstNumber, lastNumber, missingBlocks.size())); - progressionModel.setStatus(ProgressionModel.Status.FAILED); - } - } - else { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] [%s] Invalid block range [%s-%s]. Current block number is #%s", currencyName, peer, firstNumber, lastNumber, peerCurrentBlockNumber)); - } - progressionModel.setStatus(ProgressionModel.Status.SUCCESS); - } - } - } catch(Exception e) { - logger.error("Error during indexBlocksRange: " + e.getMessage(), e); - progressionModel.setStatus(ProgressionModel.Status.FAILED); - } - - return this; - } - - /** - * Create or update a block, depending on its existence and hash - * @param block - * @param updateWhenSameHash if true, always update an existing block. If false, update only if hash has changed. - * @param wait wait indexBlocksFromNode end - * @throws DuplicateIndexIdException - */ - public void saveBlock(BlockchainBlock block, boolean updateWhenSameHash, boolean wait) throws DuplicateIndexIdException { - Preconditions.checkNotNull(block, "block could not be null") ; - Preconditions.checkNotNull(block.getCurrency(), "block attribute 'blockchain' could not be null"); - Preconditions.checkNotNull(block.getNumber(), "block attribute 'number' could not be null"); - Preconditions.checkNotNull(block.getHash(), "block attribute 'hash' could not be null"); - - BlockchainBlock existingBlock = blockDao.getBlockById(block.getCurrency(), getBlockId(block.getNumber())); - - // Currency not exists, or has changed, so create it - if (existingBlock == null) { - if (logger.isTraceEnabled()) { - logger.trace(String.format("Insert new block [%s]", block.getNumber())); - } - - // Create new block - blockDao.create(block, wait); - } - - // Exists, so check the owner signature - else { - boolean doUpdate; - if (updateWhenSameHash) { - doUpdate = true; - if (logger.isTraceEnabled() && doUpdate) { - logger.trace(String.format("Update block [%s]", block.getNumber())); - } - } - else { - doUpdate = !StringUtils.equals(existingBlock.getHash(), block.getHash()); - if (logger.isTraceEnabled()) { - if (doUpdate) { - logger.trace(String.format("Update block [%s]: hash has been changed, old=[%s] new=[%s]", block.getNumber(), existingBlock.getHash(), block.getHash())); - } - else { - logger.trace(String.format("Skipping update block [%s]: hash is up to date.", block.getNumber())); - } - } - } - - // Update existing block - if (doUpdate) { - blockDao.update(block, wait); - } - } - } - - /** - * Index the given block, as the last (current) block. This will check is a fork has occur, and apply a rollback so. - * @param peer a source peer - * @param json block as json - */ - public BlockchainService indexLastBlockFromJson(Peer peer, String json) { - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length() > 0); - - indexBlockFromJson(peer, json, true /*is current*/, true/*check fork*/, true/*wait*/); - - return this; - } - - /** - * - * @param json block as json - * @param wait need to wait until processed ? - */ - public BlockchainService indexBlockFromJson(Peer peer, String json, boolean isCurrent, boolean detectFork, boolean wait) { - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length() > 0); - - String currencyName = blockCurrencyParser.getValue(json); - Integer number = blockNumberParser.getValue(json); - String hash = blockHashParser.getValue(json); - - Preconditions.checkNotNull(number); - logger.info(I18n.t("duniter4j.blockIndexerService.indexBlock", currencyName, peer, number, hash)); - if (logger.isTraceEnabled()) { - logger.trace(json); - } - - // Detecting fork and rollback is necessary - if (detectFork) { - String previousHash = blockPreviousHashParser.getValue(json); - boolean resolved = detectAndResolveFork(peer, currencyName, previousHash, number - 1); - if (!resolved) { - // Bad blockchain ! Skipping block indexation - logger.error(I18n.t("duniter4j.blockIndexerService.detectFork.invalidBlockchain", currencyName, peer, number, hash)); - return this; - } - } - - // Workaround for https://github.com/duniter/duniter/issues/1042 - if (json.contains("[object Object]")) { - // Getting block using GET request '/blochain/block' - json = blockchainRemoteService.getBlockAsJson(peer, number); - } - - // Index new block - blockDao.create(currencyName, getBlockId(number), json.getBytes(), wait); - - // Update current - if (isCurrent) { - indexCurrentBlockFromJson(currencyName, json, true /*wait*/); - } - - return this; - } - - /** - * - * @param currentBlock - */ - public void indexCurrentBlock(BlockchainBlock currentBlock, boolean wait) { - Preconditions.checkNotNull(currentBlock); - Preconditions.checkArgument(StringUtils.isNotBlank(currentBlock.getCurrency())); - Preconditions.checkNotNull(currentBlock.getHash()); - Preconditions.checkNotNull(currentBlock.getNumber()); - - // Serialize into JSON - // WARN: must use GSON, to have same JSON result (e.g identities and joiners field must be converted into String) - try { - String json = getObjectMapper().writeValueAsString(currentBlock); - indexCurrentBlockFromJson(currentBlock.getCurrency(), json, wait); - } catch(IOException e) { - throw new TechnicalException(e); - } - } - - /** - * - * @param currencyName - * @param json block as JSON - * @pram wait need to wait until block processed ? - */ - public void indexCurrentBlockFromJson(final String currencyName, final String json, final boolean wait) { - Preconditions.checkNotNull(json); - Preconditions.checkArgument(json.length() > 0); - Preconditions.checkArgument(StringUtils.isNotBlank(currencyName)); - - // Preparing indexBlocksFromNode - if (blockDao.isExists(currencyName, CURRENT_BLOCK_ID)) { - blockDao.update(currencyName, CURRENT_BLOCK_ID, json.getBytes(), wait); - } - else { - blockDao.create(currencyName, CURRENT_BLOCK_ID, json.getBytes(), wait); - } - } - - public BlockchainBlock getBlockById(final String currencyName, final int number) { - return blockDao.getBlockById(currencyName, String.valueOf(number)); - } - - public BlockchainBlock getCurrentBlock(final String currencyName) { - return blockDao.getBlockById(currencyName, CURRENT_BLOCK_ID); - } - - public void deleteFrom(final String currencyName, final int fromBlock) { - int maxBlock = blockDao.getMaxBlockNumber(currencyName); - - blockDao.deleteRange(currencyName, fromBlock, maxBlock); - - // Delete current also - blockDao.deleteById(currencyName, CURRENT_BLOCK_ID); - - } - - - public void deleteRange(final String currencyName, final int fromBlock, int toBlock) { - int maxBlock = blockDao.getMaxBlockNumber(currencyName); - - boolean isLastBlock = toBlock >= maxBlock; - - blockDao.deleteRange(currencyName, fromBlock, (isLastBlock ? maxBlock : toBlock)); - - // Delete current also, if last block - if (isLastBlock) { - blockDao.deleteById(currencyName, CURRENT_BLOCK_ID); - } - - } - - /* -- Internal methods -- */ - - private Collection<String> indexBlocksNoBulk(Peer peer, String currencyName, int firstNumber, int lastNumber, ProgressionModel progressionModel, boolean isLastCurrent) { - Set<String> missingBlockNumbers = new LinkedHashSet<>(); - - for (int curNumber = firstNumber; curNumber <= lastNumber; curNumber++) { - if (curNumber != 0 && curNumber % 1000 == 0) { - - // Check is stopped - if (progressionModel.isCancel()) { - progressionModel.setStatus(ProgressionModel.Status.STOPPED); - if (logger.isInfoEnabled()) { - logger.info(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.stopped", peer)); - } - return missingBlockNumbers; - } - - // Report progress - reportIndexBlocksProgress(progressionModel, currencyName, peer, firstNumber, lastNumber, curNumber); - } - - try { - String blockAsJson = blockchainRemoteService.getBlockAsJson(peer, curNumber); - blockDao.create(currencyName, getBlockId(curNumber), blockAsJson.getBytes(), true /*wait*/); - - // If last block - if (isLastCurrent && curNumber == lastNumber - 1) { - // update the current block - indexCurrentBlockFromJson(currencyName, blockAsJson, true /*wait*/); - } - } - catch(Throwable t) { - logger.debug(String.format("Error while getting block #%s: %s. Skipping this block.", curNumber, t.getMessage())); - missingBlockNumbers.add(String.valueOf(curNumber)); - } - } - - return missingBlockNumbers; - } - - private Collection<String> indexBlocksUsingBulk(Peer peer, String currencyName, int firstNumber, int lastNumber, ProgressionModel progressionModel, - boolean isLastCurrentNumber) { - Set<String> missingBlockNumbers = new LinkedHashSet<>(); - - boolean debug = logger.isDebugEnabled(); - - int batchSize = pluginSettings.getIndexBulkSize(); - String currentBlockJson = null; - - for (int batchFirstNumber = firstNumber; batchFirstNumber < lastNumber; ) { - // Check if stop (e.g. ask by user) - if (progressionModel.isCancel()) { - progressionModel.setStatus(ProgressionModel.Status.STOPPED); - if (logger.isInfoEnabled()) { - logger.info(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.stopped", currencyName, peer.getUrl())); - } - return missingBlockNumbers; - } - - String[] blocksAsJson = null; - try { - final int batchFirstNumberFinal = batchFirstNumber; - blocksAsJson = executeWithRetry(()->blockchainRemoteService.getBlocksAsJson(peer, batchSize, batchFirstNumberFinal)); - } catch(TechnicalException e) { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] [%s] Error while getting blocks from #%s (count=%s): %s. Skipping blocks.",currencyName, peer, batchFirstNumber, batchSize, e.getMessage())); - } - } - - // Peer send no blocks - if (CollectionUtils.isEmpty(blocksAsJson)) { - - // Add range to missing blocks - missingBlockNumbers.add(batchFirstNumber + "-" + (batchFirstNumber+batchSize)); - - // Update counter - batchFirstNumber += batchSize; - } - - // Process received blocks - else { - - List<Integer> processedBlockNumbers = Lists.newArrayList(); - BulkRequestBuilder bulkRequest = client.prepareBulk(); - for (String blockAsJson : blocksAsJson) { - Integer itemNumber = blockNumberParser.getValue(blockAsJson); - - // update curNumber with max number; - if (itemNumber > batchFirstNumber) { - batchFirstNumber = itemNumber; - } - - if (itemNumber != null && !processedBlockNumbers.contains(itemNumber)) { - // Add to bulk - bulkRequest.add(client.prepareIndex(currencyName, BLOCK_TYPE, itemNumber.toString()) - .setRefresh(false) // recommended for heavy indexing - .setSource(blockAsJson) - ); - processedBlockNumbers.add(itemNumber); - } - - // If last block : also update the current block - if (isLastCurrentNumber && itemNumber == lastNumber) { - currentBlockJson = blockAsJson; - } - } - - if (bulkRequest.numberOfActions() > 0) { - - // Flush the bulk if not empty - BulkResponse bulkResponse = bulkRequest.get(); - - // If failures, continue but save missing blocks - if (bulkResponse.hasFailures()) { - // process failures by iterating through each bulk response item - for (BulkItemResponse itemResponse : bulkResponse) { - boolean skip = !itemResponse.isFailed() - || Objects.equal(CURRENT_BLOCK_ID, itemResponse.getId()) - || missingBlockNumbers.contains(Integer.parseInt(itemResponse.getId())); - if (!skip) { - int itemNumber = Integer.parseInt(itemResponse.getId()); - if (debug) { - logger.debug(String.format("Error while getting block #%s: %s. Skipping this block.", itemNumber, itemResponse.getFailureMessage())); - } - missingBlockNumbers.add(itemResponse.getId()); - } - } - } - } - } - - // Report progress - reportIndexBlocksProgress(progressionModel, currencyName, peer, firstNumber, lastNumber, batchFirstNumber); - batchFirstNumber++; // increment for next loop - } - - if (StringUtils.isNotBlank(currentBlockJson)) { - indexCurrentBlockFromJson(currencyName, currentBlockJson, true); - } - - return missingBlockNumbers; - } - - /** - * Get blocks from other peers. - * WARNING: given list must be ordered (with ascending order) - * @param peer - * @param currentBlock - * @param sortedMissingBlocks - * @param tryCounter - */ - private Collection<String> indexMissingBlocksFromOtherPeers(Peer peer, BlockchainBlock currentBlock, Collection<String> sortedMissingBlocks, int tryCounter) { - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(currentBlock); - Preconditions.checkNotNull(currentBlock.getHash()); - Preconditions.checkNotNull(currentBlock.getNumber()); - Preconditions.checkArgument(CollectionUtils.isNotEmpty(sortedMissingBlocks)); - Preconditions.checkArgument(tryCounter >= 1); - - NetworkRemoteService networkRemoteService = ServiceLocator.instance().getNetworkRemoteService(); - BlockchainRemoteService blockchainRemoteService = ServiceLocator.instance().getBlockchainRemoteService(); - String currencyName = currentBlock.getCurrency(); - boolean debug = logger.isDebugEnabled(); - - Set<String> newMissingBlocks = new LinkedHashSet<>(); - newMissingBlocks.addAll(sortedMissingBlocks); - - if (debug) { - logger.debug(String.format("Missing blocks are: %s", newMissingBlocks.toString())); - } - - // Select other peers, in filtering on the same blockchain version - - // TODO : a activer quand les peers seront bien mis à jour (UP/DOWN, block, hash...) - //List<Peer> otherPeers = networkRemoteService.findPeers(peer, "UP", EndpointApi.BASIC_MERKLED_API, - // currentBlock.getNumber(), currentBlock.getHash()); - List<Peer> otherPeers = networkRemoteService.findPeers(peer, null, EndpointApi.BASIC_MERKLED_API, - null, null); - - for(Peer childPeer: otherPeers) { - if (logger.isInfoEnabled()) { - logger.info(String.format("[%s] Trying to get missing blocks from other peer [%s]...", currencyName, childPeer)); - } - try { - for(String blockNumberStr: ImmutableSet.copyOf(sortedMissingBlocks)) { - - boolean isBlockRange = blockNumberStr.indexOf('-') != -1; - - // Get using bulk - if (isBlockRange) { - String[] rangeParts = blockNumberStr.split("-"); - int firstNumber = Integer.parseInt(rangeParts[0]); - int lastNumber = Integer.parseInt(rangeParts[1]); - - // Remove current blocks range - newMissingBlocks.remove(blockNumberStr); - - Collection<String> bulkMissingBlocks = indexBlocksUsingBulk(childPeer, currencyName, firstNumber, lastNumber, new ProgressionModelImpl(), true); - - // Re add if new missing blocks - if (CollectionUtils.isNotEmpty(bulkMissingBlocks)) { - newMissingBlocks.addAll(bulkMissingBlocks); - } - } - - // Get blocks one by one - else { - int blockNumber = Integer.parseInt(blockNumberStr); - String blockAsJson = blockchainRemoteService.getBlockAsJson(childPeer, blockNumber); - if (StringUtils.isNotBlank(blockAsJson)) { - if (debug) { - logger.debug(String.format("Found missing block #%s on peer [%s].", blockNumber, childPeer)); - } - - // Index the missing block - blockDao.create(currencyName, getBlockId(blockNumber), blockAsJson.getBytes(), true/*wait*/); - - // Remove this block number from the final missing list - newMissingBlocks.remove(blockNumber); - } - } - } - - if (CollectionUtils.isEmpty(newMissingBlocks)) { - break; - } - - // Update the list, for the next iteration - sortedMissingBlocks = newMissingBlocks; - } - catch(TechnicalException e) { - if (debug) { - logger.debug(String.format("Error while getting blocks from peer [%s]: %s. Skipping this peer.", childPeer), e.getMessage()); - } - - continue; // skip this peer - } - } - - - if (CollectionUtils.isEmpty(newMissingBlocks)) { - return null; - } - - tryCounter++; - if (tryCounter >= SYNC_MISSING_BLOCK_MAX_RETRY) { - // Max retry : stop here - logger.error("Some blocks are still missing, after %s try: %s", SYNC_MISSING_BLOCK_MAX_RETRY, newMissingBlocks.toArray(new String[0])); - return newMissingBlocks; - } - - if (debug) { - logger.debug("Some blocks are still missing: %s. Will retry later (%s/%s)...", newMissingBlocks.toArray(new String[0]), tryCounter, SYNC_MISSING_BLOCK_MAX_RETRY); - } - try { - Thread.sleep(60 *1000); // wait 1 min - - } - catch (InterruptedException e) { - return null; // stop here - } - - // retrying, with the new new blockchain - BlockchainBlock newCurrentBlock = blockchainRemoteService.getCurrentBlock(peer); - return indexMissingBlocksFromOtherPeers(peer, newCurrentBlock, newMissingBlocks, tryCounter); - } - - private void reportIndexBlocksProgress(ProgressionModel progressionModel, String currencyName, Peer peer, int firstNumber, int lastNumber, int curNumber) { - int pct = (curNumber - firstNumber) * 100 / (lastNumber - firstNumber); - progressionModel.setCurrent(pct); - - progressionModel.setMessage(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.progress", currencyName, peer, curNumber, lastNumber, pct)); - if (logger.isInfoEnabled()) { - logger.info(I18n.t("duniter4j.blockIndexerService.indexLastBlocks.progress", currencyName, peer, curNumber, lastNumber, pct)); - } - - } - - private boolean isBlockIndexed(String currencyName, int number, String hash) { - Preconditions.checkNotNull(currencyName); - Preconditions.checkNotNull(hash); - // Check if previous block exists - BlockchainBlock block = getBlockById(currencyName, number); - boolean blockExists = block != null; - if (!blockExists) { - return blockExists; - } - return ObjectUtils.equals(block.getHash(), hash); - } - - private boolean detectAndResolveFork(Peer peer, final String currencyName, final String hash, final int number){ - int forkResyncWindow = pluginSettings.getNodeForkResyncWindow(); - String forkOriginHash = hash; - int forkOriginNumber = number; - boolean sameBlockIndexed = isBlockIndexed(currencyName, forkOriginNumber, forkOriginHash); - while (!sameBlockIndexed && forkOriginNumber > 0) { - - if (!sameBlockIndexed && logger.isInfoEnabled()) { - logger.info(I18n.t("duniter4j.blockIndexerService.detectFork.invalidBlock", currencyName, peer, forkOriginNumber, forkOriginHash)); - } - forkOriginNumber -= forkResyncWindow; - if (forkOriginNumber < 0) { - forkOriginNumber = 0; - } - - // Get remote block (with auto-retry) - try { - final int currentNumberFinal = forkOriginNumber; - String testBlock = executeWithRetry(() -> - blockchainRemoteService.getBlockAsJson(peer, currentNumberFinal)); - forkOriginHash = blockHashParser.getValue(testBlock); - - // Check is exists on ES index - sameBlockIndexed = isBlockIndexed(currencyName, forkOriginNumber, forkOriginHash); - } catch (TechnicalException e) { - logger.warn(I18n.t("duniter4j.blockIndexerService.detectFork.remoteBlockNotFound", currencyName, peer, forkOriginNumber, e.getMessage())); - sameBlockIndexed = false; // continue (go back again) - } - } - - if (!sameBlockIndexed) { - return false; // sync could not be done (bad blockchain: no common blocks !) - } - - if (forkOriginNumber < number) { - logger.info(I18n.t("duniter4j.blockIndexerService.detectFork.resync", currencyName, peer, forkOriginNumber)); - // Remove some previous block - blockDao.deleteRange(currencyName, forkOriginNumber/*from*/, number+forkResyncWindow/*to*/); - - // Re-indexing blocks - indexBlocksUsingBulk(peer, currencyName, forkOriginNumber/*from*/, number, nullProgressionModel, true); - } - - return true; // sync OK - } - - - private String getBlockId(int number) { - return number == -1 ? CURRENT_BLOCK_ID : String.valueOf(number); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java deleted file mode 100644 index 0323203a874d6fc4fd0e492095c0d9cf4e2a7324..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/CurrencyService.java +++ /dev/null @@ -1,240 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.dao.CurrencyDao; -import org.duniter.core.client.dao.PeerDao; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.bma.BlockchainParameters; -import org.duniter.core.client.model.elasticsearch.Currency; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.bma.BlockchainRemoteService; -import org.duniter.core.client.service.exception.HttpConnectException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.Preconditions; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.dao.*; -import org.duniter.elasticsearch.exception.AccessDeniedException; -import org.duniter.elasticsearch.exception.DuplicateIndexIdException; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; - -import java.util.HashMap; -import java.util.Map; - -/** - * Created by Benoit on 30/03/2015. - */ -public class CurrencyService extends AbstractService { - - public static final String INDEX = CurrencyExtendDao.INDEX; - public static final String RECORD_TYPE = CurrencyExtendDao.RECORD_TYPE; - - private BlockchainRemoteService blockchainRemoteService; - private CurrencyExtendDao currencyDao; - private Map<String, IndexDao<?>> currencyDataDaos = new HashMap<>(); - private Injector injector; - - @Inject - public CurrencyService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - CurrencyDao currencyDao, - ThreadPool threadPool, - Injector injector, - final ServiceLocator serviceLocator) { - super("duniter." + INDEX, client, settings, cryptoService); - this.currencyDao = (CurrencyExtendDao)currencyDao; - this.injector = injector; - - threadPool.scheduleOnStarted(() -> { - this.blockchainRemoteService = serviceLocator.getBlockchainRemoteService(); - setIsReady(true); - }); - } - - public CurrencyService createIndexIfNotExists() { - currencyDao.createIndexIfNotExists(); - return this; - } - - public CurrencyService deleteIndex() { - currencyDao.deleteIndex(); - return this; - } - - public boolean isCurrencyExists(String currencyName) { - return currencyDao.isExists(currencyName); - } - - /** - * Retrieve the blockchain data, from peer - * - * @param peer - * @param autoReconnect - * @return the created blockchain - */ - public Currency indexCurrencyFromPeer(Peer peer, boolean autoReconnect) { - if (!autoReconnect) { - return indexCurrencyFromPeer(peer); - } - - while(true) { - try { - return indexCurrencyFromPeer(peer); - } catch (HttpConnectException e) { - // log then retry - logger.warn(String.format("[%s] Unable to connect. Retrying in 10s...", peer.toString())); - } - - try { - Thread.sleep(10 * 1000); // wait 10s - } catch(Exception e) { - throw new TechnicalException(e); - } - } - } - - /** - * Retrieve the blockchain data, from peer - * - * @param peer - * @return the created blockchain - */ - public Currency indexCurrencyFromPeer(Peer peer) { - - waitReady(); - - BlockchainParameters parameters = blockchainRemoteService.getParameters(peer); - BlockchainBlock firstBlock = blockchainRemoteService.getBlock(peer, 0l); - BlockchainBlock currentBlock = blockchainRemoteService.getCurrentBlock(peer); - Long lastUD = blockchainRemoteService.getLastUD(peer); - - - Currency result = new Currency(); - result.setCurrencyName(parameters.getCurrency()); - result.setFirstBlockSignature(firstBlock.getSignature()); - result.setMembersCount(currentBlock.getMembersCount()); - result.setLastUD(lastUD); - result.setParameters(parameters); - - // Save it - saveCurrency(result); - - return result; - } - - /** - * Save a blockchain (update or create) into the blockchain index. - * @param currency - * @throws DuplicateIndexIdException - * @throws AccessDeniedException if exists and user if not the original blockchain sender - */ - public void saveCurrency(Currency currency) throws DuplicateIndexIdException { - Preconditions.checkNotNull(currency, "currency could not be null") ; - Preconditions.checkNotNull(currency.getId(), "currency attribute 'currency' could not be null"); - - boolean exists = currencyDao.isExists(currency.getId()); - - // Currency not exists, so create it - if (!exists) { - // Save it - currencyDao.create(currency); - - // Create data index (delete first if exists) - getCurrencyDataDao(currency.getId()) - .deleteIndex() - .createIndexIfNotExists(); - - } - - // Exists, so check the owner signature - else { - - // Save changes - currencyDao.update(currency); - - // Create data index (if need) - getCurrencyDataDao(currency.getId()) - .createIndexIfNotExists(); - } - } - - /* -- Internal methods -- */ - - protected IndexDao<?> getCurrencyDataDao(final String currencyId) { - // Create data - IndexDao<?> dataDao = currencyDataDaos.get(currencyId); - if (dataDao == null) { - dataDao = new AbstractIndexDao(currencyId) { - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", currencyId)); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(currencyId); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - - // Add peer type - TypeDao<?> peerDao = (TypeDao<?>)ServiceLocator.instance().getBean(PeerDao.class); - createIndexRequestBuilder.addMapping(peerDao.getType(), peerDao.createTypeMapping()); - - // Add block type - BlockDao blockDao = ServiceLocator.instance().getBean(BlockDao.class); - createIndexRequestBuilder.addMapping(blockDao.getType(), blockDao.createTypeMapping()); - - // Add movement type - MovementDao operationDao = ServiceLocator.instance().getBean(MovementDao.class); - createIndexRequestBuilder.addMapping(operationDao.getType(), operationDao.createTypeMapping()); - - // Add blockStat type - BlockStatDao blockStatDao = injector.getInstance(BlockStatDao.class); - createIndexRequestBuilder.addMapping(blockStatDao.getType(), blockStatDao.createTypeMapping()); - - // Add synchro execution - SynchroExecutionDao synchroExecutionDao = injector.getInstance(SynchroExecutionDao.class); - createIndexRequestBuilder.addMapping(synchroExecutionDao.getType(), synchroExecutionDao.createTypeMapping()); - - // Creating the index - createIndexRequestBuilder.execute().actionGet(); - } - }; - injector.injectMembers(dataDao); - currencyDataDaos.put(currencyId, dataDao); - } - - return dataDao; - - - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/DocStatService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/DocStatService.java deleted file mode 100644 index 64a34539eda0ddf37f81970f0a57dbfd980129e8..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/DocStatService.java +++ /dev/null @@ -1,190 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.apache.commons.collections4.CollectionUtils; -import org.duniter.core.util.DateUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.dao.DocStatDao; -import org.duniter.elasticsearch.model.DocStat; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; - -import java.util.*; -import java.util.concurrent.TimeUnit; - -/** - * Maintained stats on doc (count records) - * Created by Benoit on 30/03/2015. - */ -public class DocStatService extends AbstractService { - - private DocStatDao docStatDao; - private ThreadPool threadPool; - private List<StatDef> statDefs = new ArrayList<>(); - - public interface ComputeListener { - void onCompute(DocStat stat); - } - - public class StatDef { - String index; - String type; - List<ComputeListener> listeners; - StatDef(String index, String type) { - this.index=index; - this.type=type; - } - - @Override - public boolean equals(Object obj) { - return (obj instanceof StatDef) && - Objects.equals(((StatDef)obj).index, index) && - Objects.equals(((StatDef)obj).type, type); - } - - public void addListener(ComputeListener listener) { - if (listeners == null) { - listeners = new ArrayList<>(); - } - listeners.add(listener); - } - } - - @Inject - public DocStatService(Duniter4jClient client, PluginSettings settings, ThreadPool threadPool, - DocStatDao docStatDao){ - super("duniter.data.stats", client, settings); - this.threadPool = threadPool; - this.docStatDao = docStatDao; - setIsReady(true); - } - - public DocStatService createIndexIfNotExists() { - docStatDao.createIndexIfNotExists(); - return this; - } - - public DocStatService deleteIndex() { - docStatDao.deleteIndex(); - return this; - } - - public DocStatService registerIndex(String index, String type) { - return registerIndex(index, type, null); - } - - public DocStatService registerIndex(String index, String type, ComputeListener listener) { - Preconditions.checkArgument(StringUtils.isNotBlank(index)); - StatDef statDef = new StatDef(index, type); - if (!statDefs.contains(statDef)) { - statDefs.add(statDef); - } - - if (listener != null) { - addListener(index, type, listener); - } - - return this; - } - - public DocStatService addListener(String index, String type, ComputeListener listener) { - Preconditions.checkArgument(StringUtils.isNotBlank(index)); - Preconditions.checkNotNull(listener); - - // Find the existsing def - StatDef spec = new StatDef(index, type); - StatDef statDef = statDefs.stream().filter(sd -> sd.equals(spec)).findFirst().get(); - Preconditions.checkNotNull(statDef); - - statDef.addListener(listener); - return this; - } - - /** - * Start scheduling doc stats update - * @return - */ - public DocStatService startScheduling() { - long delayBeforeNextHour = DateUtils.delayBeforeNextHour(); - - threadPool.scheduleAtFixedRate( - this::computeStats, - delayBeforeNextHour, - 60 * 60 * 1000 /* every hour */, - TimeUnit.MILLISECONDS); - return this; - } - - public void computeStats() { - - // Skip if empty - if (CollectionUtils.isEmpty(statDefs)) return; - - int bulkSize = pluginSettings.getIndexBulkSize(); - long now = System.currentTimeMillis()/1000; - BulkRequestBuilder bulkRequest = client.prepareBulk(); - - DocStat stat = new DocStat(); - stat.setTime(now); - - int counter = 0; - - for (StatDef statDef: statDefs) { - long count = docStatDao.countDoc(statDef.index, statDef.type); - - // Update stat properties (resue existing obj) - stat.setIndex(statDef.index); - stat.setIndexType(statDef.type); - stat.setCount(count); - - // Call compute listeners if any - if (CollectionUtils.isNotEmpty(statDef.listeners)) { - statDef.listeners.forEach(l -> l.onCompute(stat)); - } - - // Add insertion into bulk - IndexRequestBuilder request = docStatDao.prepareIndex(stat); - bulkRequest.add(request); - counter++; - - // Flush the bulk if not empty - if ((counter % bulkSize) == 0) { - client.flushBulk(bulkRequest); - bulkRequest = client.prepareBulk(); - } - } - - // last flush - if ((counter % bulkSize) != 0) { - client.flushBulk(bulkRequest); - } - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/PeerService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/PeerService.java deleted file mode 100644 index 87d2bbbaac1a9ececa2ce77e347bffc6697bb4c8..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/PeerService.java +++ /dev/null @@ -1,164 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.ImmutableList; -import com.google.common.collect.Lists; -import org.duniter.core.client.dao.PeerDao; -import org.duniter.core.client.model.bma.BlockchainParameters; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.local.NetworkService; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.inject.Inject; -import org.nuiton.i18n.I18n; - -import java.util.List; - -/** - * Created by Benoit on 30/03/2015. - */ -public class PeerService extends AbstractService { - - private org.duniter.core.client.service.bma.BlockchainRemoteService blockchainRemoteService; - private org.duniter.core.client.service.local.NetworkService networkService; - private org.duniter.core.client.service.local.PeerService delegate; - private PeerDao peerDao; - private ThreadPool threadPool; - - // Define endpoint API to include - private List<String> includeEndpointApis = Lists.newArrayList( - EndpointApi.BASIC_MERKLED_API.name(), - EndpointApi.BMAS.name(), - EndpointApi.WS2P.name()); - - @Inject - public PeerService(Duniter4jClient client, PluginSettings settings, ThreadPool threadPool, - CryptoService cryptoService, PeerDao peerDao, - final ServiceLocator serviceLocator){ - super("duniter.network.peer", client, settings, cryptoService); - this.threadPool = threadPool; - this.peerDao = peerDao; - threadPool.scheduleOnStarted(() -> { - this.blockchainRemoteService = serviceLocator.getBlockchainRemoteService(); - this.networkService = serviceLocator.getNetworkService(); - this.delegate = serviceLocator.getPeerService(); - setIsReady(true); - }); - } - - public PeerService addIncludeEndpointApi(String api) { - Preconditions.checkNotNull(api); - if (!includeEndpointApis.contains(api)) { - includeEndpointApis.add(api); - } - return this; - } - - public PeerService addIncludeEndpointApi(EndpointApi api) { - Preconditions.checkNotNull(api); - addIncludeEndpointApi(api.name()); - return this; - } - - public PeerService indexPeers(Peer peer) { - - try { - // Get the blockchain name from node - BlockchainParameters parameter = blockchainRemoteService.getParameters(peer); - if (parameter == null) { - logger.error(I18n.t("duniter4j.es.networkService.indexPeers.remoteParametersError", peer)); - return this; - } - String currencyName = parameter.getCurrency(); - - indexPeers(currencyName, peer); - - } catch(Exception e) { - logger.error("Error during indexAllPeers: " + e.getMessage(), e); - } - - return this; - } - - public PeerService indexPeers(String currencyName, Peer firstPeer) { - long timeStart = System.currentTimeMillis(); - - try { - logger.info(I18n.t("duniter4j.es.networkService.indexPeers.task", currencyName, firstPeer)); - - // Default filter - org.duniter.core.client.service.local.NetworkService.Filter filterDef = new org.duniter.core.client.service.local.NetworkService.Filter(); - filterDef.filterType = null; - filterDef.filterStatus = Peer.PeerStatus.UP; - filterDef.filterEndpoints = ImmutableList.copyOf(includeEndpointApis); - - // Default sort - org.duniter.core.client.service.local.NetworkService.Sort sortDef = new org.duniter.core.client.service.local.NetworkService.Sort(); - sortDef.sortType = null; - - List<Peer> peers = networkService.getPeers(firstPeer, filterDef, sortDef, threadPool.scheduler()); - delegate.save(currencyName, peers, true); - logger.info(I18n.t("duniter4j.es.networkService.indexPeers.succeed", currencyName, firstPeer, peers.size(), (System.currentTimeMillis() - timeStart))); - } catch(Exception e) { - logger.error("Error during indexBlocksFromNode: " + e.getMessage(), e); - } - - return this; - } - - public void listenAndIndexPeers(final Peer mainPeer) { - // Get the blockchain name from node - BlockchainParameters parameter = blockchainRemoteService.getParameters(mainPeer); - if (parameter == null) { - logger.error(I18n.t("duniter4j.es.networkService.indexPeers.remoteParametersError", mainPeer)); - return; - } - String currencyName = parameter.getCurrency(); - - // Default filter - NetworkService.Filter filterDef = new NetworkService.Filter(); - filterDef.filterType = null; - filterDef.filterStatus = Peer.PeerStatus.UP; - filterDef.filterEndpoints = ImmutableList.copyOf(includeEndpointApis); - filterDef.currency = currencyName; - - // Default sort - NetworkService.Sort sortDef = new NetworkService.Sort(); - sortDef.sortType = null; - - networkService.addPeersChangeListener(mainPeer, - peers -> logger.debug(String.format("[%s] Update peers: %s found", currencyName, CollectionUtils.size(peers))), - filterDef, sortDef, true /*autoreconnect*/, threadPool.scheduler()); - } - - public Long getMaxLastUpTime(String currencyId) { - return peerDao.getMaxLastUpTime(currencyId); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java deleted file mode 100644 index cdee4cd1a97509cd2b2fed78f3235180ea4283b8..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java +++ /dev/null @@ -1,126 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.beans.Bean; -import org.duniter.core.client.dao.CurrencyDao; -import org.duniter.core.client.dao.PeerDao; -import org.duniter.core.client.service.DataContext; -import org.duniter.core.client.service.HttpService; -import org.duniter.core.client.service.HttpServiceImpl; -import org.duniter.core.client.service.bma.*; -import org.duniter.core.client.service.local.CurrencyService; -import org.duniter.core.client.service.local.*; -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.duniter.elasticsearch.beans.ESBeanFactory; -import org.duniter.elasticsearch.dao.BlockDao; -import org.duniter.elasticsearch.dao.impl.BlockDaoImpl; -import org.duniter.elasticsearch.dao.impl.CurrencyDaoImpl; -import org.duniter.elasticsearch.dao.impl.PeerDaoImpl; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; -import org.elasticsearch.common.inject.Singleton; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; - -import java.io.IOException; - -@Singleton -public class ServiceLocator - extends org.duniter.core.client.service.ServiceLocator - { - private static final ESLogger logger = ESLoggerFactory.getLogger("duniter.service"); - - private static ESBeanFactory beanFactory = null; - - @Inject - public ServiceLocator() { - super(getOrCreateBeanFactory()); - if (logger.isDebugEnabled()) { - logger.debug("Starting Duniter4j ServiceLocator..."); - } - - org.duniter.core.client.service.ServiceLocator.setInstance(this); - } - - @Override - public void close() { - try { - super.close(); - } - catch (IOException e) { - throw new TechnicalException(e); - } - org.duniter.core.client.service.ServiceLocator.setInstance(null); - } - - public static ESBeanFactory getESBeanFactory() { - return getOrCreateBeanFactory(); - } - - /* -- Internal methods -- */ - - public static ESBeanFactory getOrCreateBeanFactory() { - if (beanFactory != null) { - return beanFactory; - } - beanFactory = new ESBeanFactory(); - - beanFactory.bind(BlockchainRemoteService.class, BlockchainRemoteServiceImpl.class) - .bind(NetworkRemoteService.class, NetworkRemoteServiceImpl.class) - .bind(WotRemoteService.class, WotRemoteServiceImpl.class) - .bind(TransactionRemoteService.class, TransactionRemoteServiceImpl.class) - .bind(CryptoService.class, Ed25519CryptoServiceImpl.class) - .bind(org.duniter.core.client.service.local.PeerService.class, PeerServiceImpl.class) - .bind(MailService.class, MailServiceImpl.class) - .bind(CurrencyService.class, CurrencyServiceImpl.class) - .bind(NetworkService.class, NetworkServiceImpl.class) - .bind(HttpService.class, HttpServiceImpl.class) - // Dao - .bind(CurrencyDao.class, CurrencyDaoImpl.class) - .bind(PeerDao.class, PeerDaoImpl.class) - .bind(BlockDao.class, BlockDaoImpl.class) - - .add(DataContext.class); - - return beanFactory; - } - - public static class Provider<T extends Bean> implements org.elasticsearch.common.inject.Provider<T> { - - private final Class<T> clazz; - - public Provider(Class<T> clazz) { - this.clazz = clazz; - } - - public T get() { - return getOrCreateBeanFactory().getBean(clazz); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java deleted file mode 100644 index 744cd2b59324842f3d9c41c7b4d940f2ef403419..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceModule.java +++ /dev/null @@ -1,86 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%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.core.beans.Bean; -import org.duniter.core.client.service.DataContext; -import org.duniter.core.client.service.HttpService; -import org.duniter.core.client.service.bma.BlockchainRemoteService; -import org.duniter.core.client.service.bma.NetworkRemoteService; -import org.duniter.core.client.service.bma.TransactionRemoteService; -import org.duniter.core.client.service.bma.WotRemoteService; -import org.duniter.core.client.service.local.CurrencyService; -import org.duniter.core.service.CryptoService; -import org.duniter.core.service.MailService; -import org.duniter.elasticsearch.PluginInit; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.service.changes.ChangeService; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class ServiceModule extends AbstractModule implements Module { - - @Override protected void configure() { - bind(ServiceLocator.class).asEagerSingleton(); - - // common services - bind(PluginSettings.class).asEagerSingleton(); - bind(ThreadPool.class).asEagerSingleton(); - bind(PluginInit.class).asEagerSingleton(); - bind(ChangeService.class).asEagerSingleton(); - bind(DocStatService.class).asEagerSingleton(); - bind(SynchroService.class).asEagerSingleton(); - - // blockchain indexation services - bind(BlockchainService.class).asEagerSingleton(); - bind(BlockchainListenerService.class).asEagerSingleton(); - bind(PeerService.class).asEagerSingleton(); - - // Duniter Client API beans - bindWithLocator(BlockchainRemoteService.class); - bindWithLocator(NetworkRemoteService.class); - bindWithLocator(WotRemoteService.class); - bindWithLocator(TransactionRemoteService.class); - bindWithLocator(org.duniter.core.client.service.local.PeerService.class); - bindWithLocator(CurrencyService.class); - bindWithLocator(HttpService.class); - //bindWithLocator(CurrencyDao.class); - //bindWithLocator(PeerDao.class); - bindWithLocator(DataContext.class); - - // Duniter Shared API beans - bindWithLocator(CryptoService.class); - bindWithLocator(MailService.class); - } - - - - /* protected methods */ - - protected <T extends Bean> void bindWithLocator(Class<T> clazz) { - bind(clazz).toProvider(new ServiceLocator.Provider<>(clazz)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeEvent.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeEvent.java deleted file mode 100644 index b01a30f13c29d2d6bf6f1e31be2a0bb6d9af1e5e..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeEvent.java +++ /dev/null @@ -1,111 +0,0 @@ -package org.duniter.elasticsearch.service.changes; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import com.fasterxml.jackson.annotation.JsonIgnore; -import org.elasticsearch.common.bytes.BytesReference; -import org.joda.time.DateTime; - -public class ChangeEvent { - private final String id; - private final String index; - private final String type; - private final DateTime timestamp; - private final Operation operation; - private final long version; - private final BytesReference source; - - public enum Operation { - INDEX,CREATE,DELETE - } - - public ChangeEvent(String index, String type, String id, DateTime timestamp, Operation operation, long version, BytesReference source) { - this.id = id; - this.index = index; - this.type = type; - this.timestamp = timestamp; - this.operation = operation; - this.version = version; - this.source = source; - } - - protected ChangeEvent(ChangeEvent event, boolean copySource) { - this.id = event.getId(); - this.index = event.getIndex(); - this.type = event.getType(); - this.timestamp = event.getTimestamp(); - this.operation = event.getOperation(); - this.version = event.getVersion(); - this.source = copySource ? event.getSource() : null; - } - - public String getId() { - return id; - } - - public Operation getOperation() { - return operation; - } - - public DateTime getTimestamp() { - return timestamp; - } - - public String getIndex() { - return index; - } - - public String getType() { - return type; - } - - public long getVersion() { - return version; - } - - public BytesReference getSource() { - return source; - } - - @JsonIgnore - public boolean hasSource() { - return source != null; - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeEvents.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeEvents.java deleted file mode 100644 index d7b5bb09fa43367d9b9d90faf5171cfa4dc6e389..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeEvents.java +++ /dev/null @@ -1,135 +0,0 @@ -package org.duniter.elasticsearch.service.changes; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import com.fasterxml.jackson.annotation.JsonIgnore; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.exception.InvalidFormatException; -import org.duniter.elasticsearch.util.bytes.BytesJsonNode; -import org.elasticsearch.common.bytes.BytesReference; -import org.elasticsearch.common.io.stream.BytesStreamOutput; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.json.JsonXContent; -import org.joda.time.DateTime; - -import java.io.IOException; - -public class ChangeEvents { - - private ChangeEvents() { - // helper class - } - - public static ChangeEvent fromJson(String json) { - return fromJson(new ObjectMapper(), json); - } - - public static ChangeEvent fromJson(ObjectMapper objectMapper, String json) { - try { - JsonNode actualObj = objectMapper.readTree(json); - String index = actualObj.get("_index").asText(); - String type = actualObj.get("_type").asText(); - String id = actualObj.get("_id").asText(); - DateTime timestamp = new DateTime(actualObj.get("_timestamp").asLong()); - ChangeEvent.Operation operation = ChangeEvent.Operation.valueOf(actualObj.get("_operation").asText()); - long version = actualObj.get("_version").asLong(); - - JsonNode sourceNode = actualObj.get("_source"); - BytesReference source = null; - if (sourceNode != null) { - source = new BytesJsonNode(sourceNode); - } - - return new ChangeEvent(index, type, id, timestamp, operation, version, source); - } catch (IOException e) { - throw new InvalidFormatException("Invalid record JSON: " + e.getMessage(), e); - } - } - - public static String toJson(ChangeEvent event) { - try { - XContentBuilder builder = new XContentBuilder(JsonXContent.jsonXContent, new BytesStreamOutput()); - builder.startObject() - .field("_index", event.getIndex()) - .field("_type", event.getType()) - .field("_id", event.getId()) - .field("_timestamp", event.getTimestamp()) - .field("_version", event.getVersion()) - .field("_operation", event.getOperation().toString()); - if (event.hasSource()) { - builder.rawField("_source", event.getSource()); - } - builder.endObject(); - - return builder.string(); - } catch (IOException e) { - throw new TechnicalException("Error while generating JSON from change event", e); - } - } - - public static JsonNode readTree(BytesReference source) throws IOException { - if (source == null) return null; - - if (source instanceof BytesJsonNode) { - // Avoid new deserialization - return ((BytesJsonNode) source).toJsonNode(); - } - - return new ObjectMapper().readTree(source.streamInput()); - } - - public static JsonNode readTree(ObjectMapper objectMapper, BytesReference source) throws IOException { - if (source == null) return null; - - if (source instanceof BytesJsonNode) { - // Avoid new deserialization - return ((BytesJsonNode) source).toJsonNode(); - } - - return objectMapper.readTree(source.streamInput()); - } - - public static <T> T readValue(BytesReference source, Class<T> clazz) throws IOException { - if (source == null) return null; - - return new ObjectMapper().readValue(source.streamInput(), clazz); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeService.java deleted file mode 100644 index 5f6be682b39efde3799e8ecc82e4f5178967411b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeService.java +++ /dev/null @@ -1,248 +0,0 @@ -package org.duniter.elasticsearch.service.changes; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.CollectionUtils; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.index.engine.Engine; -import org.elasticsearch.index.indexing.IndexingOperationListener; -import org.elasticsearch.index.shard.IndexShard; -import org.elasticsearch.indices.IndicesLifecycle; -import org.elasticsearch.indices.IndicesService; -import org.joda.time.DateTime; - -import java.util.*; - -public class ChangeService { - - public interface ChangeListener { - String getId(); - void onChange(ChangeEvent change); - Collection<ChangeSource> getChangeSources(); - } - - private static final String SETTING_PRIMARY_SHARD_ONLY = "duniter.changes.primaryShardOnly"; - - private final ESLogger log = Loggers.getLogger(ChangeService.class); - - private static final Map<String, ChangeListener> LISTENERS = new HashMap<>(); - - private static Map<String, ChangeSource> LISTENERS_SOURCES = new HashMap<>(); - private static Map<String, Integer> LISTENERS_SOURCES_USAGE_COUNT = new HashMap<>(); - - @Inject - public ChangeService(final Settings settings, IndicesService indicesService) { - final boolean allShards = !settings.getAsBoolean(SETTING_PRIMARY_SHARD_ONLY, Boolean.FALSE); - - - indicesService.indicesLifecycle().addListener(new IndicesLifecycle.Listener() { - @Override - public void afterIndexShardStarted(IndexShard indexShard) { - final String indexName = indexShard.routingEntry().getIndex(); - if (allShards || indexShard.routingEntry().primary()) { - - indexShard.indexingService().addListener(new IndexingOperationListener() { - @Override - public void postCreate(Engine.Create create) { - if (!hasListener(indexName, create.type(), create.id())) { - return; - } - - ChangeEvent change=new ChangeEvent( - indexName, - create.type(), - create.id(), - new DateTime(), - ChangeEvent.Operation.CREATE, - create.version(), - create.source() - ); - - addChange(change); - } - - @Override - public Engine.Delete preDelete(Engine.Delete delete) { - - return delete; - } - - @Override - public void postDelete(Engine.Delete delete) { - if (!hasListener(indexName, delete.type(), delete.id())) { - return; - } - - ChangeEvent change=new ChangeEvent( - indexName, - delete.type(), - delete.id(), - new DateTime(), - ChangeEvent.Operation.DELETE, - delete.version(), - null - ); - - addChange(change); - } - - @Override - public void postIndex(Engine.Index index, boolean created) { - if (!hasListener(indexName, index.type(), index.id())) { - return; - } - - ChangeEvent change = new ChangeEvent( - indexName, - index.type(), - index.id(), - new DateTime(), - created ? ChangeEvent.Operation.CREATE : ChangeEvent.Operation.INDEX, - index.version(), - index.source() - ); - - addChange(change); - } - - private boolean hasListener(String index, String type, String id) { - if (LISTENERS_SOURCES.isEmpty()) return false; - - for (ChangeSource source : LISTENERS_SOURCES.values()) { - if (source.apply(index, type, id)) { - return true; - } - } - - return false; - } - - private boolean apply(ChangeListener listener, ChangeEvent change) { - Collection<ChangeSource> sources = listener.getChangeSources(); - if (CollectionUtils.isEmpty(sources)) return true; - - for (ChangeSource source : sources) { - if (source.apply(change.getIndex(), change.getType(), change.getId())) { - return true; - } - } - - return false; - } - - private void addChange(ChangeEvent change) { - for (ChangeListener listener : LISTENERS.values()) { - if (apply(listener, change)) { - try { - listener.onChange(change); - } catch (Exception e) { - log.error("Failed to send message", e); - } - } - } - - } - }); - } - } - - }); - } - - public static void registerListener(ChangeListener listener) { - Preconditions.checkNotNull(listener); - Preconditions.checkNotNull(listener.getId()); - if (LISTENERS.containsKey(listener.getId())) { - throw new IllegalArgumentException("Listener with id [%s] already registered. Id should be unique"); - } - - // Add to list - LISTENERS.put(listener.getId(), listener); - - // Update sources - if (CollectionUtils.isNotEmpty(listener.getChangeSources())) { - for (ChangeSource source: listener.getChangeSources()) { - String sourceKey = source.toString(); - if (!LISTENERS_SOURCES.containsKey(sourceKey)) { - LISTENERS_SOURCES.put(sourceKey, source); - LISTENERS_SOURCES_USAGE_COUNT.put(sourceKey, 1); - } - else { - LISTENERS_SOURCES_USAGE_COUNT.put(sourceKey, LISTENERS_SOURCES_USAGE_COUNT.get(sourceKey)+1); - } - } - } - } - - /** - * Usefull when listener sources has changed - * @param listener - */ - public static void refreshListener(ChangeListener listener) { - unregisterListener(listener); - registerListener(listener); - } - - public static void unregisterListener(ChangeListener listener) { - LISTENERS.remove(listener.getId()); - - // Update sources - if (CollectionUtils.isNotEmpty(listener.getChangeSources())) { - for (ChangeSource source: listener.getChangeSources()) { - String sourceKey = source.toString(); - if (LISTENERS_SOURCES.containsKey(sourceKey)) { - int usageCount = LISTENERS_SOURCES_USAGE_COUNT.get(sourceKey) - 1; - if (usageCount > 0) { - LISTENERS_SOURCES_USAGE_COUNT.put(sourceKey, usageCount); - } - else { - LISTENERS_SOURCES.remove(sourceKey); - LISTENERS_SOURCES_USAGE_COUNT.remove(sourceKey); - } - } - } - } - } - - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeSource.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeSource.java deleted file mode 100644 index 046391cf2dacd743d7e58d1e720a1e9feabfcccf..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/changes/ChangeSource.java +++ /dev/null @@ -1,178 +0,0 @@ -package org.duniter.elasticsearch.service.changes; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import com.google.common.base.Joiner; -import com.google.common.collect.Sets; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import com.google.common.collect.ImmutableSet; -import org.duniter.core.util.StringUtils; - -import java.util.Set; - -public class ChangeSource { - private final Set<String> indices; - private final Set<String> types; - private final Set<String> ids; - - public ChangeSource() { - this.indices = Sets.newHashSet(); - this.types = Sets.newHashSet(); - this.ids = Sets.newHashSet(); - } - - public ChangeSource(String source) { - String[] parts = source.split("/"); - - indices = parts[0].equals("*") ? null : ImmutableSet.copyOf(parts[0].split(",")); - - if (parts.length > 1) { - types = parts[1].equals("*") ? null : ImmutableSet.copyOf(parts[1].split(",")); - } else { - types = null; - } - - if (parts.length > 2) { - ids = parts[2].equals("*") ? null : ImmutableSet.copyOf(parts[2].split(",")); - } else { - ids = null; - } - } - - public ChangeSource(String index, String type) { - this(index, type, null); - } - - public ChangeSource(String index, String type, String id) { - Preconditions.checkArgument(StringUtils.isNotBlank(index)); - indices = index.equals("*") ? null : ImmutableSet.of(index); - types = StringUtils.isBlank(type) || type.equals("*") ? null : ImmutableSet.of(type); - ids = StringUtils.isBlank(id) || id.equals("*") ? null : ImmutableSet.of(id); - } - - public Set<String> getIds() { - return ids; - } - - public Set<String> getIndices() { - return indices; - } - - public Set<String> getTypes() { - return types; - } - - public ChangeSource addIndex(String index){ - this.indices.add(index); - return this; - } - public ChangeSource addType(String type){ - this.types.add(type); - return this; - } - public ChangeSource addId(String id){ - this.ids.add(id); - return this; - } - - public String toString() { - StringBuilder sb = new StringBuilder(); - - // Add indices - Joiner joiner = Joiner.on(','); - if (CollectionUtils.isEmpty(indices)) { - sb.append('*'); - } - else { - joiner.appendTo(sb, indices); - } - - // Add types - if (CollectionUtils.isEmpty(types)) { - if (CollectionUtils.isNotEmpty(ids)) { - sb.append("/*"); - } - } - else { - sb.append('/'); - joiner.appendTo(sb, types); - } - - // Add ids - if (CollectionUtils.isNotEmpty(ids)) { - sb.append('/'); - joiner.appendTo(sb, ids); - } - return sb.toString(); - } - - public boolean apply(String index, String type, String id) { - if (indices != null && !indices.contains(index)) { - return false; - } - - if (types != null && !types.contains(type)) { - return false; - } - - if (ids != null && !ids.contains(id)) { - return false; - } - - return true; - } - - public boolean isEmpty() { - return indices == null && types == null && ids == null; - } - - public void merge(ChangeSource s) { - if (s == null) return; - if (CollectionUtils.isNotEmpty(s.getIndices())) { - indices.addAll(s.getIndices()); - } - if (CollectionUtils.isNotEmpty(s.getTypes())) { - types.addAll(s.getTypes()); - } - if (CollectionUtils.isNotEmpty(s.getIds())) { - ids.addAll(s.getIds()); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/AbstractSynchroAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/AbstractSynchroAction.java deleted file mode 100644 index a94411bd739e531c9137362ed43278b1eb43bb4d..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/AbstractSynchroAction.java +++ /dev/null @@ -1,596 +0,0 @@ -package org.duniter.elasticsearch.synchro; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.google.common.collect.Lists; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.methods.HttpUriRequest; -import org.apache.http.entity.StringEntity; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.HttpService; -import org.duniter.core.client.service.exception.HttpUnauthorizeException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.DuniterElasticsearchException; -import org.duniter.elasticsearch.exception.InvalidFormatException; -import org.duniter.elasticsearch.exception.InvalidSignatureException; -import org.duniter.elasticsearch.model.SearchResponse; -import org.duniter.elasticsearch.model.SearchScrollResponse; -import org.duniter.elasticsearch.model.SynchroResult; -import org.duniter.elasticsearch.service.AbstractService; -import org.duniter.elasticsearch.service.ServiceLocator; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeEvents; -import org.duniter.elasticsearch.service.changes.ChangeSource; -import org.duniter.elasticsearch.synchro.impl.NullSynchroActionResult; -import org.duniter.elasticsearch.synchro.impl.SynchroActionResultImpl; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.action.bulk.BulkItemResponse; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.bulk.BulkResponse; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.io.stream.BytesStreamOutput; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.json.JsonXContent; -import org.elasticsearch.index.query.QueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; - -import java.io.IOException; -import java.text.DateFormat; -import java.util.*; - -public abstract class AbstractSynchroAction extends AbstractService implements SynchroAction { - - private static final String SCROLL_PARAM_VALUE = "1m"; - - private static SynchroActionResult NULL_ACTION_RESULT = new NullSynchroActionResult(); - - private String fromIndex; - private String fromType; - private String toIndex; - private String toType; - private String issuerFieldName = Record.PROPERTY_ISSUER; - private String versionFieldName = Record.PROPERTY_TIME; - private String timeFieldName = versionFieldName; - private ChangeSource changeSource; - - private HttpService httpService; - - private boolean enableUpdate = false; - private boolean enableSignatureValidation = true; - private boolean enableTimeValidation = true; - private List<SourceConsumer> insertionListeners; - private List<SourceConsumer> updateListeners; - private List<SourceConsumer> validationListeners; - - private boolean trace = false; - - public AbstractSynchroAction(String index, String type, - Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool) { - this(index, type, index, type, client, pluginSettings, cryptoService, threadPool); - } - - public AbstractSynchroAction(String fromIndex, String fromType, - String toIndex, String toType, - Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool) { - super("duniter.p2p." + toIndex, client, pluginSettings, cryptoService); - this.fromIndex = fromIndex; - this.fromType = fromType; - this.toIndex = toIndex; - this.toType = toType; - this.changeSource = new ChangeSource() - .addIndex(fromIndex) - .addType(fromType); - this.trace = logger.isTraceEnabled(); - threadPool.scheduleOnStarted(() -> httpService = ServiceLocator.instance().getHttpService()); - } - - - @Override - public EndpointApi getEndPointApi() { - return EndpointApi.ES_USER_API; - } - - @Override - public ChangeSource getChangeSource() { - return changeSource; - } - - @Override - public void handleSynchronize(Peer peer, - long fromTime, - SynchroResult result) { - Preconditions.checkNotNull(peer); - Preconditions.checkArgument(fromTime >= 0); - Preconditions.checkNotNull(result); - - if (logger.isDebugEnabled()) { - if (Record.PROPERTY_TIME.equals(versionFieldName)) { - logger.debug(String.format("[%s] [%s] [%s/%s] Synchronization {since %s}...", peer.getCurrency(), peer, toIndex, toType, - DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.MEDIUM) - .format(new Date(fromTime * 1000)))); - } - else { - logger.debug(String.format("[%s] [%s] [%s/%s] Synchronization {where %s > %s}...", peer.getCurrency(), peer, toIndex, toType, versionFieldName, fromTime)); - } - } - - try { - QueryBuilder query = createQuery(fromTime); - synchronize(peer, query, result); - } - catch(Exception e1) { - // Log the first error - if (logger.isDebugEnabled()) { - logger.error(e1.getMessage(), e1); - } - else { - logger.error(e1.getMessage()); - } - } - } - - @Override - public void handleChange(Peer peer, ChangeEvent changeEvent) { - - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(changeEvent); - Preconditions.checkNotNull(changeEvent.getOperation()); - - String id = changeEvent.getId(); - String logPrefix = String.format("[%s] [%s] [%s/%s/%s] [WS]", peer.getCurrency(), peer, toIndex, toType, id); - - boolean skip = changeEvent.getOperation() == ChangeEvent.Operation.DELETE || - !enableUpdate && changeEvent.getOperation() == ChangeEvent.Operation.INDEX || - !changeEvent.hasSource(); - if (skip) { - if (trace) { - logger.trace(String.format("%s Ignoring change event of type [%s]", logPrefix, changeEvent.getOperation().name())); - } - return; - } - try { - if (trace) { - logger.trace(String.format("%s Processing new change event...", logPrefix)); - } - - JsonNode source = ChangeEvents.readTree(changeEvent.getSource()); - - // Save doc - save(changeEvent.getId(), source, logPrefix); - } - catch(Exception e1) { - // Log the first error - if (logger.isDebugEnabled()) { - logger.error(e1.getMessage(), e1); - } - else { - logger.error(e1.getMessage()); - } - } - } - - public void addInsertionListener(SourceConsumer listener) { - if (insertionListeners == null) { - insertionListeners = Lists.newArrayList(); - } - insertionListeners.add(listener); - } - - public void addUpdateListener(SourceConsumer listener) { - if (updateListeners == null) { - updateListeners = Lists.newArrayList(); - } - updateListeners.add(listener); - } - - public void addValidationListener(SourceConsumer listener) { - if (validationListeners == null) { - validationListeners = Lists.newArrayList(); - } - validationListeners.add(listener); - } - - /* -- protected methods -- */ - - protected void notifyInsertion(final String id, final JsonNode source, final SynchroActionResult actionResult) throws Exception { - if (CollectionUtils.isNotEmpty(insertionListeners)) { - for (SourceConsumer listener: insertionListeners) { - listener.accept(id, source, actionResult); - } - } - } - - protected void notifyUpdate(final String id, final JsonNode source, final SynchroActionResult actionResult) throws Exception { - if (CollectionUtils.isNotEmpty(updateListeners)) { - for (SourceConsumer listener: updateListeners) { - listener.accept(id, source, actionResult); - } - } - } - - protected void notifyValidation(final String id, - final JsonNode source, - final boolean allowOldDocuments, - final SynchroActionResult actionResult, - final String logPrefix) throws Exception { - - // Validate signature - if (enableSignatureValidation) { - try { - readAndVerifyIssuerSignature(source, issuerFieldName); - } catch (InvalidSignatureException e) { - // FIXME: some user/profile document failed ! - see issue #11 - // Il semble que le format JSON ne soit pas le même que celui qui a été signé - actionResult.addInvalidSignature(); - if (trace) { - logger.warn(String.format("%s %s.\n%s", logPrefix, e.getMessage(), source.toString())); - } - } - } - - // Validate time - if (enableTimeValidation) { - try { - verifyTime(source, allowOldDocuments, timeFieldName); - } catch (InvalidSignatureException e) { - actionResult.addInvalidTime(); - if (trace) { - logger.warn(String.format("%s %s.", logPrefix, e.getMessage())); - } - } - } - - if (CollectionUtils.isNotEmpty(validationListeners)) { - for (SourceConsumer listener : validationListeners) { - listener.accept(id, source, actionResult); - } - } - } - - protected QueryBuilder createQuery(long fromTime) { - - return QueryBuilders.boolQuery() - .should(QueryBuilders.rangeQuery("time").gte(fromTime)); - } - - private HttpPost createScrollRequest(Peer peer, - String fromIndex, - String fromType, - QueryBuilder query) { - HttpPost httpPost = new HttpPost(httpService.getPath(peer, fromIndex, fromType, "_search?scroll=" + SCROLL_PARAM_VALUE)); - httpPost.setHeader("Content-Type", "application/json;charset=UTF-8"); - - try { - // Query to String - BytesStreamOutput bos = new BytesStreamOutput(); - XContentBuilder builder = new XContentBuilder(JsonXContent.jsonXContent, bos); - query.toXContent(builder, null); - builder.flush(); - - // Sort on "_doc" - see https://www.elastic.co/guide/en/elasticsearch/reference/2.4/search-request-scroll.html - String content = String.format("{\"query\":%s,\"size\":%s, \"sort\": [\"_doc\"]}", - bos.bytes().toUtf8(), - pluginSettings.getIndexBulkSize()); - httpPost.setEntity(new StringEntity(content, "UTF-8")); - - if (trace) { - logger.trace(String.format("[%s] [%s] [%s/%s] Sending POST scroll request: %s", peer.getCurrency(), peer, fromIndex, fromType, content)); - } - - } catch (IOException e) { - throw new TechnicalException("Error while preparing search query: " + e.getMessage(), e); - } - - return httpPost; - } - - private HttpPost createNextScrollRequest(Peer peer, - String scrollId) { - - HttpPost httpPost = new HttpPost(httpService.getPath(peer, "_search", "scroll")); - httpPost.setHeader("Content-Type", "application/json;charset=UTF-8"); - httpPost.setEntity(new StringEntity(String.format("{\"scroll\": \"%s\", \"scroll_id\": \"%s\"}", - SCROLL_PARAM_VALUE, - scrollId), "UTF-8")); - return httpPost; - } - - private SearchScrollResponse executeAndParseRequest(Peer peer, HttpUriRequest request) { - try { - // Execute query & parse response - JsonNode node = httpService.executeRequest(request, JsonNode.class, String.class); - return node == null ? null : new SearchScrollResponse(node); - } catch (HttpUnauthorizeException e) { - throw new TechnicalException(String.format("[%s] [%s] [%s/%s] Unable to access (%s).", peer.getCurrency(), peer, fromIndex, fromType, e.getMessage()), e); - } catch (TechnicalException e) { - throw new TechnicalException(String.format("[%s] [%s] [%s/%s] Unable to scroll request: %s", peer.getCurrency(), peer, fromIndex, fromType, e.getMessage()), e); - } catch (Exception e) { - throw new TechnicalException(String.format("[%s] [%s] [%s/%s] Unable to parse response: ", peer.getCurrency(), peer, fromIndex, fromType, e.getMessage()), e); - } - } - - private void synchronize(Peer peer, - QueryBuilder query, - SynchroResult result) { - - if (!client.existsIndex(toIndex)) { - throw new TechnicalException(String.format("Unable to import changes. Index [%s] not exists", toIndex)); - } - - ObjectMapper objectMapper = getObjectMapper(); - - // DEV ONLY: skip - //if (!"user".equalsIgnoreCase(fromIndex) || !"profile".equalsIgnoreCase(fromType)) { - // return; - //} - - long counter = 0; - boolean stop = false; - String scrollId = null; - int total = 0; - while(!stop) { - SearchScrollResponse response; - if (scrollId == null) { - HttpUriRequest request = createScrollRequest(peer, fromIndex, fromType, query); - response = executeAndParseRequest(peer, request); - if (response != null) { - scrollId = response.getScrollId(); - total = response.getHits().getTotalHits(); - if (total > 0 && logger.isDebugEnabled()) { - logger.debug(String.format("[%s] [%s] [%s/%s] %s docs to check...", peer.getCurrency(), peer, toIndex, toType, total)); - } - } - } - else { - HttpUriRequest request = createNextScrollRequest(peer, scrollId); - response = executeAndParseRequest(peer, request); - } - - if (response == null) { - stop = true; - } - else { - counter += fetchAndSave(peer, response, objectMapper, result); - stop = counter >= total; - } - } - } - - private long fetchAndSave(final Peer peer, - final SearchScrollResponse response, - final ObjectMapper objectMapper, - final SynchroResult result) { - - - long counter = 0; - - SynchroActionResult actionResult = new SynchroActionResultImpl(); - - BulkRequestBuilder bulkRequest = client.prepareBulk(); - bulkRequest.setRefresh(true); - - for (Iterator<SearchResponse.SearchHit> hits = response.getHits(); hits.hasNext();){ - SearchResponse.SearchHit hit = hits.next(); - String id = hit.getId(); - JsonNode source = hit.getSource(); - - String logPrefix = String.format("[%s] [%s] [%s/%s/%s]", peer.getCurrency(), peer, toIndex, toType, id); - - if (source == null) { - logger.error(String.format("%s No source found. Skipping.", logPrefix)); - } - else { - counter++; - - // Save (create or update) - save(id, source, - objectMapper, - bulkRequest, - true, // allow old documents - actionResult, - logPrefix); - } - } - - if (bulkRequest.numberOfActions() > 0) { - - // Flush the bulk if not empty - BulkResponse bulkResponse = bulkRequest.get(); - Set<String> missingDocIds = new LinkedHashSet<>(); - - // If failures, continue but saveInBulk missing blocks - if (bulkResponse.hasFailures()) { - // process failures by iterating through each bulk response item - for (BulkItemResponse itemResponse : bulkResponse) { - boolean skip = !itemResponse.isFailed() - || missingDocIds.contains(itemResponse.getId()); - if (!skip) { - if (trace) { - logger.debug(String.format("[%s] [%s] [%s/%s] could not process _id=%s: %s. Skipping.", peer.getCurrency(), peer, toIndex, toType, itemResponse.getId(), itemResponse.getFailureMessage())); - } - missingDocIds.add(itemResponse.getId()); - } - } - } - } - - // update result - result.addInserts(toIndex, toType, actionResult.getInserts()); - result.addUpdates(toIndex, toType, actionResult.getUpdates()); - result.addDeletes(toIndex, toType, actionResult.getDeletes()); - result.addInvalidSignatures(toIndex, toType, actionResult.getInvalidSignatures()); - result.addInvalidTimes(toIndex, toType, actionResult.getInvalidTimes()); - - return counter; - } - - protected void save(String id, JsonNode source, String logPrefix) { - save(id, source, getObjectMapper(), null, false, NULL_ACTION_RESULT, logPrefix); - } - - protected void save(final String id, - final JsonNode source, - final ObjectMapper objectMapper, - final BulkRequestBuilder bulkRequest, - final boolean allowOldDocuments, - final SynchroActionResult actionResult, - final String logPrefix) { - - try { - String issuer = source.get(issuerFieldName).asText(); - if (StringUtils.isBlank(issuer)) { - throw new InvalidFormatException(String.format("Invalid format: missing or null %s field.", issuerFieldName)); - } - long version = source.get(versionFieldName).asLong(-1); - if (version == -1) { - throw new InvalidFormatException(String.format("Invalid format: missing or null %s field.", versionFieldName)); - } - - Map<String, Object> existingFields = client.getFieldsById(toIndex, toType, id, versionFieldName, issuerFieldName); - boolean exists = existingFields != null; - - // Insert (new doc) - if (!exists) { - - if (trace) { - logger.trace(String.format("%s insert found\n%s", logPrefix, source.toString())); - } - - // Validate doc - notifyValidation(id, source, allowOldDocuments, actionResult, logPrefix); - - // Execute insertion - IndexRequestBuilder request = client.prepareIndex(toIndex, toType, id) - .setSource(objectMapper.writeValueAsBytes(source)); - if (bulkRequest != null) { - bulkRequest.add(request); - } - else { - client.safeExecuteRequest(request, false); - } - - // Notify insert listeners - notifyInsertion(id, source, actionResult); - - actionResult.addInsert(); - } - - // Existing doc: do update (if enable) - else if (enableUpdate){ - - // Check same issuer - String existingIssuer = (String) existingFields.get(issuerFieldName); - if (!Objects.equals(issuer, existingIssuer)) { - throw new InvalidFormatException(String.format("Invalid document: not same [%s].", issuerFieldName)); - } - - // Check version - Number existingVersion = ((Number) existingFields.get(versionFieldName)); - boolean doUpdate = (existingVersion == null || version > existingVersion.longValue()); - - if (doUpdate) { - if (trace) { - logger.trace(String.format("%s found update\n%s", logPrefix, source.toString())); - } - - // Validate source - notifyValidation(id, source, allowOldDocuments, actionResult, logPrefix); - - // Execute update - UpdateRequestBuilder request = client.prepareUpdate(toIndex, toType, id); - request.setDoc(objectMapper.writeValueAsBytes(source)); - if (bulkRequest != null) { - bulkRequest.add(request); - } - else { - request.setRefresh(true); - client.safeExecuteRequest(request, false); - } - - // Notify insert listeners - notifyUpdate(id, source, actionResult); - - actionResult.addUpdate(); - } - } - - } catch (DuniterElasticsearchException e) { - // Skipping document: log, then continue - if (logger.isDebugEnabled()) { - logger.warn(String.format("%s %s. Skipping.\n%s", logPrefix, e.getMessage(), source.toString())); - } else { - logger.warn(String.format("%s %s. Skipping.", logPrefix, e.getMessage())); - } - } catch (Exception e) { - // Skipping document: log, then continue - logger.error(String.format("%s %s. Skipping.", logPrefix, e.getMessage()), e); - } - } - - protected void setIssuerFieldName(String issuerFieldName) { - this.issuerFieldName = issuerFieldName; - } - - protected void setVersionFieldName(String versionFieldName) { - this.versionFieldName = versionFieldName; - } - - protected void setTimeFieldName(String timeFieldName) { - this.timeFieldName = timeFieldName; - } - - protected ObjectMapper getObjectMapper() { - return JacksonUtils.getThreadObjectMapper(); - } - - protected void setEnableUpdate(boolean enableUpdate) { - this.enableUpdate = enableUpdate; - } - - protected void setEnableSignatureValidation(boolean enableSignatureValidation) { - this.enableSignatureValidation = enableSignatureValidation; - } - - protected void setEnableTimeValidation(boolean enableTimeValidation) { - this.enableTimeValidation = enableTimeValidation; - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroAction.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroAction.java deleted file mode 100644 index 7532c169c5e0529618878c5a46bb5e99a1176758..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroAction.java +++ /dev/null @@ -1,54 +0,0 @@ -package org.duniter.elasticsearch.synchro; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import com.google.common.collect.Lists; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.local.Peer; -import org.duniter.elasticsearch.model.SynchroResult; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeSource; - -public interface SynchroAction { - - interface SourceConsumer { - void accept(String id, JsonNode source, SynchroActionResult result) throws Exception; - } - - EndpointApi getEndPointApi(); - - ChangeSource getChangeSource(); - - void handleSynchronize(Peer peer, - long fromTime, - SynchroResult result); - - void handleChange(Peer peer, ChangeEvent changeEvent); - - void addInsertionListener(SourceConsumer listener); - - void addUpdateListener(SourceConsumer listener); - - void addValidationListener(SourceConsumer listener); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroActionResult.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroActionResult.java deleted file mode 100644 index 8a6e53cb8d5fcc738dd17afb99af520e1664f288..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroActionResult.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.synchro; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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% - */ - -public interface SynchroActionResult { - - void addInsert(); - void addUpdate(); - void addDelete(); - void addInvalidSignature(); - void addInvalidTime(); - - long getInserts(); - long getUpdates(); - long getDeletes(); - long getInvalidSignatures(); - long getInvalidTimes(); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroService.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroService.java deleted file mode 100644 index 5ecb5611df1c7c3d11d0e8bef4f25fc13c120a5c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/SynchroService.java +++ /dev/null @@ -1,504 +0,0 @@ -package org.duniter.elasticsearch.synchro; - -/* - * #%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 com.google.common.collect.*; -import org.apache.commons.io.IOUtils; -import org.apache.commons.lang3.ArrayUtils; -import org.duniter.core.client.dao.CurrencyDao; -import org.duniter.core.client.dao.PeerDao; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.bma.Endpoints; -import org.duniter.core.client.model.bma.NetworkPeering; -import org.duniter.core.client.model.local.Currency; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.HttpService; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.DateUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.core.util.websocket.WebsocketClientEndpoint; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.dao.SynchroExecutionDao; -import org.duniter.elasticsearch.model.SynchroExecution; -import org.duniter.elasticsearch.model.SynchroResult; -import org.duniter.elasticsearch.service.AbstractService; -import org.duniter.elasticsearch.service.ServiceLocator; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeEvents; -import org.duniter.elasticsearch.service.changes.ChangeSource; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.inject.Inject; - -import java.io.IOException; -import java.text.DateFormat; -import java.util.*; -import java.util.concurrent.TimeUnit; -import java.util.stream.Collectors; - -/** - * Created by blavenie on 27/10/16. - */ -public class SynchroService extends AbstractService { - - private static final String WS_CHANGES_URL = "/ws/_changes"; - - private HttpService httpService; - //private NetworkService networkService; - private final Set<EndpointApi> peerApiFilters = Sets.newHashSet(); - private final ThreadPool threadPool; - private final PeerDao peerDao; - private final CurrencyDao currencyDao; - private final SynchroExecutionDao synchroExecutionDao; - private List<WebsocketClientEndpoint> wsClientEndpoints = Lists.newArrayList(); - private List<SynchroAction> actions = Lists.newArrayList(); - private boolean forceFullResync = false; - - @Inject - public SynchroService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - ThreadPool threadPool, - CurrencyDao currencyDao, - PeerDao peerDao, - SynchroExecutionDao synchroExecutionDao, - final ServiceLocator serviceLocator) { - super("duniter.p2p", client, settings, cryptoService); - this.threadPool = threadPool; - this.currencyDao = currencyDao; - this.peerDao = peerDao; - this.synchroExecutionDao = synchroExecutionDao; - threadPool.scheduleOnStarted(() -> { - httpService = serviceLocator.getHttpService(); - //networkService = serviceLocator.getNetworkService(); - setIsReady(true); - }); - } - - public void register(SynchroAction action) { - Preconditions.checkNotNull(action); - Preconditions.checkNotNull(action.getEndPointApi()); - - if (!peerApiFilters.contains(action.getEndPointApi())) { - peerApiFilters.add(action.getEndPointApi()); - } - actions.add(action); - } - - /** - * Start scheduling doc stats update - * @return - */ - public SynchroService startScheduling() { - // Launch once, at startup (after a delay of 10s) - threadPool.schedule(() -> { - boolean launchAtStartup; - try { - // wait for some peers - launchAtStartup = waitPeersReady(); - } catch (InterruptedException e) { - return; // stop - } - - // If can be launched now: do it - if (launchAtStartup) { - - forceFullResync = pluginSettings.fullResyncAtStartup(); - - synchronize(); - - forceFullResync = false; - } - - // Schedule next execution, to 5 min before each hour - // (to make sure to be ready when computing doc stat - see DocStatService) - long nextExecutionDelay = DateUtils.nextHour().getTime() - System.currentTimeMillis() - 5 * 60 * 1000; - - // If next execution is too close, skip it - if (launchAtStartup && nextExecutionDelay < 5 * 60 * 1000) { - // add an hour - nextExecutionDelay += 60 * 60 * 1000; - } - - // Schedule every hour - threadPool.scheduleAtFixedRate( - this::synchronize, - nextExecutionDelay, - 60 * 60 * 1000 /* every hour */, - TimeUnit.MILLISECONDS); - }, - 10 * 1000 /*wait 10 s */ , - TimeUnit.MILLISECONDS); - - return this; - } - - public void synchronize() { - - final boolean enableSynchroWebsocket = pluginSettings.enableSynchroWebsocket(); - - // Closing all opened WS - if (enableSynchroWebsocket) { - closeWsClientEndpoints(); - } - - List<String> currencyIds; - try { - currencyIds = currencyDao.getCurrencyIds(); - } - catch (Exception e) { - logger.error("Could not retrieve indexed currencies", e); - currencyIds = null; - } - - if (CollectionUtils.isEmpty(currencyIds) || CollectionUtils.isEmpty(peerApiFilters)) { - logger.warn("Skipping synchronization: no indexed currency or no API configured"); - return; - } - - currencyIds.forEach(currencyId -> peerApiFilters.forEach(peerApiFilter -> { - - logger.info(String.format("[%s] [%s] Starting synchronization... {discovery: %s}", currencyId, peerApiFilter.name(), pluginSettings.enableSynchroDiscovery())); - - // Get peers for currencies and API - Collection<Peer> peers = getPeersFromApi(currencyId, peerApiFilter); - if (CollectionUtils.isNotEmpty(peers)) { - peers.forEach(p -> synchronizePeer(p, enableSynchroWebsocket)); - logger.info(String.format("[%s] [%s] Synchronization [OK]", currencyId, peerApiFilter.name())); - } else { - logger.info(String.format("[%s] [%s] Synchronization [OK] - no endpoint to synchronize", currencyId, peerApiFilter.name())); - } - } - )); - } - - public SynchroResult synchronizePeer(final Peer peer, boolean enableSynchroWebsocket) { - long startExecutionTime = System.currentTimeMillis(); - - // Check if peer alive and valid - boolean isAliveAndValid = isAliveAndValid(peer); - if (!isAliveAndValid) { - logger.warn(String.format("[%s] [%s] Not reachable, or not running on this currency. Skipping.", peer.getCurrency(), peer)); - return null; - } - - SynchroResult result = new SynchroResult(); - - // Get the last execution time (or 0 is never synchronized) - // If not the first synchro, add a delay to last execution time - // to avoid missing data because incorrect clock configuration - long lastExecutionTime = forceFullResync ? 0 : getLastExecutionTime(peer); - if (logger.isDebugEnabled() && lastExecutionTime > 0) { - logger.debug(String.format("[%s] [%s] Found last synchronization execution at {%s}. Will apply time offset of {-%s ms}", peer.getCurrency(), peer, - DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.MEDIUM) - .format(new Date(lastExecutionTime * 1000)), - pluginSettings.getSynchroTimeOffset())); - } - - final long fromTime = lastExecutionTime > 0 ? lastExecutionTime - pluginSettings.getSynchroTimeOffset() : 0; - - - if (logger.isInfoEnabled()) { - if (fromTime == 0) { - logger.info(String.format("[%s] [%s] Synchronization {ALL}...", peer.getCurrency(), peer)); - } - else { - logger.info(String.format("[%s] [%s] Synchronization delta since {%s}...", - peer.getCurrency(), - peer, - DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.MEDIUM) - .format(new Date(fromTime * 1000)))); - } - } - - // Execute actions - List<SynchroAction> executedActions = actions.stream() - .filter(a -> a.getEndPointApi().name().equals(peer.getApi())) - .map(a -> { - try { - a.handleSynchronize(peer, fromTime, result); - } catch(Exception e) { - logger.error(String.format("[%s] [%s] Failed to execute synchro action: %s", peer.getCurrency(), peer, e.getMessage()), e); - } - return a; - }) - .collect(Collectors.toList()); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] [%s] Synchronized in %s ms: %s", peer.getCurrency(), peer, System.currentTimeMillis() - startExecutionTime, result.toString())); - } - - saveExecution(peer, result, startExecutionTime); - - // Start listen changes on this peer - if (enableSynchroWebsocket) { - startListenChangesOnPeer(peer, executedActions); - } - - return result; - } - - /* -- protected methods -- */ - - protected List<Peer> getConfigIncludesPeers(final String currencyId, final EndpointApi api) { - Preconditions.checkNotNull(currencyId); - String[] endpoints = pluginSettings.getSynchroIncludesEndpoints(); - if (ArrayUtils.isEmpty(endpoints)) return null; - - List<Peer> peers = Lists.newArrayList(); - for (String endpoint: endpoints) { - try { - String[] endpointPart = endpoint.split(":"); - if (endpointPart.length > 2) { - logger.warn(String.format("Error in config: Unable to parse P2P endpoint [%s]: %s", endpoint)); - } - String epCurrencyId = (endpointPart.length == 2) ? endpointPart[0] : null /*optional*/; - - NetworkPeering.Endpoint ep = (endpointPart.length == 2) ? Endpoints.parse(endpointPart[1]) : Endpoints.parse(endpoint); - if (ep.api == api && (epCurrencyId == null || currencyId.equals(epCurrencyId))) { - Peer peer = Peer.newBuilder() - .setEndpoint(ep) - .setCurrency(currencyId) - .build(); - - String hash = cryptoService.hash(peer.computeKey()); - peer.setHash(hash); - peer.setId(hash); - - peers.add(peer); - } - - } catch (IOException e) { - logger.warn(String.format("Unable to parse P2P endpoint [%s]: %s", endpoint, e.getMessage())); - } - } - return peers; - } - - protected Collection<Peer> getPeersFromApi(final String currencyId, final EndpointApi api) { - Preconditions.checkNotNull(api); - Preconditions.checkArgument(StringUtils.isNotBlank(currencyId)); - - try { - - // Use map by URL, to avoid duplicated peer - Map<String, Peer> peersByUrls = Maps.newHashMap(); - - // Get peers from config - List<Peer> configPeers = getConfigIncludesPeers(currencyId, api); - if (CollectionUtils.isNotEmpty(configPeers)) { - configPeers.forEach(p -> peersByUrls.put(p.getUrl(), p)); - } - - // Get peers by pubkeys, from config - String[] includePubkeys = pluginSettings.getSynchroIncludesPubkeys(); - if (ArrayUtils.isNotEmpty(includePubkeys)) { - - // Get from DAO, by API and pubkeys - List<Peer> pubkeysPeers = peerDao.getPeersByCurrencyIdAndApiAndPubkeys(currencyId, api.name(), includePubkeys); - if (CollectionUtils.isNotEmpty(pubkeysPeers)) { - pubkeysPeers.stream() - .filter(Objects::nonNull) - .forEach(p -> peersByUrls.put(p.getUrl(), p)); - } - } - - // Add discovered peers - if (pluginSettings.enableSynchroDiscovery()) { - List<Peer> discoveredPeers = peerDao.getPeersByCurrencyIdAndApi(currencyId, api.name()); - if (CollectionUtils.isNotEmpty(discoveredPeers)) { - discoveredPeers.stream() - .filter(Objects::nonNull) - .forEach(p -> peersByUrls.put(p.getUrl(), p)); - } - } - - return peersByUrls.values(); - } - catch (Exception e) { - logger.error(String.format("Could not get peers for Api [%s]", api.name()), e); - return null; - } - } - - protected boolean hasSomePeers() { - - List<String> currencyIds = currencyDao.getCurrencyIds(); - if (CollectionUtils.isEmpty(currencyIds)) return false; - - for (String currencyId: currencyIds) { - boolean hasSome = peerDao.hasPeersUpWithApi(currencyId, peerApiFilters); - if (hasSome) return true; - } - - return false; - } - - protected boolean waitPeersReady() throws InterruptedException{ - final int sleepTime = 10 * 1000 /*10s*/; - - int maxWaitingDuration = 5 * 6 * sleepTime; // 5 min - int waitingDuration = 0; - while (!isReady() && !hasSomePeers()) { - // Wait 10s - Thread.sleep(sleepTime); - waitingDuration += sleepTime; - if (waitingDuration >= maxWaitingDuration) { - logger.warn(String.format("Could not start data synchronisation. No Peer found (after waiting %s min).", waitingDuration/60/1000)); - return false; // stop here - } - } - - // Wait again, to make sure all peers have been saved by NetworkService - Thread.sleep(sleepTime*2); - - return true; - } - - - protected long getLastExecutionTime(Peer peer) { - Preconditions.checkNotNull(peer); - - try { - SynchroExecution execution = synchroExecutionDao.getLastExecution(peer); - return execution != null ? execution.getTime() : 0; - } - catch (Exception e) { - logger.error(String.format("Error while saving last synchro execution time, for peer [%s]. Will resync all.", peer), e); - return 0; - } - } - - protected void saveExecution(Peer peer, SynchroResult result, long startExecutionTime) { - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(peer.getId()); - Preconditions.checkNotNull(result); - - try { - SynchroExecution execution = new SynchroExecution(); - execution.setCurrency(peer.getCurrency()); - execution.setPeer(peer.getId()); - execution.setApi(peer.getApi()); - execution.setExecutionTime(System.currentTimeMillis() - startExecutionTime); - execution.setResult(result); - - // Start execution time (in seconds) - execution.setTime(startExecutionTime/1000); - - synchroExecutionDao.save(execution); - } - catch (Exception e) { - logger.error(String.format("Error while saving synchro execution on peer [%s]", peer), e); - } - } - - protected void closeWsClientEndpoints() { - synchronized(wsClientEndpoints) { - // Closing all opened WS - wsClientEndpoints.forEach(IOUtils::closeQuietly); - wsClientEndpoints.clear(); - } - } - - protected void startListenChangesOnPeer(final Peer peer, - final List<SynchroAction> actions) { - // Listens changes on this peer - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(actions); - - // Compute a change source for ALL indices/types - final ChangeSource changeSource = new ChangeSource(); - actions.stream() - .map(SynchroAction::getChangeSource) - .filter(Objects::nonNull) - .forEach(changeSource::merge); - - // Prepare a map of actions by index/type - final ArrayListMultimap<String, SynchroAction> actionsBySource = ArrayListMultimap.create(actions.size(), 2); - actions.stream() - .forEach(a -> { - if (a.getChangeSource() != null) { - actionsBySource.put(a.getChangeSource().toString(), a); - } - }); - - // Get (or create) the websocket endpoint - WebsocketClientEndpoint wsClientEndPoint = httpService.getWebsocketClientEndpoint(peer, WS_CHANGES_URL, false); - - // filter on selected sources - wsClientEndPoint.sendMessage(changeSource.toString()); - - // add listener - wsClientEndPoint.registerListener( message -> { - try { - ChangeEvent changeEvent = ChangeEvents.fromJson(getObjectMapper(), message); - String source = changeEvent.getIndex() + "/" + changeEvent.getType(); - List<SynchroAction> sourceActions = actionsBySource.get(source); - - // Call each mapped actions - if (CollectionUtils.isNotEmpty(sourceActions)) { - sourceActions.forEach(a -> a.handleChange(peer, changeEvent)); - } - - } catch (Exception e) { - if (logger.isDebugEnabled()) { - logger.warn(String.format("[%s] Unable to process changes received by [/ws/_changes]: %s", peer, e.getMessage()), e); - } - else { - logger.warn(String.format("[%s] Unable to process changes received by [/ws/_changes]: %s", peer, e.getMessage())); - } - } - }); - - // Add to list - synchronized(wsClientEndpoints) { - wsClientEndpoints.add(wsClientEndPoint); - } - } - - protected boolean isAliveAndValid(Peer peer) { - Preconditions.checkNotNull(peer); - Preconditions.checkNotNull(peer.getCurrency()); - - try { - // TODO: check version is compatible - //String version = networkService.getVersion(peer); - - Currency currency = currencyDao.getById(peer.getCurrency()); - if (currency == null) return false; - - BlockchainBlock block = httpService.executeRequest(peer, String.format("/%s/block/0/_source", peer.getCurrency()), BlockchainBlock.class); - - return Objects.equals(block.getCurrency(), peer.getCurrency()) && - Objects.equals(block.getSignature(), currency.getFirstBlockSignature()); - - } - catch(Exception e) { - logger.debug(String.format("[%s] [%s] Peer not alive or invalid: %s", peer.getCurrency(), peer, e.getMessage())); - return false; - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/impl/NullSynchroActionResult.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/impl/NullSynchroActionResult.java deleted file mode 100644 index 1155602caa503054f389ccdecd14e3ff24517d89..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/impl/NullSynchroActionResult.java +++ /dev/null @@ -1,74 +0,0 @@ -package org.duniter.elasticsearch.synchro.impl; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.synchro.SynchroActionResult; - -public class NullSynchroActionResult implements SynchroActionResult { - - @Override - public void addInsert(){ - } - - @Override - public void addUpdate() { - } - - @Override - public void addDelete() { - } - - @Override - public void addInvalidSignature() { - } - - @Override - public void addInvalidTime() { - - } - - @Override - public long getInserts() { - return 0; - } - - @Override - public long getUpdates() { - return 0; - } - - @Override - public long getDeletes() { - return 0; - } - - @Override - public long getInvalidSignatures() { - return 0; - } - - @Override - public long getInvalidTimes() { - return 0; - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/impl/SynchroActionResultImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/impl/SynchroActionResultImpl.java deleted file mode 100644 index 1f7a4b550928132b088236f4be467f667b455d0e..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/synchro/impl/SynchroActionResultImpl.java +++ /dev/null @@ -1,80 +0,0 @@ -package org.duniter.elasticsearch.synchro.impl; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.util.PrimitiveIterators; -import org.duniter.elasticsearch.synchro.SynchroActionResult; - -public class SynchroActionResultImpl implements SynchroActionResult { - - private final PrimitiveIterators.OfLong insertHits = PrimitiveIterators.newLongSequence(); - private final PrimitiveIterators.OfLong updateHits = PrimitiveIterators.newLongSequence(); - private final PrimitiveIterators.OfLong deleteHits = PrimitiveIterators.newLongSequence(); - private final PrimitiveIterators.OfLong invalidSignatureHits = PrimitiveIterators.newLongSequence(); - private final PrimitiveIterators.OfLong invalidTimesHits = PrimitiveIterators.newLongSequence(); - - @Override - public void addInsert() { - insertHits.nextLong(); - } - - @Override - public void addUpdate() { - updateHits.nextLong(); - } - - @Override - public void addDelete() { - deleteHits.nextLong(); - } - - @Override - public void addInvalidSignature() { - invalidSignatureHits.nextLong(); - } - @Override - public void addInvalidTime() { - invalidTimesHits.nextLong(); - } - - @Override - public long getInserts() { - return insertHits.current(); - } - @Override - public long getUpdates() { - return updateHits.current(); - } - @Override - public long getDeletes() { - return deleteHits.current(); - } - @Override - public long getInvalidSignatures() { - return invalidSignatureHits.current(); - } - @Override - public long getInvalidTimes() { - return invalidTimesHits.current(); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/CompletableActionFuture.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/CompletableActionFuture.java deleted file mode 100644 index 6240361945386694002c8825d3f029844bc29082..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/CompletableActionFuture.java +++ /dev/null @@ -1,133 +0,0 @@ -package org.duniter.elasticsearch.threadpool; - -/* - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.ElasticsearchException; -import org.elasticsearch.ElasticsearchTimeoutException; -import org.elasticsearch.action.ActionFuture; -import org.elasticsearch.common.unit.TimeValue; -import org.elasticsearch.common.util.concurrent.UncategorizedExecutionException; - -import java.util.concurrent.CompletableFuture; -import java.util.concurrent.ExecutionException; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.TimeoutException; - -/** - * Classe qui offre l'interface de ES, tout en étant compatible avec java.util.concurrent.CompletableFuture - * (poar exemple pour faire des thenCompose()... - * @param <T> - */ -public class CompletableActionFuture<T> implements ActionFuture<T> { - - private final CompletableFuture<T> delegate; - - public CompletableActionFuture(CompletableFuture<T> delegate) { - this.delegate = delegate; - } - - @Override - public T actionGet() { - try { - return get(); - } catch (InterruptedException e) { - throw new IllegalStateException("Future got interrupted", e); - } catch (ExecutionException e) { - throw rethrowExecutionException(e); - } - } - - @Override - public T actionGet(String timeout) { - return actionGet(TimeValue.parseTimeValue(timeout, null, getClass().getSimpleName() + ".actionGet.timeout")); - } - - @Override - public T actionGet(long timeoutMillis) { - return actionGet(timeoutMillis, TimeUnit.MILLISECONDS); - } - - @Override - public T actionGet(TimeValue timeout) { - return actionGet(timeout.millis(), TimeUnit.MILLISECONDS); - } - - @Override - public T actionGet(long timeout, TimeUnit unit) { - try { - return get(timeout, unit); - } catch (TimeoutException e) { - throw new ElasticsearchTimeoutException(e.getMessage()); - } catch (InterruptedException e) { - throw new IllegalStateException("Future got interrupted", e); - } catch (ExecutionException e) { - throw rethrowExecutionException(e); - } - } - - @Override - public boolean cancel(boolean mayInterruptIfRunning) { - return delegate.cancel(mayInterruptIfRunning); - } - - @Override - public boolean isCancelled() { - return delegate.isCancelled(); - } - - @Override - public boolean isDone() { - return delegate.isDone(); - } - - @Override - public T get() throws InterruptedException, ExecutionException { - return delegate.get(); - } - - @Override - public T get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { - return delegate.get(timeout, unit); - } - - public CompletableFuture<T> getCompletableFuture() { - return delegate; - } - - static RuntimeException rethrowExecutionException(ExecutionException e) { - if (e.getCause() instanceof ElasticsearchException) { - ElasticsearchException esEx = (ElasticsearchException) e.getCause(); - Throwable root = esEx.unwrapCause(); - if (root instanceof ElasticsearchException) { - return (ElasticsearchException) root; - } else if (root instanceof RuntimeException) { - return (RuntimeException) root; - } - return new UncategorizedExecutionException("Failed execution", root); - } else if (e.getCause() instanceof RuntimeException) { - return (RuntimeException) e.getCause(); - } else { - return new UncategorizedExecutionException("Failed execution", e); - } - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/LoggingScheduledThreadPoolExecutor.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/LoggingScheduledThreadPoolExecutor.java deleted file mode 100644 index 4fb5efd9bc6cd49c9d6158b333d48d38bc79166b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/LoggingScheduledThreadPoolExecutor.java +++ /dev/null @@ -1,111 +0,0 @@ -package org.duniter.elasticsearch.threadpool; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.common.logging.ESLogger; - -import java.util.concurrent.*; - -public class LoggingScheduledThreadPoolExecutor extends ScheduledThreadPoolExecutor { - - private final ESLogger logger; - - public LoggingScheduledThreadPoolExecutor(ESLogger logger, - int corePoolSize, - ThreadFactory threadFactory, - RejectedExecutionHandler handler) { - super(corePoolSize, threadFactory, handler); - this.logger =logger; - } - - protected <V> RunnableScheduledFuture<V> decorateTask( - Runnable r, RunnableScheduledFuture<V> task) { - return new LoggingTask<V>(logger, task); - } - - protected <V> RunnableScheduledFuture<V> decorateTask( - Callable<V> c, RunnableScheduledFuture<V> task) { - return new LoggingTask<V>(logger, task); - } - - - static class LoggingTask<V> implements RunnableScheduledFuture<V> { - private final RunnableScheduledFuture<V> task; - private final ESLogger logger; - - public LoggingTask(ESLogger logger, RunnableScheduledFuture<V> task) { - this.task = task; - this.logger = logger; - } - - @Override - public void run() { - try { - task.run(); - } catch (Throwable e) { - logger.warn(String.format("Failed to execute a task: %s", e.getMessage()), e); - } - } - - @Override - public boolean isPeriodic() { - return task.isPeriodic(); - } - - @Override - public int compareTo(Delayed o) { - return task.compareTo(o); - } - - @Override - public boolean cancel(boolean mayInterruptIfRunning) { - return task.cancel(mayInterruptIfRunning); - } - - @Override - public boolean isCancelled() { - return task.isCancelled(); - } - - @Override - public boolean isDone() { - return task.isDone(); - } - - @Override - public V get() throws InterruptedException, ExecutionException { - return task.get(); - } - - @Override - public V get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { - return task.get(timeout, unit); - } - - @Override - public long getDelay(TimeUnit unit) { - return task.getDelay(unit); - } - } - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/RetryPolicy.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/RetryPolicy.java deleted file mode 100644 index 04ed708df0d2a5bb3346b8352cf7629cd18678dd..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/RetryPolicy.java +++ /dev/null @@ -1,65 +0,0 @@ -package org.duniter.elasticsearch.threadpool; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.util.concurrent.EsAbortPolicy; - -import java.util.concurrent.ScheduledThreadPoolExecutor; -import java.util.concurrent.ThreadPoolExecutor; -import java.util.concurrent.TimeUnit; - -public class RetryPolicy extends EsAbortPolicy { - - private final ESLogger logger; - private final long retryDelay; - private final TimeUnit timeUnit; - - public RetryPolicy(long retryDelay, TimeUnit timeUnit) { - this(Loggers.getLogger("duniter.threadpool.policy"), retryDelay, timeUnit); - } - - public RetryPolicy(ESLogger logger, long retryDelay, TimeUnit timeUnit) { - super(); - this.logger = logger; - this.retryDelay = retryDelay; - this.timeUnit = timeUnit; - } - - @Override - public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) { - - if (executor instanceof ScheduledThreadPoolExecutor) { - ScheduledThreadPoolExecutor scheduledExecutorService = (ScheduledThreadPoolExecutor)executor; - scheduledExecutorService.schedule(r, retryDelay, timeUnit); - logger.warn(String.format("Scheduler queue is full (max pool size = %s). Will retry later...", - executor.getMaximumPoolSize())); - } - else { - logger.error("Scheduler queue is full (max pool size = %s). Operation is rejected !"); - super.rejectedExecution(r, executor); - } - } - -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/ScheduledActionFuture.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/ScheduledActionFuture.java deleted file mode 100644 index 473e807f6ea292a2ac1771538b3719ceb78d7743..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/ScheduledActionFuture.java +++ /dev/null @@ -1,124 +0,0 @@ -package org.duniter.elasticsearch.threadpool; - -/* - * #%L - * Duniter4j :: ElasticSearch Core 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.elasticsearch.ElasticsearchException; -import org.elasticsearch.ElasticsearchTimeoutException; -import org.elasticsearch.action.ActionFuture; -import org.elasticsearch.common.unit.TimeValue; -import org.elasticsearch.common.util.concurrent.UncategorizedExecutionException; - -import java.util.concurrent.ExecutionException; -import java.util.concurrent.ScheduledFuture; -import java.util.concurrent.TimeUnit; -import java.util.concurrent.TimeoutException; - -public class ScheduledActionFuture<T> implements ActionFuture<T> { - - private final ScheduledFuture<T> delegate; - - public ScheduledActionFuture(ScheduledFuture<T> delegate) { - this.delegate = delegate; - } - - @Override - public T actionGet() { - try { - return get(); - } catch (InterruptedException e) { - throw new IllegalStateException("Future got interrupted", e); - } catch (ExecutionException e) { - throw rethrowExecutionException(e); - } - } - - @Override - public T actionGet(String timeout) { - return actionGet(TimeValue.parseTimeValue(timeout, null, getClass().getSimpleName() + ".actionGet.timeout")); - } - - @Override - public T actionGet(long timeoutMillis) { - return actionGet(timeoutMillis, TimeUnit.MILLISECONDS); - } - - @Override - public T actionGet(TimeValue timeout) { - return actionGet(timeout.millis(), TimeUnit.MILLISECONDS); - } - - @Override - public T actionGet(long timeout, TimeUnit unit) { - try { - return get(timeout, unit); - } catch (TimeoutException e) { - throw new ElasticsearchTimeoutException(e.getMessage()); - } catch (InterruptedException e) { - throw new IllegalStateException("Future got interrupted", e); - } catch (ExecutionException e) { - throw rethrowExecutionException(e); - } - } - - @Override - public boolean cancel(boolean mayInterruptIfRunning) { - return delegate.cancel(mayInterruptIfRunning); - } - - @Override - public boolean isCancelled() { - return delegate.isCancelled(); - } - - @Override - public boolean isDone() { - return delegate.isDone(); - } - - @Override - public T get() throws InterruptedException, ExecutionException { - return delegate.get(); - } - - @Override - public T get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { - return delegate.get(timeout, unit); - } - - static RuntimeException rethrowExecutionException(ExecutionException e) { - if (e.getCause() instanceof ElasticsearchException) { - ElasticsearchException esEx = (ElasticsearchException) e.getCause(); - Throwable root = esEx.unwrapCause(); - if (root instanceof ElasticsearchException) { - return (ElasticsearchException) root; - } else if (root instanceof RuntimeException) { - return (RuntimeException) root; - } - return new UncategorizedExecutionException("Failed execution", root); - } else if (e.getCause() instanceof RuntimeException) { - return (RuntimeException) e.getCause(); - } else { - return new UncategorizedExecutionException("Failed execution", e); - } - } -} \ No newline at end of file diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/ThreadPool.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/ThreadPool.java deleted file mode 100644 index 06689ebafa78951bde7f7be57a0cc5849633acf8..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/threadpool/ThreadPool.java +++ /dev/null @@ -1,294 +0,0 @@ -package org.duniter.elasticsearch.threadpool; - -/* - * #%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 com.google.common.collect.Lists; -import org.duniter.core.util.Preconditions; -import org.elasticsearch.action.admin.cluster.stats.ClusterStatsRequestBuilder; -import org.elasticsearch.action.admin.cluster.stats.ClusterStatsResponse; -import org.elasticsearch.client.Client; -import org.elasticsearch.cluster.health.ClusterHealthStatus; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.component.Lifecycle; -import org.elasticsearch.common.component.LifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.common.unit.TimeValue; -import org.elasticsearch.common.util.concurrent.EsExecutors; -import org.elasticsearch.transport.TransportService; -import org.nuiton.i18n.I18n; - -import java.util.List; -import java.util.concurrent.*; - -/** - * Manage thread pool, to execute tasks asynchronously. - * Created by eis on 17/06/16. - */ -public class ThreadPool extends AbstractLifecycleComponent<ThreadPool> { - - private ScheduledThreadPoolExecutor scheduler = null; - private final Injector injector; - private final ESLogger logger = Loggers.getLogger("duniter.threadpool"); - - private final org.elasticsearch.threadpool.ThreadPool delegate; - - private final List<Runnable> afterStartedCommands; - - @Inject - public ThreadPool(Settings settings, - Injector injector, - org.elasticsearch.threadpool.ThreadPool esThreadPool - ) { - super(settings); - this.injector = injector; - this.afterStartedCommands = Lists.newArrayList(); - - this.delegate = esThreadPool; - - int availableProcessors = EsExecutors.boundedNumberOfProcessors(settings); - this.scheduler = new LoggingScheduledThreadPoolExecutor(logger, availableProcessors, - EsExecutors.daemonThreadFactory(settings, "duniter-scheduler"), - new RetryPolicy(1, TimeUnit.SECONDS)); - this.scheduler.setExecuteExistingDelayedTasksAfterShutdownPolicy(false); - this.scheduler.setContinueExistingPeriodicTasksAfterShutdownPolicy(false); - this.scheduler.setRemoveOnCancelPolicy(true); - } - - public void doStart(){ - if (logger.isDebugEnabled()) { - logger.debug("Starting Duniter4j ThreadPool..."); - } - - if (!afterStartedCommands.isEmpty()) { - scheduleOnStarted(() -> { - afterStartedCommands.forEach(command -> command.run()); - this.afterStartedCommands.clear(); - }); - } - } - - public void doStop(){ - scheduler.shutdown(); - } - - public void doClose() {} - - /** - * Schedules an rest when node is started (allOfToList services and modules ready) - * - * @param job the rest to execute when node started - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public void scheduleOnStarted(Runnable job) { - Preconditions.checkNotNull(job); - scheduleAfterServiceState(TransportService.class, Lifecycle.State.STARTED, job); - } - - /** - * Schedules an rest when cluster is ready AND has one of the expected health status - * - * @param job the rest to execute - * @param expectedStatus expected health status, to run the job - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public void scheduleOnClusterHealthStatus(Runnable job, ClusterHealthStatus... expectedStatus) { - Preconditions.checkNotNull(job); - - Preconditions.checkArgument(expectedStatus.length > 0); - - scheduleOnStarted(() -> { - if (waitClusterHealthStatus(expectedStatus)) { - // continue - job.run(); - } - }); - } - - /** - * Schedules an rest when cluster is ready - * - * @param job the rest to execute - * @param expectedStatus expected health status, to run the job - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public void scheduleOnClusterReady(Runnable job) { - scheduleOnClusterHealthStatus(job, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); - } - - /** - * Schedules an rest that runs on the scheduler thread, when possible (0 delay). - * - * @param command the rest to take - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public ScheduledActionFuture<?> schedule(Runnable command) { - return schedule(command, new TimeValue(0)); - } - - /** - * Schedules an rest that runs on the scheduler thread, after a delay. - * - * @param command the rest to take - * @param name @see {@link org.elasticsearch.threadpool.ThreadPool.Names} - * @param delay the delay interval - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public ScheduledActionFuture<?> schedule(Runnable command, String name, TimeValue delay) { - if (name == null) { - return new ScheduledActionFuture<>(scheduler.schedule(command, delay.millis(), TimeUnit.MILLISECONDS)); - } - return new ScheduledActionFuture<>(delegate.schedule(delay, - name, - command)); - } - - - public ScheduledActionFuture<?> schedule(Runnable command, - long delay, TimeUnit unit) { - - return new ScheduledActionFuture<>(scheduler.schedule(command, delay, unit)); - } - - /** - * Schedules an rest that runs on the scheduler thread, after a delay. - * - * @param command the rest to take - * @param delay the delay interval - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public ScheduledActionFuture<?> schedule(Runnable command, TimeValue delay) { - return schedule(command, null, delay); - } - - /** - * Schedules a periodic rest that always runs on the scheduler thread. - * - * @param command the rest to take - * @param initialDelay the initial delay - * @param period the period - * @param timeUnit the time unit - * @return a ScheduledFuture who's get will return when the task is complete and throw an exception if it is canceled - */ - public ScheduledActionFuture<?> scheduleAtFixedRate(Runnable command, long initialDelay, long period, TimeUnit timeUnit) { - long initialDelayMs = new TimeValue(initialDelay, timeUnit).millis(); - long periodMs = new TimeValue(period, timeUnit).millis(); - return new ScheduledActionFuture<>(scheduleAtFixedRateWorkaround(command, initialDelayMs, periodMs)); - } - - /* -- protected methods -- */ - - protected <T extends LifecycleComponent<T>> ScheduledActionFuture<?> scheduleAfterServiceState(Class<T> waitingServiceClass, - final Lifecycle.State waitingState, - final Runnable job) { - Preconditions.checkNotNull(waitingServiceClass); - Preconditions.checkNotNull(waitingState); - Preconditions.checkNotNull(job); - - final T service = injector.getInstance(waitingServiceClass); - return schedule(() -> { - while(service.lifecycleState() != waitingState) { - try { - Thread.sleep(100); // wait 100 ms - } - catch(InterruptedException e) { - } - } - - // continue - job.run(); - }, TimeValue.timeValueSeconds(10)); - } - - public boolean waitClusterHealthStatus(ClusterHealthStatus... expectedStatus) { - Preconditions.checkNotNull(expectedStatus); - Preconditions.checkArgument(expectedStatus.length > 0); - - Client client = injector.getInstance(Client.class); - ClusterStatsRequestBuilder statsRequest = client.admin().cluster().prepareClusterStats(); - ClusterStatsResponse stats = null; - boolean canContinue = false; - boolean firstTry = true; - while (!canContinue) { - try { - if (stats != null) Thread.sleep(100); // wait 100 ms - stats = statsRequest.execute().get(); - for (ClusterHealthStatus status: expectedStatus) { - if (stats.getStatus() == status) { - if (!firstTry && logger.isDebugEnabled()) { - logger.debug(I18n.t("duniter4j.threadPool.clusterHealthStatus.changed", status.name())); - } - canContinue = true; - break; - } - } - firstTry = false; - } catch (ExecutionException e) { - // Continue - } catch (InterruptedException e) { - return false; // stop - } - } - - return canContinue; - } - - /** - * This method use a workaround to execution schedule at fixed time, because standard call of scheduler.scheduleAtFixedRate - * does not worked !! - **/ - protected ScheduledFuture<?> scheduleAtFixedRateWorkaround(final Runnable command, final long initialDelayMs, final long periodMs) { - final long expectedNextExecutionTime = System.currentTimeMillis() + initialDelayMs + periodMs; - - return scheduler.schedule( - () -> { - try { - command.run(); - } catch (Throwable t) { - logger.error("Error while processing subscriptions", t); - } - - long nextDelayMs = expectedNextExecutionTime - System.currentTimeMillis(); - - // When an execution duration is too long, go to next execution time. - while (nextDelayMs < 0) { - nextDelayMs += periodMs; - } - - // Schedule the next execution - scheduleAtFixedRateWorkaround(command, nextDelayMs, periodMs); - }, - initialDelayMs, - TimeUnit.MILLISECONDS) - ; - } - - public ScheduledExecutorService scheduler() { - return scheduler; - } - - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/Desktop.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/Desktop.java deleted file mode 100644 index 670ad02f06ea2947d70715d1ed9972cfaef569d1..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/Desktop.java +++ /dev/null @@ -1,63 +0,0 @@ -package org.duniter.elasticsearch.util; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.util.os.win.WindowsPower; -import org.apache.commons.lang3.SystemUtils; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public abstract class Desktop { - - private static final Logger LOG = LoggerFactory.getLogger(Desktop.class); - - static DesktopPower desktopPower = null; - - public static DesktopPower getDesktopPower() { - if (desktopPower == null) { - - - if (SystemUtils.IS_OS_WINDOWS) { - // All Windows version are handled with WindowsPower class - try { - desktopPower = new WindowsPower(); - } catch (Exception e) { - LOG.error(e.getLocalizedMessage(), e); - } - - - } else if (SystemUtils.IS_OS_LINUX) { - - // TODO create a Linux/UnixPower because (for example) Kubuntu sends KILL signal when it shutdown, it should sent TERM !! - } - - - } - - return desktopPower; - } - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/DesktopPower.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/DesktopPower.java deleted file mode 100644 index 70030d4cb40a79247bd5a75fce21599c064d7772..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/DesktopPower.java +++ /dev/null @@ -1,55 +0,0 @@ -package org.duniter.elasticsearch.util; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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 java.util.HashSet; -import java.util.Set; - -public abstract class DesktopPower { - - public interface Listener { - /** - * os asks to App to quit. - * - * Windows machines - on reboot / logout. Mac - on reboot / logout + - * Command + Q Linux - reboot / logout - */ - void quit(); - - } - - protected Set<Listener> listeners = new HashSet<Listener>(); - - public void addListener(Listener l) { - listeners.add(l); - } - - public void removeListener(Listener l) { - listeners.remove(l); - } - - abstract public void close(); - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/bytes/BytesJsonNode.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/bytes/BytesJsonNode.java deleted file mode 100644 index a8c8e9aaffa2bc452c9c02e8844fd796bb9b5d3f..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/bytes/BytesJsonNode.java +++ /dev/null @@ -1,152 +0,0 @@ -package org.duniter.elasticsearch.util.bytes; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.JsonNode; -import com.fasterxml.jackson.databind.ObjectMapper; -import org.apache.lucene.util.BytesRef; -import org.elasticsearch.ElasticsearchException; -import org.elasticsearch.common.bytes.BytesArray; -import org.elasticsearch.common.bytes.BytesReference; -import org.elasticsearch.common.io.stream.StreamInput; -import org.jboss.netty.buffer.ChannelBuffer; - -import java.io.IOException; -import java.io.OutputStream; -import java.nio.channels.GatheringByteChannel; -import java.util.Objects; - -public class BytesJsonNode implements BytesReference { - - private JsonNode node; - private BytesArray delegate; - private ObjectMapper objectMapper; - - public BytesJsonNode(JsonNode node) { - this(node, new ObjectMapper()); - } - - public BytesJsonNode(JsonNode node, ObjectMapper objectMapper) { - this.node = node; - this.objectMapper = objectMapper; - } - - public byte get(int index) { - return getOrInitDelegate().get(index); - } - - public int length() { - return getOrInitDelegate().length(); - } - - public BytesReference slice(int from, int length) { - return getOrInitDelegate().slice(from, length); - } - - public StreamInput streamInput() { - return getOrInitDelegate().streamInput(); - } - - public void writeTo(OutputStream os) throws IOException { - objectMapper.writeValue(os, node); - } - - public void writeTo(GatheringByteChannel channel) throws IOException { - getOrInitDelegate().writeTo(channel); - } - - public byte[] toBytes() { - try { - return objectMapper.writeValueAsBytes(node); - } - catch(JsonProcessingException e) { - throw new ElasticsearchException(e); - } - } - - public BytesArray toBytesArray() { - return getOrInitDelegate(); - } - - public BytesArray copyBytesArray() { - return getOrInitDelegate().copyBytesArray(); - } - - public ChannelBuffer toChannelBuffer() { - return getOrInitDelegate().toChannelBuffer(); - } - - public boolean hasArray() { - return true; - } - - public byte[] array() { - return toBytes(); - } - - public int arrayOffset() { - return getOrInitDelegate().arrayOffset(); - } - - public String toUtf8() { - return getOrInitDelegate().toUtf8(); - } - - public BytesRef toBytesRef() { - return getOrInitDelegate().toBytesRef(); - } - - public BytesRef copyBytesRef() { - return getOrInitDelegate().copyBytesRef(); - } - - public int hashCode() { - return getOrInitDelegate().hashCode(); - } - - public JsonNode toJsonNode() { - return node; - } - - public JsonNode copyJsonNode() { - return node.deepCopy(); - } - - public boolean equals(Object obj) { - return Objects.equals(this, obj); - } - - - protected BytesArray getOrInitDelegate() { - if (delegate == null) { - try { - this.delegate = new BytesArray(objectMapper.writeValueAsBytes(node)); - } - catch(JsonProcessingException e) { - throw new ElasticsearchException(e); - } - } - return delegate; - } -} 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 deleted file mode 100644 index 655e14603d4fc75452a72fbbe9cde1cb10b5a4a5..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/opengraph/OGData.java +++ /dev/null @@ -1,16 +0,0 @@ -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; - - public Integer imageHeight; - public Integer imageWidth; - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/WindowsPower.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/WindowsPower.java deleted file mode 100644 index eff638201756b12f2ea4f9c8d8ba806dfd9783bf..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/WindowsPower.java +++ /dev/null @@ -1,232 +0,0 @@ -package org.duniter.elasticsearch.util.os.win; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.Native; -import com.sun.jna.platform.win32.Kernel32; -import com.sun.jna.platform.win32.User32; -import com.sun.jna.platform.win32.WinDef.*; -import com.sun.jna.platform.win32.WinUser.HHOOK; -import com.sun.jna.platform.win32.WinUser.HOOKPROC; -import com.sun.jna.platform.win32.WinUser.MSG; -import org.duniter.elasticsearch.util.DesktopPower; -import org.duniter.elasticsearch.util.os.win.handle.CWPSSTRUCT; -import org.duniter.elasticsearch.util.os.win.handle.HANDLER_ROUTINE; -import org.duniter.elasticsearch.util.os.win.handle.WNDPROC; -import org.duniter.elasticsearch.util.os.win.libs.Kernel32Ex; -import org.duniter.elasticsearch.util.os.win.wrap.GetLastErrorException; -import org.duniter.elasticsearch.util.os.win.wrap.WNDCLASSEXWrap; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import javax.swing.*; - -public class WindowsPower extends DesktopPower { - - private static final Logger LOG = LoggerFactory.getLogger(WindowsPower.class); - - public static final int WM_QUERYENDSESSION = 17; - public static final int WM_ENDSESSION = 22; - public static final int WH_CALLWNDPROC = 4; - - public class MessagePump implements Runnable { - Thread t; - - WNDCLASSEXWrap wc; - WNDPROC WndProc; - HWND hWnd; - HINSTANCE hInstance; - - final Object lock = new Object(); - - public MessagePump() { - t = new Thread(this, WindowsPower.class.getSimpleName()); - } - - public void start() { - synchronized (lock) { - t.start(); - try { - lock.wait(); - } catch (InterruptedException e) { - Thread.currentThread().interrupt(); - } - } - } - - void create() { - WndProc = new WNDPROC() { - public LRESULT callback(HWND hWnd, int msg, WPARAM wParam, LPARAM lParam) { - switch (msg) { - case WM_ENDSESSION: - return new LRESULT(0); - case WM_QUERYENDSESSION: - JOptionPane.showMessageDialog(null, "exit"); - callListeners("WM_QUERYENDSESSION callback"); - return new LRESULT(0); - case User32.WM_QUIT: - User32.INSTANCE.PostMessage(hWnd, User32.WM_QUIT, null, null); - break; - } - - return User32.INSTANCE.DefWindowProc(hWnd, msg, wParam, lParam); - } - }; - hWnd = createWindow(); - } - - // http://osdir.com/ml/java.jna.user/2008-07/msg00049.html - - HWND createWindow() { - hInstance = Kernel32.INSTANCE.GetModuleHandle(null); - - wc = new WNDCLASSEXWrap(hInstance, WndProc, WindowsPower.class.getSimpleName()); - - HWND hwnd = User32.INSTANCE.CreateWindowEx(0, wc.getClassName(), wc.getName(), User32.WS_OVERLAPPED, 0, 0, - 0, 0, null, null, hInstance, null); - - if (hwnd == null) - throw new GetLastErrorException(); - - return hwnd; - } - - @Override - public void run() { - create(); - - synchronized (lock) { - lock.notifyAll(); - } - - MSG msg = new MSG(); - - while (User32.INSTANCE.GetMessage(msg, null, 0, 0) > 0) { - User32.INSTANCE.DispatchMessage(msg); - } - - destory(); - } - - void destory() { - if (hWnd != null) { - if (!User32.INSTANCE.DestroyWindow(hWnd)) - throw new GetLastErrorException(); - hWnd = null; - } - - if (wc != null) { - wc.close(); - wc = null; - } - } - - void close() { - User32.INSTANCE.PostQuitMessage(0); - - try { - if (!Thread.currentThread().equals(t)) - t.join(); - } catch (InterruptedException e) { - Thread.currentThread().interrupt(); - } - } - } - - MessagePump mp = new MessagePump(); - - HANDLER_ROUTINE hr = new HANDLER_ROUTINE() { - @Override - public long callback(long dwCtrlType) { - if ((dwCtrlType & HANDLER_ROUTINE.CTRL_CLOSE_EVENT) == HANDLER_ROUTINE.CTRL_CLOSE_EVENT) { - callListeners("HANDLER_ROUTINE.CTRL_CLOSE_EVENT"); - } - if ((dwCtrlType & HANDLER_ROUTINE.CTRL_LOGOFF_EVENT) == HANDLER_ROUTINE.CTRL_LOGOFF_EVENT) { - callListeners("HANDLER_ROUTINE.CTRL_LOGOFF_EVENT"); - } - if ((dwCtrlType & HANDLER_ROUTINE.CTRL_SHUTDOWN_EVENT) == HANDLER_ROUTINE.CTRL_SHUTDOWN_EVENT) { - callListeners("HANDLER_ROUTINE.CTRL_SHUTDOWN_EVENT"); - } - return 1; - } - }; - - HOOKPROC hp = new HOOKPROC() { - @SuppressWarnings("unused") - public LRESULT callback(int nCode, WPARAM wParam, CWPSSTRUCT hookProcStruct) { - switch (hookProcStruct.message) { - case WM_QUERYENDSESSION: - callListeners("WM_QUERYENDSESSION hook"); - break; - } - return new LRESULT(); - } - }; - HHOOK hHook; - JFrame f = new JFrame(); - - public WindowsPower() { - if (!Kernel32Ex.INSTANCE.SetProcessShutdownParameters(0x03FF, 0)) - throw new GetLastErrorException(); - - mp.start(); - - if (!Kernel32Ex.INSTANCE.SetConsoleCtrlHandler(hr, true)) - throw new GetLastErrorException(); - - final HWND hwnd = new HWND(); - f.pack(); - hwnd.setPointer(Native.getComponentPointer(f)); - - int wID = User32.INSTANCE.GetWindowThreadProcessId(hwnd, null); - hHook = User32.INSTANCE.SetWindowsHookEx(WH_CALLWNDPROC, hp, null, wID); - if (hHook == null) - throw new GetLastErrorException(); - } - - @Override - public void close() { - if (!User32.INSTANCE.UnhookWindowsHookEx(hHook)) - throw new GetLastErrorException(); - - f.dispose(); - f = null; - - mp.close(); - - if (!Kernel32Ex.INSTANCE.SetConsoleCtrlHandler(hr, false)) - throw new GetLastErrorException(); - } - - protected void callListeners(String source) { - - if (LOG.isDebugEnabled()) LOG.debug("call listeners from " + source); - - for (Listener l : listeners) { - l.quit(); - } - - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/CWPSSTRUCT.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/CWPSSTRUCT.java deleted file mode 100644 index 671a1e2309feb0d17801ba2fb6f2d97883d61985..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/CWPSSTRUCT.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.util.os.win.handle; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.Structure; -import com.sun.jna.platform.win32.WinDef.HWND; -import com.sun.jna.platform.win32.WinDef.LPARAM; -import com.sun.jna.platform.win32.WinDef.WPARAM; - -import java.util.Arrays; -import java.util.List; - -public class CWPSSTRUCT extends Structure { - public LPARAM lParam; - public WPARAM wParam; - public int message; - public HWND hwnd; - - @Override - protected List<?> getFieldOrder() { - return Arrays.asList("lParam", "wParam", "message", "hwnd"); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/HANDLER_ROUTINE.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/HANDLER_ROUTINE.java deleted file mode 100644 index 5995d47bbab0cb93479549721370d10fdb7a512b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/HANDLER_ROUTINE.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.duniter.elasticsearch.util.os.win.handle; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.win32.StdCallLibrary.StdCallCallback; - -public interface HANDLER_ROUTINE extends StdCallCallback { - int CTRL_CLOSE_EVENT = 2; - int CTRL_LOGOFF_EVENT = 5; - int CTRL_SHUTDOWN_EVENT = 6; - - long callback(long dwCtrlType); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/WNDPROC.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/WNDPROC.java deleted file mode 100644 index 113d77ae75be4ddc962352caf63520bc23a8601c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/handle/WNDPROC.java +++ /dev/null @@ -1,35 +0,0 @@ -package org.duniter.elasticsearch.util.os.win.handle; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.platform.win32.WinDef.HWND; -import com.sun.jna.platform.win32.WinDef.LPARAM; -import com.sun.jna.platform.win32.WinDef.LRESULT; -import com.sun.jna.platform.win32.WinDef.WPARAM; -import com.sun.jna.win32.StdCallLibrary.StdCallCallback; - -public interface WNDPROC extends StdCallCallback { - LRESULT callback(HWND hWnd, int uMsg, WPARAM uParam, LPARAM lParam); -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/libs/Kernel32Ex.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/libs/Kernel32Ex.java deleted file mode 100644 index ffcfa722e60ed49f22c3c3c2a32281922d35db57..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/libs/Kernel32Ex.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.duniter.elasticsearch.util.os.win.libs; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.Library; -import com.sun.jna.Native; -import com.sun.jna.win32.W32APIOptions; -import org.duniter.elasticsearch.util.os.win.handle.HANDLER_ROUTINE; - -public interface Kernel32Ex extends Library { - - Kernel32Ex INSTANCE = (Kernel32Ex) Native.loadLibrary("kernel32", Kernel32Ex.class, W32APIOptions.DEFAULT_OPTIONS); - - /** - * BOOL WINAPI SetProcessShutdownParameters( _In_ DWORD dwLevel, _In_ DWORD - * dwFlags ); - * <p/> - * http://msdn.microsoft.com/en-us/library/windows/desktop/ms686227(v=vs.85).aspx - */ - boolean SetProcessShutdownParameters(long dwLevel, long dwFlags); - - /** - * BOOL WINAPI SetConsoleCtrlHandler( _In_opt_ PHANDLER_ROUTINE - * HandlerRoutine, _In_ BOOL Add ); - * <p/> - * http://msdn.microsoft.com/en-us/library/windows/desktop/ms686016(v=vs.85).aspx - */ - boolean SetConsoleCtrlHandler(HANDLER_ROUTINE HandlerRoutine, boolean Add); - -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/wrap/GetLastErrorException.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/wrap/GetLastErrorException.java deleted file mode 100644 index 29f3e87ca35e4f481927fccfec70dcb5fde88e0a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/wrap/GetLastErrorException.java +++ /dev/null @@ -1,41 +0,0 @@ -package org.duniter.elasticsearch.util.os.win.wrap; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.LastErrorException; -import com.sun.jna.platform.win32.Kernel32; -import com.sun.jna.platform.win32.Kernel32Util; - -public class GetLastErrorException extends LastErrorException { - - public GetLastErrorException() { - super(Kernel32.INSTANCE.GetLastError()); - } - - @Override - public String getLocalizedMessage() { - return super.getMessage() + " : " + Kernel32Util.formatMessage(getErrorCode()); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/wrap/WNDCLASSEXWrap.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/wrap/WNDCLASSEXWrap.java deleted file mode 100644 index a50d8c1a46ba396a5f1818775fd7fd1d9aebd553..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/os/win/wrap/WNDCLASSEXWrap.java +++ /dev/null @@ -1,83 +0,0 @@ -package org.duniter.elasticsearch.util.os.win.wrap; - -/* - * #%L - * Reef DB :: UI - * $Id:$ - * $HeadURL:$ - * %% - * Copyright (C) 2014 - 2015 Ifremer - * %% - * 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.sun.jna.WString; -import com.sun.jna.platform.win32.User32; -import com.sun.jna.platform.win32.WinDef.ATOM; -import com.sun.jna.platform.win32.WinDef.HINSTANCE; -import com.sun.jna.platform.win32.WinUser; -import org.duniter.elasticsearch.util.os.win.handle.WNDPROC; - -public class WNDCLASSEXWrap { - - WString klass; - ATOM wcatom; - HINSTANCE hInstance; - - public WNDCLASSEXWrap(HINSTANCE hInstance, WNDPROC WndProc, String klass) { - this.klass = new WString(klass); - this.hInstance = hInstance; - - WinUser.WNDCLASSEX wc = new WinUser.WNDCLASSEX(); - wc.cbSize = wc.size(); - wc.style = 0; - wc.lpfnWndProc = WndProc; - wc.cbClsExtra = 0; - wc.cbWndExtra = 0; - wc.hInstance = hInstance; - wc.hIcon = null; - wc.hbrBackground = null; - wc.lpszMenuName = null; - wc.lpszClassName = new WString(klass); - - wcatom = User32.INSTANCE.RegisterClassEx(wc); - if (wcatom == null) - throw new GetLastErrorException(); - } - - public void close() { - if (wcatom != null) { - if (!User32.INSTANCE.UnregisterClass(klass, hInstance)) - throw new GetLastErrorException(); - wcatom = null; - } - } - - public WString getClassName() { - return klass; - } - - public String getName() { - return klass.toString(); - } - - protected void finalize() throws Throwable { - close(); - - super.finalize(); - } - -} 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 deleted file mode 100644 index 507fd6169aa0a88bfac49f54f0117eb58e2bf46a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/DateRenderer.java +++ /dev/null @@ -1,67 +0,0 @@ -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 deleted file mode 100644 index cd0cb61b9f40bc0a38a92d411daa633c35253073..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/STUtils.java +++ /dev/null @@ -1,25 +0,0 @@ -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 deleted file mode 100644 index 01e3ac3e5ebfcbaf053a22c9efed87f7250854c7..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/util/springtemplate/StringRenderer.java +++ /dev/null @@ -1,52 +0,0 @@ -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/java/org/duniter/elasticsearch/websocket/WebSocketChangesEndPoint.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketChangesEndPoint.java deleted file mode 100644 index 32b22b1185e9a9dbea71e62420f039444086c6e2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketChangesEndPoint.java +++ /dev/null @@ -1,147 +0,0 @@ -package org.duniter.elasticsearch.websocket; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import com.google.common.collect.Maps; -import org.apache.commons.collections4.MapUtils; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeEvents; -import org.duniter.elasticsearch.service.changes.ChangeService; -import org.duniter.elasticsearch.service.changes.ChangeSource; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; - -import javax.websocket.*; -import javax.websocket.server.ServerEndpoint; -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; -import java.util.Map; - -@ServerEndpoint(value = "/_changes") -public class WebSocketChangesEndPoint implements ChangeService.ChangeListener{ - - public static String PATH_PARAM_INDEX = "index"; - public static String PATH_PARAM_TYPE = "type"; - - public static Collection<ChangeSource> DEFAULT_SOURCES = null; - - public static class Init { - - @Inject - public Init(WebSocketServer webSocketServer, PluginSettings pluginSettings) { - webSocketServer.addEndPoint(WebSocketChangesEndPoint.class); - final String[] sourcesStr = pluginSettings.getWebSocketChangesListenSource(); - List<ChangeSource> sources = new ArrayList<>(); - for(String sourceStr : sourcesStr) { - sources.add(new ChangeSource(sourceStr)); - } - DEFAULT_SOURCES = sources; - } - } - - private final ESLogger log = Loggers.getLogger("duniter.ws.changes"); - private Session session; - private Map<String, ChangeSource> sources; - - @OnOpen - public void onOpen(Session session) { - log.debug("Connected ... " + session.getId()); - this.session = session; - this.sources = null; - ChangeService.registerListener(this); - } - - @Override - public void onChange(ChangeEvent changeEvent) { - session.getAsyncRemote().sendText(ChangeEvents.toJson(changeEvent)); - } - - @Override - public String getId() { - return session == null ? null : session.getId(); - } - - @Override - public Collection<ChangeSource> getChangeSources() { - if (MapUtils.isEmpty(sources)) return DEFAULT_SOURCES; - return sources.values(); - } - - @OnMessage - public void onMessage(String message) { - addSourceFilter(message); - } - - @OnClose - public void onClose(CloseReason reason) { - log.debug("Closing websocket: "+reason); - ChangeService.unregisterListener(this); - this.session = null; - } - - @OnError - public void onError(Throwable t) { - log.error("Error on websocket "+(session == null ? null : session.getId()), t); - } - - - /* -- internal methods -- */ - - private void addSourceFilter(String filter) { - - ChangeSource source = new ChangeSource(filter); - if (source.isEmpty()) { - log.debug("Rejecting changes filter (seems to be empty): " + filter); - return; - } - - String sourceKey = source.toString(); - if (sources == null || !sources.containsKey(sourceKey)) { - log.debug("Adding changes filter: " + filter); - if (sources == null) { - sources = Maps.newHashMap(); - } - sources.put(sourceKey, source); - ChangeService.refreshListener(this); - } - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketModule.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketModule.java deleted file mode 100644 index 7152c435f85b8197afe04e84a1e995eae9b2fbf2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketModule.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.duniter.elasticsearch.websocket; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import org.elasticsearch.common.inject.AbstractModule; - -public class WebSocketModule extends AbstractModule { - @Override - protected void configure() { - bind(WebSocketServer.class).asEagerSingleton(); - bind(WebSocketChangesEndPoint.Init.class).asEagerSingleton(); - } -} diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketServer.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketServer.java deleted file mode 100644 index 4abb50fd042aacede7ef78a80de4ab688ee1eac5..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/websocket/WebSocketServer.java +++ /dev/null @@ -1,153 +0,0 @@ -package org.duniter.elasticsearch.websocket; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.Preconditions; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.glassfish.tyrus.server.Server; - -import java.net.BindException; -import java.security.AccessController; -import java.security.PrivilegedActionException; -import java.security.PrivilegedExceptionAction; -import java.util.ArrayList; -import java.util.List; - -public class WebSocketServer { - - - public static final String WS_PATH = "/ws"; - - private final ESLogger logger; - private static final String PORT_RANGE_REGEXP = "[0-9]+-[0-9]+"; - private List<Class<?>> endPoints = new ArrayList<>(); - - @Inject - public WebSocketServer(final PluginSettings pluginSettings, ThreadPool threadPool) { - logger = Loggers.getLogger("duniter.ws", pluginSettings.getSettings(), new String[0]); - // If WS enable - if (pluginSettings.getWebSocketEnable()) { - // When node started - threadPool.scheduleOnStarted(() -> { - // startScheduling WS server - startServer(pluginSettings.getWebSocketHost(), - pluginSettings.getWebSocketPort(), - getEndPoints()); - }); - } - } - - public void addEndPoint(Class<?> endPoint) { - endPoints.add(endPoint); - } - - /* -- private medthod -- */ - - private Class[] getEndPoints() { - return endPoints.toArray(new Class<?>[endPoints.size()]); - } - - private void startServer(String host, String portOrRange, Class<?>[] endPoints) { - Preconditions.checkNotNull(host); - Preconditions.checkNotNull(portOrRange); - Preconditions.checkArgument(portOrRange.matches(PORT_RANGE_REGEXP) || portOrRange.matches("[0-9]+")); - - logger.info(String.format("Starting Websocket server... {%s:%s}", host, portOrRange)); - - String[] rangeParts = portOrRange.split("-"); - int port = Integer.parseInt(rangeParts[0]); - int endPort = rangeParts.length == 1 ? port : Integer.parseInt(rangeParts[1]); - - boolean started = false; - while (!started && port <= endPort) { - - final Server server = new Server(host, port, WS_PATH, null, endPoints) ; - try { - AccessController.doPrivileged(new PrivilegedExceptionAction<Server>() { - @Override - public Server run() throws Exception { - // Tyrus tries to load the server code using reflection. In Elasticsearch 2.x Java - // security manager is used which breaks the reflection code as it can't find the class. - // This is a workaround for that - Thread.currentThread().setContextClassLoader(getClass().getClassLoader()); - server.start(); - return server; - } - }); - started = true; - } - catch (PrivilegedActionException e) { - // port already use: retry with a new port - if (isBindException(e)) { - server.stop(); // destroy server - port++; - } - else { - throw new TechnicalException("Failed to startScheduling Websocket server", e); - } - } - - } - - if (started) { - logger.info(String.format("Websocket server started {%s:%s%s}", host, port, WS_PATH)); - } - else { - String error = String.format("Failed to startScheduling Websocket server. Could not bind address {%s:%s}", host, port); - logger.error(error); - throw new TechnicalException(error); - } - } - - /* -- protected method -- */ - - protected boolean isBindException(Throwable t) { - - if (t instanceof BindException) return true; - if (t.getCause() != null){ - return isBindException(t.getCause()); - } - return false; - } -} diff --git a/duniter4j-es-core/src/main/resources/META-INF/services/javax.websocket.ContainerProvider b/duniter4j-es-core/src/main/resources/META-INF/services/javax.websocket.ContainerProvider deleted file mode 100644 index 3b4d294e679fbd8b9c28122f36e5dc9b12639df2..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/META-INF/services/javax.websocket.ContainerProvider +++ /dev/null @@ -1 +0,0 @@ -org.glassfish.tyrus.client.ClientManager \ No newline at end of file diff --git a/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean deleted file mode 100644 index c3c319867f1d4438cb5f6f538a1ff0a494b1232f..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean +++ /dev/null @@ -1,13 +0,0 @@ -org.duniter.core.client.service.bma.BlockchainRemoteServiceImpl -org.duniter.core.client.service.bma.NetworkRemoteServiceImpl -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 -org.duniter.core.client.service.local.CurrencyServiceImpl -org.duniter.core.client.service.local.NetworkServiceImpl - 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 deleted file mode 100644 index 9a1bb01daa4734bbaf2aff4f6a3c15a5847620e3..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_en_GB.properties +++ /dev/null @@ -1,53 +0,0 @@ -duniter4j-es-core.config= -duniter4j.blockIndexerService.detectFork.invalidBlock=[%s] [%s] Detecting fork\: block \#%s -> new hash [%s] -duniter4j.blockIndexerService.detectFork.invalidBlockchain=[%s] [%s] Peer has another blockchain (no common blocks \!). Skipping block \#%s - hash [%s]. -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.indexBlocksRange.remoteParametersError= -duniter4j.blockIndexerService.indexBlocksRange.succeed= -duniter4j.blockIndexerService.indexBlocksRange.task= -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 -duniter4j.blockIndexerService.indexLastBlocks.stopped=[%s] [%s] Indexing last block - stopped -duniter4j.blockIndexerService.indexLastBlocks.succeed=[%s] [%s] All blocks indexed [%s ms] -duniter4j.blockIndexerService.indexLastBlocks.task=[%s] [%s] Indexing last blocks... -duniter4j.config.option.basedir.description= -duniter4j.config.option.data.directory.description= -duniter4j.config.option.elasticsearch.bulk.enable.description= -duniter4j.config.option.elasticsearch.bulk.size.description= -duniter4j.config.option.elasticsearch.cluster.name.description= -duniter4j.config.option.elasticsearch.embedded.enable.description= -duniter4j.config.option.elasticsearch.host.description= -duniter4j.config.option.elasticsearch.local.description= -duniter4j.config.option.elasticsearch.network.host.description= -duniter4j.config.option.elasticsearch.string.analyze.description= -duniter4j.config.option.i18n.directory.description= -duniter4j.config.option.i18n.locale.description= -duniter4j.config.option.launch.mode.description= -duniter4j.config.option.node.elasticsearch.daemon.description= -duniter4j.config.option.node.elasticsearch.http.enable.description= -duniter4j.config.option.node.elasticsearch.port.description= -duniter4j.config.option.node.host.description= -duniter4j.config.option.node.port.description= -duniter4j.config.option.plugins.directory.description= -duniter4j.config.option.tasks.queueCapacity.description= -duniter4j.config.option.temp.directory.description= -duniter4j.config.option.version.description= -duniter4j.config.parse.error= -duniter4j.es.networkService.indexPeer=[%s] Indexing peer [%s]... -duniter4j.es.networkService.indexPeers.progress=[%s] [%s] Indexing peers (%s%%)... -duniter4j.es.networkService.indexPeers.remoteParametersError=[%s] Error when calling [/blockchain/parameters]\: %s -duniter4j.es.networkService.indexPeers.succeed=[%s] [%s] All peers indexed\: found [%s] in [%s ms] -duniter4j.es.networkService.indexPeers.task=[%s] [%s] Indexing peers... -duniter4j.executor.task.waitingExecution= -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 deleted file mode 100644 index 43a57823caeaca108c0a12c4a9d9f8cb79baa43a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/i18n/duniter4j-es-core_fr_FR.properties +++ /dev/null @@ -1,55 +0,0 @@ -duniter4j-es-core.config= -duniter4j.blockIndexerService.detectFork.invalidBlock=[%s] [%s] Detecting fork\: block \#%s -> new hash [%s] -duniter4j.blockIndexerService.detectFork.invalidBlockchain=[%s] [%s] Peer has another blockchain (no common blocks \!). Skipping block \#%s - hash [%s]. -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.indexBlocksRange.invalidBlockchain=[%s] [%s] Peer has another blockchain (no common blocks \!). Skipping blocks range indexation. -duniter4j.blockIndexerService.indexBlocksRange.remoteParametersError=[%s] Error when calling [/blockchain/parameters]\: %s -duniter4j.blockIndexerService.indexBlocksRange.stopped=[%s] [%s] Indexing blocks from range - stopped -duniter4j.blockIndexerService.indexBlocksRange.succeed=[%s] [%s] Blocks [%s-%s] indexed [%s ms] -duniter4j.blockIndexerService.indexBlocksRange.task=[%s] [%s] Indexing block [%s-%s]... -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 -duniter4j.blockIndexerService.indexLastBlocks.stopped=[%s] [%s] Indexing last block - stopped -duniter4j.blockIndexerService.indexLastBlocks.succeed=[%s] [%s] All blocks indexed [%s ms] -duniter4j.blockIndexerService.indexLastBlocks.task=[%s] [%s] Indexing last blocks... -duniter4j.config.option.basedir.description= -duniter4j.config.option.data.directory.description= -duniter4j.config.option.elasticsearch.bulk.enable.description= -duniter4j.config.option.elasticsearch.bulk.size.description= -duniter4j.config.option.elasticsearch.cluster.name.description= -duniter4j.config.option.elasticsearch.embedded.enable.description= -duniter4j.config.option.elasticsearch.host.description= -duniter4j.config.option.elasticsearch.local.description= -duniter4j.config.option.elasticsearch.network.host.description= -duniter4j.config.option.elasticsearch.string.analyze.description= -duniter4j.config.option.i18n.directory.description= -duniter4j.config.option.i18n.locale.description= -duniter4j.config.option.launch.mode.description= -duniter4j.config.option.node.elasticsearch.daemon.description= -duniter4j.config.option.node.elasticsearch.http.enable.description= -duniter4j.config.option.node.elasticsearch.port.description= -duniter4j.config.option.node.host.description= -duniter4j.config.option.node.port.description= -duniter4j.config.option.plugins.directory.description= -duniter4j.config.option.tasks.queueCapacity.description= -duniter4j.config.option.temp.directory.description= -duniter4j.config.option.version.description= -duniter4j.config.parse.error= -duniter4j.es.networkService.indexPeer=[%s] Indexing peer [%s]... -duniter4j.es.networkService.indexPeers.progress=[%s] [%s] Indexing peers (%s%%)... -duniter4j.es.networkService.indexPeers.remoteParametersError=[%s] Error when calling [/blockchain/parameters]\: %s -duniter4j.es.networkService.indexPeers.succeed=[%s] [%s] All peers indexed\: found [%s] in [%s ms] -duniter4j.es.networkService.indexPeers.task=[%s] [%s] Indexing peers... -duniter4j.executor.task.waitingExecution= -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-core/src/main/resources/org/duniter/elasticsearch/templates/cesium_logo.st b/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/cesium_logo.st deleted file mode 100644 index 83cf445c1c6d63f4dfb4612f81a06253aff3769b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/cesium_logo.st +++ /dev/null @@ -1,6 +0,0 @@ -cesium_logo(url, data) ::= << -$if(data)$<img height="144" width="144" src="data:image/png;base64,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"> -$else$ -<img height="144" width="144" src="$url$/img/logo_128px.png"> -$endif$ ->> \ No newline at end of file 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 deleted file mode 100644 index 1b03f96d40bad593c53154292ffc852d5e0c0658..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/org/duniter/elasticsearch/templates/html_share.st +++ /dev/null @@ -1,76 +0,0 @@ -html_share(type, title, summary, description, image, imageHeight, imageWidth, siteName, locale, url, redirectUrl, redirectMessage) ::= << -<!DOCTYPE html> -<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(imageHeight)$ - <meta property="og:image:height" content="$imageHeight$" /> - $endif$ - $if(imageWidth)$ - <meta property="og:image:width" content="$imageWidth$" /> - $endif$ - - $if(locale)$ - <meta property="og:locale" content="$locale$" /> - $endif$ - - $if(url)$ - <meta property="og:url" content="$url$"/> - $endif$ - - $if(redirectUrl)$ - - <script type="text/javascript"> - window.location.href = "$redirectUrl$" - </script> - <!--<META HTTP-EQUIV="Refresh" CONTENT="0; URL=$redirectUrl$"> - --> - $endif$ - </head> - <body> - $if(image)$ - <p> - <img src="$image$"/> - </p> - $endif$ - - <h1>$title$</h1> - - <p>$description$</p> - - $if(redirectUrl)$ - <p> - $redirectMessage$ <a href='$redirectUrl$'>$title$</a>. - </p> - $endif$ - </body> -</html> ->> diff --git a/duniter4j-es-core/src/main/resources/plugin-security.policy b/duniter4j-es-core/src/main/resources/plugin-security.policy deleted file mode 100644 index 3cc974ff17c10b2a5fcf00c39c9feda28b665210..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/plugin-security.policy +++ /dev/null @@ -1,5 +0,0 @@ -grant codeBase "file:${es.path.home}/plugins/duniter4j-es-core/"{ - 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"; -}; \ No newline at end of file diff --git a/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml deleted file mode 100644 index e79deaaec9d05112de3919da4dc01205ab36bb5c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml +++ /dev/null @@ -1,179 +0,0 @@ -# ======================== Elasticsearch Configuration ========================= -# -# NOTE: Elasticsearch comes with reasonable defaults for most settings. -# Before you set out to tweak and tune the configuration, make sure you -# understand what are you trying to accomplish and the consequences. -# -# The primary way of configuring a node is via this file. This template lists -# the most important settings you may want to configure for a production cluster. -# -# Please see the documentation for further information on configuration options: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/setup-configuration.html> -# -# ---------------------------------- Cluster ----------------------------------- -# -# Use a descriptive name for your cluster: -# -# cluster.name: my-application -cluster.name: duniter4j-elasticsearch-TEST -# -# ------------------------------------ Node ------------------------------------ -# -# Use a descriptive name for the node: -# -# node.name: node-1 -# -# Add custom attributes to the node: -# -# node.rack: r1 -# -# ----------------------------------- Paths ------------------------------------ -# -# Path to directory where to store the data (separate multiple locations by comma): -# -# path.data: /path/to/data -# -# Path to log files: -# -# path.logs: /path/to/logs -# -# ----------------------------------- Memory ----------------------------------- -# -# Lock the memory on startup: -# -# bootstrap.mlockall: true -# -# Make sure that the `ES_HEAP_SIZE` environment variable is set to about half the memory -# available on the system and that the owner of the process is allowed to use this limit. -# -# Elasticsearch performs poorly when the system is swapping the memory. -# -# ---------------------------------- Network ----------------------------------- -# -# Set the bind address to a specific IP (IPv4 or IPv6): -# -# network.host: 192.168.233.1 -# -# Set a custom port for HTTP: -# -# http.port: 9200-9300 - -http.cors.allow-origin: "/.*/" -http.cors.enabled: true - -# Internal transport layer -# -# transport.tcp.port: 9210-9220 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-network.html> -# -# --------------------------------- Discovery ---------------------------------- -# -# Pass an initial list of hosts to perform discovery when new node is started: -# The default list of hosts is ["127.0.0.1", "[::1]"] -# -# discovery.zen.ping.unicast.hosts: ["host1", "host2"] -#discovery.zen.ping.unicast.hosts: ["127.0.0.1", ""] -# -# Prevent the "split brain" by configuring the majority of nodes (total number of nodes / 2 + 1): -# -# discovery.zen.minimum_master_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-discovery.html> -# -# ---------------------------------- Gateway ----------------------------------- -# -# Block initial recovery after a full cluster restart until N nodes are started: -# -# gateway.recover_after_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-gateway.html> -# -# ---------------------------------- Various ----------------------------------- -# -# Disable starting multiple nodes on a single system: -# -# node.max_local_storage_nodes: 1 -# -# Require explicit names when deleting indices: -# -# rest.destructive_requires_name: true - -security.manager.enabled: false - -# -# ---------------------------------- Duniter4j --------------------------------- -# -# Disbale duniter4j plugin -# -# duniter.enabled: false -# -# Reset and reload all Duniter4j data at startup - DO SET to true in production -# -# duniter.indices.reload: true -# -# Default string analyzer -# -duniter.string.analyzer: french -# -# Enabling node blockchain synchronization -# -duniter.blockchain.enable: false -# -# Duniter node to synchronizePeer -# -duniter.host: g1-test.duniter.org -duniter.port: 10900 -# -# ---------------------------------- 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 -# -# Security token prefix (default: 'duniter-') -# -# duniter.auth.token.prefix: duniter- -# -# Token validity duration, in seconds (default: 600) -# -# duniter.auth.tokenValidityDuration: 3600 # = 1hour -# -# ---------------------------------- Duniter4j P2P sync ------------------------- -# -# Should synchronizePeer data using P2P -# -duniter.p2p.enable: false - -# ---------------------------------- 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] - -# ---------------------------------- Duniter4j Websocket server ---------------------- -# -# Websocket port (usefull for listen changes) -# -duniter.ws.port: 9400-9410 diff --git a/duniter4j-es-core/src/test/es-home/config/logging.yml b/duniter4j-es-core/src/test/es-home/config/logging.yml deleted file mode 100644 index daa022f81ae00a7a29a88e76af5e8d584ac9a2a9..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/es-home/config/logging.yml +++ /dev/null @@ -1,100 +0,0 @@ -# you can override this using by setting a system property, for example -Des.logger.level=DEBUG -es.logger.level: INFO -rootLogger: ${es.logger.level}, console, file -logger: - # log rest execution errors for easier debugging - action: DEBUG - - # deprecation logging, turn to DEBUG to see them - deprecation: INFO, deprecation_log_file - - # reduce the logging for aws, too much is logged under the default INFO - com.amazonaws: WARN - # aws will try to do some sketchy JMX stuff, but its not needed. - com.amazonaws.jmx.SdkMBeanRegistrySupport: ERROR - com.amazonaws.metrics.AwsSdkMetrics: ERROR - - # Duniter4j levels - org.duniter: INFO - #org.duniter.elasticsearch: DEBUG - duniter : INFO - #duniter.network.p2p: TRACE - - security: INFO - cluster.routing.allocation: ERROR - - org.nuiton.i18n: WARN - org.nuiton.config: WARN - org.nuiton.converter: WARN - org.apache.http: WARN - org.apache.http.client: ERROR - org.glassfish.grizzly: WARN - org.glassfish.tyrus: WARN - - # gateway - #gateway: DEBUG - #index.gateway: DEBUG - - # peer shard recovery - #indices.recovery: DEBUG - - # discovery - #discovery: TRACE - - index.search.slowlog: TRACE, index_search_slow_log_file - index.indexing.slowlog: TRACE, index_indexing_slow_log_file - -additivity: - index.search.slowlog: false - index.indexing.slowlog: false - deprecation: false - -appender: - console: - type: console - layout: - type: consolePattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %.10000m%n" - - # Use the following log4j-extras RollingFileAppender to enable gzip compression of log files. - # For more information see https://logging.apache.org/log4j/extras/apidocs/org/apache/log4j/rolling/RollingFileAppender.html - #file: - #type: extrasRollingFile - #file: ${path.logs}/${cluster.name}.log - #rollingPolicy: timeBased - #rollingPolicy.FileNamePattern: ${path.logs}/${cluster.name}.log.%d{yyyy-MM-dd}.gz - #layout: - #type: pattern - #conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - deprecation_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_deprecation.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_search_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_search_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_indexing_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_indexing_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestFixtures.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestFixtures.java deleted file mode 100644 index f8d16e2251fb942a05de9b3f7ca1eec772276766..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestFixtures.java +++ /dev/null @@ -1,28 +0,0 @@ -package org.duniter.elasticsearch; - -/* - * #%L - * Duniter4j :: ElasticSearch Indexer - * %% - * 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% - */ - - -public class TestFixtures extends org.duniter.core.test.TestFixtures { - -} diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java deleted file mode 100644 index 516baccefecd3d279ea777153e85b94f4cabb2ec..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java +++ /dev/null @@ -1,91 +0,0 @@ -package org.duniter.elasticsearch; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.Lists; -import org.apache.commons.io.FileUtils; -import org.duniter.core.client.config.Configuration; -import org.duniter.core.client.config.ConfigurationOption; -import org.duniter.core.client.service.ServiceLocator; -import org.elasticsearch.bootstrap.Elasticsearch; -import org.junit.runner.Description; -import org.nuiton.config.ApplicationConfig; -import org.nuiton.i18n.I18n; -import org.nuiton.i18n.init.DefaultI18nInitializer; -import org.nuiton.i18n.init.UserI18nInitializer; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.File; -import java.io.IOException; -import java.util.List; -import java.util.Locale; - -public class TestResource extends org.duniter.core.test.TestResource { - - private static final Logger log = LoggerFactory.getLogger(TestResource.class); - - public static TestResource create() { - return new TestResource(null); - } - - public static TestResource create(String configName) { - return new TestResource(configName); - } - - private TestFixtures fixtures = new TestFixtures(); - - protected TestResource(String configName) { - super(configName); - } - - public TestFixtures getFixtures() { - return fixtures; - } - - protected void before(Description description) throws Throwable { - super.before(description); - - // Prepare ES home - File esHomeDir = getResourceDirectory("es-home"); - - System.setProperty("es.path.home", esHomeDir.getCanonicalPath()); - - FileUtils.copyDirectory(new File("src/test/es-home"), esHomeDir); - FileUtils.copyDirectory(new File("target/classes"), new File(esHomeDir, "plugins/duniter4j-es-core")); - - Elasticsearch.main(new String[]{"start"}); - } - - /** - * Return configuration files prefix (i.e. 'allegro-test') - * Could be override by external project - * - * @return the prefix to use to retrieve configuration files - */ - protected String getConfigFilesPrefix() { - return "duniter4j-es-core-test"; - } - -} diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/model/SynchroExecutionTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/model/SynchroExecutionTest.java deleted file mode 100644 index 3ef43e6789618e17149751ca0f1276795ee09e49..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/model/SynchroExecutionTest.java +++ /dev/null @@ -1,59 +0,0 @@ -package org.duniter.elasticsearch.model; - -/*- - * #%L - * Duniter4j :: ElasticSearch Core 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.bma.jackson.JacksonUtils; -import org.junit.Assert; -import org.junit.Test; - -import java.io.IOException; - - -public class SynchroExecutionTest { - - @Test - public void deserialize() { - String json = "{\n" + - " \"issuer\" : null,\n" + - " \"hash\" : null,\n" + - " \"signature\" : null,\n" + - " \"time\" : 1505836503,\n" + - " \"currency\" : \"g1\",\n" + - " \"peer\" : \"CA99448CDD90AB3772474A4CBCCC5A392F4E9AD3F9FA1C4018C6FB432BC04BA8\",\n" + - " \"result\" : {\n" + - " \"inserts\" : 2,\n" + - " \"updates\" : 0,\n" + - " \"invalidSignatures\" : 0,\n" + - " \"deletes\" : 0\n" + - " }\n" + - " }"; - - try { - SynchroExecution obj = JacksonUtils.getThreadObjectMapper().readValue(json, SynchroExecution.class); - Assert.assertNotNull(obj); - } - catch(IOException e) { - Assert.fail(e.getMessage()); - } - } -} diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java deleted file mode 100644 index 0266dfe9d2f75b69d71d4c628e8182f5dd43ff30..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java +++ /dev/null @@ -1,101 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.config.Configuration; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.bma.BlockchainRemoteService; -import org.duniter.elasticsearch.TestResource; -import org.elasticsearch.ElasticsearchException; -import org.junit.*; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class BlockchainServiceTest { - - private static final Logger log = LoggerFactory.getLogger(BlockchainServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private BlockchainService service; - private BlockchainRemoteService remoteService; - private Configuration config; - private Peer peer; - - @Before - public void setUp() throws Exception { - service = ServiceLocator.instance().getBean(BlockchainService.class); - remoteService = ServiceLocator.instance().getBlockchainRemoteService(); - config = Configuration.instance(); - peer = Peer.newBuilder().setHost(config.getNodeHost()).setPort(config.getNodePort()).build(); - - // Init the currency - CurrencyService currencyService = ServiceLocator.instance().getBean(CurrencyService.class); - currencyService.createIndexIfNotExists() - .indexCurrencyFromPeer(peer); - - while(!service.isReady()) { - Thread.sleep(2000); // 2 sec - } - } - - @Test - // Ignoring (too long !) - @Ignore - public void indexLastBlocks() { - service.indexLastBlocks(peer); - } - - @Test - public void indexBlock() throws Exception { - BlockchainBlock current = remoteService.getCurrentBlock(peer); - service.indexCurrentBlock(current, true/*wait*/); - - try { - String blockStr = JacksonUtils.getThreadObjectMapper().writeValueAsString(current); - - service.indexBlockFromJson(peer, blockStr, true/*is current*/, false/*detected fork*/, true/*wait*/); - } - catch(Exception e) { - Assert.fail(e.getMessage()); - } - - // Try to get the indexed block - FIXME: delay is sometime too short - Thread.sleep(2000); - try { - BlockchainBlock retrievedBlock = service.getBlockById(current.getCurrency(), current.getNumber()); - Assert.assertNotNull(retrievedBlock); - } - catch(ElasticsearchException e) { - // Allow exception here, because sometime TU failed (if sleep time is too short) - } - - } - - /* -- internal methods */ - -} diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/CurrencyServiceTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/CurrencyServiceTest.java deleted file mode 100644 index b8555ca74a65f06493da5eb6571d4e8201e3ed50..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/CurrencyServiceTest.java +++ /dev/null @@ -1,81 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.config.Configuration; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.bma.BlockchainRemoteService; -import org.duniter.elasticsearch.TestResource; -import org.junit.*; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.util.List; - -@Ignore -public class CurrencyServiceTest { - - private static final Logger log = LoggerFactory.getLogger(CurrencyServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private BlockchainRemoteService blockchainRemoteService; - private CurrencyService service; - private Configuration config; - private Peer peer; - - @Before - public void setUp() throws Exception { - service = ServiceLocator.instance().getBean(CurrencyService.class); - blockchainRemoteService = ServiceLocator.instance().getBlockchainRemoteService(); - config = Configuration.instance(); - peer = createTestPeer(); - } - - @Test - public void createIndexIfNotExists() throws Exception { - - // drop and recreate index - service.deleteIndex().createIndexIfNotExists(); - } - - @Test - public void indexCurrencyFromPeer() throws Exception { - service.createIndexIfNotExists() - .indexCurrencyFromPeer(peer); - } - - /* -- internal methods */ - - protected Peer createTestPeer() { - Peer peer = new Peer( - Configuration.instance().getNodeHost(), - Configuration.instance().getNodePort()); - - return peer; - } - -} diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/DocStatServiceTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/DocStatServiceTest.java deleted file mode 100644 index 7b70c5070b4c10ee86d26591e40c30c91f465893..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/DocStatServiceTest.java +++ /dev/null @@ -1,80 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.config.Configuration; -import org.duniter.core.client.model.local.Peer; -import org.duniter.elasticsearch.TestResource; -import org.junit.Before; -import org.junit.ClassRule; -import org.junit.Ignore; -import org.junit.Test; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -// Ignore (it's work but too long) -@Ignore -public class DocStatServiceTest { - - private static final Logger log = LoggerFactory.getLogger(DocStatServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private CurrencyService currencyService; - private DocStatService service; - private Configuration config; - private Peer peer; - - @Before - public void setUp() throws Exception { - currencyService = ServiceLocator.instance().getBean(CurrencyService.class); - service = ServiceLocator.instance().getBean(DocStatService.class); - config = Configuration.instance(); - peer = new Peer.Builder() - .setHost(config.getNodeHost()) - .setPort(config.getNodePort()).build(); - - // Waiting services started - while(!service.isReady() || !currencyService.isReady()) { - Thread.sleep(1000); - } - - // Init the currency - currencyService.createIndexIfNotExists() - .indexCurrencyFromPeer(peer); - - Thread.sleep(5000); - } - - @Test - public void computeStats() throws Exception { - - // Add new stats def - service.registerIndex(CurrencyService.INDEX, CurrencyService.RECORD_TYPE); - - service.computeStats(); - - } -} diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/PeerServiceTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/PeerServiceTest.java deleted file mode 100644 index d78efbe8567704bb03dc41001c4e47aaff7dbc31..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/PeerServiceTest.java +++ /dev/null @@ -1,110 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.ImmutableList; -import org.duniter.core.client.config.Configuration; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.service.bma.NetworkRemoteService; -import org.duniter.elasticsearch.TestResource; -import org.junit.Assert; -import org.junit.Before; -import org.junit.ClassRule; -import org.junit.Test; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -public class PeerServiceTest { - - private static final Logger log = LoggerFactory.getLogger(PeerServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private CurrencyService currencyService; - private PeerService service; - private org.duniter.core.client.service.local.PeerService localService; - private NetworkRemoteService remoteService; - private Configuration config; - private Peer peer; - - @Before - public void setUp() throws Exception { - currencyService = ServiceLocator.instance().getBean(CurrencyService.class); - service = ServiceLocator.instance().getBean(PeerService.class); - remoteService = ServiceLocator.instance().getNetworkRemoteService(); - localService = ServiceLocator.instance().getPeerService(); - config = Configuration.instance(); - peer = new Peer.Builder() - .setHost(config.getNodeHost()) - .setPort(config.getNodePort()).build(); - - // Waiting services started - while(!service.isReady() || !currencyService.isReady()) { - Thread.sleep(1000); - } - - // Init the currency - currencyService.createIndexIfNotExists() - .indexCurrencyFromPeer(peer); - - Thread.sleep(5000); - } - - @Test - public void savePeers() throws Exception { - - // First Peer - Peer peer1 = new Peer.Builder() - .setHost(config.getNodeHost()) - .setPort(config.getNodePort()) - .setPubkey(resource.getFixtures().getUserPublicKey()) - .setCurrency(resource.getFixtures().getCurrency()) - .build(); - peer1.getStats().setLastUpTime(120000L); - - // Second peer - Peer peer2 = new Peer.Builder() - .setHost(config.getNodeHost()) - .setPort(peer1.getPort() + 1) - .setPubkey(resource.getFixtures().getUserPublicKey()) - .setCurrency(resource.getFixtures().getCurrency()) - .build(); - peer2.getStats().setLastUpTime(peer1.getStats().getLastUpTime() - 150); // Set UP just before the peer 1 - - // Save peers - localService.save(peer1.getCurrency(), ImmutableList.of(peer1, peer2), false); - - // Wait propagation - Thread.sleep(2000); - - // Try to read - Long maxLastUpTime = service.getMaxLastUpTime(peer1.getCurrency()); - // Allow null value here, because sometime TU failed (if sleep time is too short) - if (maxLastUpTime != null) { - Assert.assertEquals(peer1.getStats().getLastUpTime().longValue(), maxLastUpTime.longValue()); - } - - } -} diff --git a/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean deleted file mode 100644 index 6613b03e0d6f746558b3ac98f065d36d74cf3411..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean +++ /dev/null @@ -1,13 +0,0 @@ -org.duniter.core.client.service.bma.BlockchainRemoteServiceImpl -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 -org.duniter.core.client.service.local.CurrencyServiceImpl -org.duniter.elasticsearch.dao.impl.CurrencyDaoImpl -org.duniter.elasticsearch.dao.impl.PeerDaoImpl -org.duniter.elasticsearch.dao.impl.BlockDaoImpl diff --git a/duniter4j-es-core/src/test/resources/curl_test.sh b/duniter4j-es-core/src/test/resources/curl_test.sh deleted file mode 100755 index 4f62377a7b20ee747e1a99f1c6ab627e9caa8b35..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/curl_test.sh +++ /dev/null @@ -1,16 +0,0 @@ -#!/bin/sh - -curl -XPOST "http://data.duniter.fr/market/comment/_search?pretty" -d' -{ - "query": { - "bool":{ - "filter": [ - {"term":{ - "record":"AVbieTIAup9uzWgKipsC" - } - } - ] - } - } -}' - diff --git a/duniter4j-es-core/src/test/resources/duniter4j-es-core-test.properties b/duniter4j-es-core/src/test/resources/duniter4j-es-core-test.properties deleted file mode 100644 index 608cea0ef49f2330cc7a26f58af9bacd76da0e1a..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/duniter4j-es-core-test.properties +++ /dev/null @@ -1 +0,0 @@ -#Empty test file (need for inherited TestResource). See files 'src/test/es-home/config' \ No newline at end of file diff --git a/duniter4j-es-core/src/test/resources/log4j.properties b/duniter4j-es-core/src/test/resources/log4j.properties deleted file mode 100644 index d6ce04a626960afe06757188d5e76095fc2c7868..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/log4j.properties +++ /dev/null @@ -1,20 +0,0 @@ -### -# Global logging configuration -log4j.rootLogger=ERROR, stdout - -# Console output -log4j.appender.stdout=org.apache.log4j.ConsoleAppender -log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m%n - -# duniter4j levels -log4j.logger.org.duniter=INFO -#log4j.logger.org.duniter=DEBUG -log4j.logger.org.duniter.core=WARN -#log4j.logger.org.duniter.elasticsearch=DEBUG - -# Other frameworks levels -log4j.logger.org.elasticsearch=INFO -log4j.logger.org.apache.http=WARN -log4j.logger.org.glassfish.grizzly=WARN -log4j.logger.org.glassfish.tyrus=WARN diff --git a/duniter4j-es-core/src/test/resources/registry-test-records.json b/duniter4j-es-core/src/test/resources/registry-test-records.json deleted file mode 100644 index c9c11c01be29242dca860f9e0e692539c101e07c..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/registry-test-records.json +++ /dev/null @@ -1,23 +0,0 @@ -/* - * #%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% - */ -{"index": {"_id": "AVOt3cmVo7-63byx1Jow"}} -{"title": "Benoit Lavenier (kimamila)", "description": "Pasionné de plein de trucs...", "category": "particulier", "location":"Martigné-sur-Mayenne", "pictures": [{"src": "data:image/jpeg;base64,/9j/4AAQSkZJRgABAQEASABIAAD/2wBDAAMCAgMCAgMDAwMEAwMEBQgFBQQEBQoHBwYIDAoMDAsKCwsNDhIQDQ4RDgsLEBYQERMUFRUVDA8XGBYUGBIUFRT/2wBDAQMEBAUEBQkFBQkUDQsNFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBQUFBT/wgARCAGLAYsDAREAAhEBAxEB/8QAHAAAAAcBAQAAAAAAAAAAAAAAAAECAwQFBgcI/8QAGgEAAwEBAQEAAAAAAAAAAAAAAQIDAAQFBv/aAAwDAQACEAMQAAAByfjdgqFYKXLOJSdMbA2IVD2DE8D2IFS4jjcKGOqHsEJnErHkFAbKbYIwxVgbKQxKTcEMrAqE1AUlibKShTavcV5UhlUCsRjI8jvUAWDr4MFpiUkVFceBsFYGuBIVg2DIoA0xVAYmwkFa+8oCNILWjKWxI654bFmUUBKVylxUynCQRgFBDKbBdEoK2ykAaAw5EyfL7FnGAZKiDwLA9g2DBRAGUCGBBiIUCZWG65aobcMMOk25IDJeTNyHXSZ4pD42dZKkTqbYsAjAFLqsqQwJAAmTIJ8w61LqrBOImyqCT5ncpAMFKAzKOPAgVVUypEKxJSbgYNMYrijdcS+jBnsuhyagy1Vee4oLxlpMXUMcrXrVkNvgKSdsxKpAqwQcYxDGSRB4GcHyQtIVI5SMDlVD/i9wYm4PKS5bhYBTYUB1UYKBVlb2q7HE3EJ51U2SjKwnMNgUS89jXnmWFmy8+D9NWLuaCrvus9kiQvS83YwpVggkLjwBxHDZQBkJJp2RrBTAsylL/ldw2VQEAsZbqEJNlMoYHi0wj0nnKDE11apcWiVL6lx1nGc9lfXaDq5uhUjDnXkKdFxaUsGUr3ISXhJRno2aQhwSEhjYA4iDBPY2UZaotGKECCy8JHk9wZVEAAMy8pgppgVUAHVgHKWOFoK4g9nwWipEvrrVRY4WzzvaRess5K4haxkosPpVXRU59M/O4DDToro0S2BAUmoBxEBipsNlKK5lriFbKUAq75PpHgHDjYMqioICsTBWw2iO3PqDL1SNgsZDZBygXctgC4G1jz1NZzkpcLpONYCoCdhrujlerCCDdIcJDsrkoa5LBWABSQATcHsMIYFacpcGRdcrxvRUQTY0yiocG4AIKjYER2bmt0zjZgqnYbDFa0sJ0nzrcI+4vDs3RydDaVwju7RCVYUpGCeMYrOAmWjj5dWZ5ughiGDAyCAIlWw2MiLlrMTcGqlnc83tUAQxghlOuM4tgcrAMtNtzWzVhEV5t4OAzI9Gh5uqxSk4rNpPa9XJ1jo5Nsw0EqmCMRsMKt0wVOY6TzubLc3TClQgUjGcCAyjZWAOBZsCowSFUVWcnzu8KTxUwLFRUyAMoskKgjO23MrBnK0wZbS400vL3XKUdAmlL+8NZeG6tz6Wm1KNZTIxGw2G1c6cpvy1qHMwu3OjYyWAJPAsAwUQMTXApTbJYKwAZvzPQUoM48A6mcNlIQSdEJhy+uytMyUTsvNaw6NLz3mwd/FVBqL8nUerj0FUvnXRBrGdDBGI2Gwwy9ZcypLOStEm6FI2I4EDKBjcBQohZFKrJcggtk+R2hCdMKIYy9k4LOBwZalxx6rNAtMqTnle5573s6zYu+CeS36YdR7uHUtK0fas63nR1HGI2Gw2SV5LaGDjWNNkkoXGcGUjgVPYxjwdZac5sEzkHNeV3K2VPFYKZTBBB7KOQNzywwtcQVLA1Z4UtIWvI1tEKizoR7o236+Ha1hLZdK41p1zzVMMNhsNhhSvPjGFFN0bIxDYKA6lgrYAG4dUVdMwCZKisbyPSJVUwUQeBAmcplNjGYcWssDEHJwPGROlpC+g57TwynAqrjy2XRzdM6eN5zZMb156/kq8lRsNhsNiI4PeWciyBkjJxPYspuDGIFTq4BAcRthgrPD8n0DABRTBWwIAJ4qwz105HbMggEbOgz0tq+a9nGiDkUVGD95bXs4et9XBJebApcUXY8l5UruK42GwxG3Jry5+uZTEuAIbFgRwOXkVirLEdYqsDiURPM7yOU2UFJiYwYHsDsfZOZOyRiGPZ5KaLkvqpWSztKWWUyqOmVj18e76eWFUV6F1DrWXoSnapS6k6hklUnZdl4mQlC3iAE4hsWClxsDYKIb2hgpohYQfH7wSGBgkpVlLEUVRGLvubOSQJwcLPzrqOW+hS7EiyMTzb6Isd/JfV5+pmd5RcVqZSVourttLe03ZuY07LzSgZpHpqT46K0k6BcgktgygE9lZQQo4mWGckBaap8vvI4mMQFWydqRs6Z6IjE3PPHCVywysXVN9z9FilYGnCvKP0yZvHX9nDvI1tGlY2jas+aVrApHakQX7tyGrZaVX1aLdqKrojy4iom4hdCsQDWfJOdWZSCHApkRGVIIU0/k+gyXhpTDu84yeKt0jS3TpkmzC0xD2gMiGm2VWVkAzMzZy2Etkbslp1cvY+e22M5dJbVH5dfY5glKNc3T6fkeX93NqRI6ptPNvVdHNz/tjz8KvzPTRKjTKib8tevQm52WS7dZuEVAlGRUUHgevDDREtlriJeMvKl0sbR106Zs+hEnSHjS05K5oR2XT8XTqOe7dFVZLqnNqOzlzvSmt20SrZ9KSBTKlcTGlSXupdXqiEMV3cehkaDs5XgbKc+bXGBnSf5noU/PbIdSbUT55Ot4or+ufS6c05dFBbVjC5TwvajqWGbB9Chpv0SHVNkqTx25jdb3PSuZL9ZRlXHX59Rz9OhRwwQ2lXj1Hs8zoN+bSZLRdTV2Ea2GWmeStNx9qh09ory9sHGCtD188O/NTUnQyrguLtruatRG9PTRrwz5fTiSOiXQXjNQMKWsBjj/A9qOGq6jB9iWs1UmQ20deeZvQy567OJgVSw552MNm7zlK+hUr2cbHWe+6eHqfTxX7ifndDWEzwSteSQ6ShaS9d5SPo88MpREdKxl4n0SiRaDGtbz0rNVtxzpg3Vbek0PLprSvAqAUbFlxHz/utIMX0tADTdzyqKKZqkthuvED0rOWrLcsTmOu42i9kq+iaKbz1BqzTGTSex6OTd3l1sJNUtEeVLXwPP0SJ3s83Rujj9CDllLi2jNPz90JCg8CDVU6R2NJjzijWBE/SR1Ruenn6uFKTJQllwnz3uVVBhOrKVpOjOrFNQZG8Vs5L1o6VoKSjINjzmTp5Oq3ajSbSAWcyDnaDd9PL27o5NZE8uu3nCfTC5uq9hRdUvujl7jfl38FGKcPPlp1UtWyrXKW32Cd88tZSrLrz2PRF2ydYaLk2bQmq8/8D3aWgxdweD7SlVRJ1w89o8nIehkJ+hHKVqmNIbGUsh0IW2oVr2TBwnZNm0d+bsnTxNluGyvnFeGvRZssMbYZNu8+77mslySPP9Z0cHqpNCV66dcazVL5TjS9HLPbnDHo9ueSmYDkg594HuZ+wydi9hIEFdETZrJhuHjZpWr5/Qyk+lctVlrGcs71c7e1/J9MheYLcJzSemV7WGdLU7ZgPCFTZdw0ukLPqyizIsVmNvPtlz8DUTrGRq6dMxjnqomy6isVtNq8+n15rWJZDJU878H28f0LSUabIqpzOdcXlwJ6BaFtF5MqVCdWTl3xCzSThtzw2Frz10opKqkjITKdlrbJCCwGKZus7rTcveqQrmbWyadsMYpXzy6U0qV06w4NCV8hdam6WOFvkn2jAuOj15bmBj5iXc38H28H1JXtrCZa6OZbK64tSNtaFnG0yTKxwB6G+XrbUVTSrH02FdDJpd5zFLlBWdM6xc60ksJApaSPS2l1q8Mu21sBp6aVjhqLyCeqEpCm8eTQg9JRqTaONdW57t5WVo63p5JwMdakNzL572sXcwy1rFZNuZfQrpE+s9L0806VpaO5MxmODh1q5751liES889TYDTF06yUPUiRN10aZpjVumG3om6aemadnIXElnpuNUGFi1TOkWTsro50R2yq0j0Ww6ebV9HNb5NDbnlgx0dO3M/mveytXhh125p7y1VeZzoVjLcUjNnWUlHFK8c/qZzj7Mo2ZYWCPIwsCDLWbpCqiAjjlq9FNTqFp9RadrIX+WNjjZJ0KUeJOM1N6+TsJmQ6HVFFxa2j9XPqunlsNNcba+3NMAbDo25f897ubdqequ9XHY1nrJ6KRY1jZuktKyputGWCzjhObry+pV2nZwq8WmsHytg6XFJ3rRyQuidrKq7fqj0eu6jxwcGGGcfSJjhB1POkNTHbIGDTdouJvksba8NfOcWg1FeewnhsnHj3y/0mVrobC2vy3/RzIV6nbXW5ZraVGklLO7EuQNgodlULZzoi/MOrSQTcDPlbcpNadIavimnrzbqvLvaJtwZkqjMkriax5reLOUyFMizlsiiAQMAuRsjBJDbFksU9y35r6TAdYNle7OPUzSQGiq2srz2FJyUpISkBGphs7qZzh9C5jSv6I09Yu554MpxJAsUVTZ458UsaJuDDaUh01hMDuq8MDi3Rz5m8GcjRzRLWyVZrZGCSEnG+C4sTKK2NDWcfXTVEMGuVqsPIw0bJYPNldGLZgnLrVpWdmbbj7dPy9SqJA6I11ISg0uTvVz4zyafp37zOu053V6x2w1gjjDBZeJdUIzyJQRUsTwMMATYGMTKZyyqyi1L1FfYeavP9VgFrM2cQwXE+UMDgMlcQxtnlzsOjW8noSCrumm862ySZ6SCRxjTXF8Y3bx2ZHVKz2CVWDHw4hSGY6IICpxPKTqeJ4mCMp4GxMleVbzfcS6LPefjfw/fQ2LYLg2GxMAuMZw4hmwS2cGdDT+bsup2dGkUjMeZOkHaSrsLn3Mlk1+jt3TpDS0uNwGgtLh3VGPSYQGQCAQZymwOPKvAbKbOYPsslw6yKO8WeJ75bBsSFTZYBrkkDMCG8SAJSo6SAgvbc3TYLZ9VlJrh5O3nX0EdM5Zn9K7C6uY3BW9KaFdA6ZYfu5SZTCmcAVMBsrAtjyqbGcezoy3RbAxjO8U+H76WKhncrJJLi2eKs504AFzBJwxl7MjTVaxSr6vN53s8bKiS6SdeaaTKosbzuVVMKWiNprc8nq5Y1Zg4wASZBjGQFxtjKnseB0C1w2N1PYY+KPC+gDhwgwI6sDiGewbzJUDF7BskHTFzGz21qHfDOTpahrJFnslv0x0NI9XpLocqSVPFPQ5uZ2jvrcVcwr50t5VtRj2NwaqGw2MYbDBRwZXDkqVsptkbeKfD98HPuoXRkcbEc4ykpbUjF1gnYAzBmsDfWeMtC5nnrS3Vbdp6Cydgry9W5rvCpZfPvq8HLunn6FbksnjhTXl8O3oc59AkbDAYHsWIwTsghIMh1lOpKVFVMAx8U+J7zQL+y8IoI2IFzYYtkHsoYhhjKXE+QRMwtgXaa0DXyi6YXSjuDT6BGwxQU8iezxZ4T2V+HcX5k5ePv25wPeLIQtcA6SIloyUKVJMJT69qkoyIg9lPiw8Z+H9BC2cBfKxSUoQSvAEI2LFa4tjJkplYNVVZW8Im0e3U3zLe5LvbsGHQ+S4x55183kLuF7pW/T53Q7c+mRKptgHeYrcpl0zFERmno93w9FryVNTq+3nuujnnvIEGcYwbeLPC99RzJMh5t7R5ueLhAwbUkSrYxiOkkOYQyVjW3RLa3SVzU1Cro8vRqy6lyXfTop6y8kdcObB9rWMTs4Ou05LOwzS7DvXWBayVuay6DustknqzvH0anzejV2lc9HNPtNzIGCsQu8TeN9A/g1tIrMhok6K2cIPBlWBxgrXJ2s7zj4QpWWc9RNx182409Bz03xTrUjZw7Y7L5h9Dk47K1QF3Now/Q5OpNxz7TmuuA1I+Oki3EuX0WJ0tL822ry58dFZGnUIvsNOzrFRRTZex7eJfD+gc2TTPtN7JAlYwXioxYDGQoYAii2dIVUOhSkmy9n7p1Po5Opc5t1PWsmg5+vhXfDz1mop1hsuw6JyOvg3r8d7SWgdMmTR4zxblHH3Zrl65F+foXTwzc3M06tE0euQGrAdKHitwMPE/ifQjB45zolddEM9ydTcaSGRe0VGc6JaHojVTo+q1sqszo+wTioZ9hZ9XN6DmuzydOmuFenmarUs6VIdh56Po5ejdvn1jrvRO+KtuuKJAfGcfZg+bugh73s4+0Ny8Ul3WN4aanN1blpbIVELYE28c+b7FfydCzrTs4+/8As+Lx7k9HGeV6Umk5lErOe8lo9V7ODnserPcvYA6gJRVsF06dWfQ+7h7Py1nw2NqOWvaqWsWZqmW2dNR3ed1Lo4clfT5nSNPfc4491aqnVhXzC3yHJ2xVbq3T5cOzZ002RlcIOmc7rKhcCfNEuvF8vaxC0y0PTPo+Kjoj5k8r39J083Vu3z+DeR7TXNYHJBbUnse09kklIheZefaO7yewhPP/ACerTQtHma9jBoGnVNpbBueNaHWe3ioMNTMdHSfIels7KtZOmgYYOPVl+Hvm0hv+rz830NdIYRHcIraqwmTdfNUO92p5/wCX3uZPR3p+RW+jLhPk+tuqcfpT0PJ8j8vr0vl9yQ4DsoQhMLP2fokQtamO79PzO87k5VXuw/menCka7olBUu9iWZlmOLrldXN6G9HyKArKB6nzDnXWMhC0XUvnTLzfK8fbSSt0Ho4z6udeviUbrz83VOZjITh//8QALhAAAQQBAgUDBQADAQEBAAAAAgABAwQRBRIGEBMhMRQgIhUjMDJBJDNCNAcl/9oACAEBAAEFAvK/rJmWO7e7PtZY5fxZ5P7sc8LCxzwsLC8LHLwrLdv73TJ1hM3Jm54WOWOQ8m7+zHPPInZmr1ysjPJ0UNiQ1DLukaqRAIGzEzi7LCxzZueOz8v7zsN8MLHJ+WF4TP7tvsx7p7DQg+tluk1s3Uc0tqT6jHQoVB68cVLfG2i7ap4alpUASy364KOwMckl2u7D81454TN+CVshywsLD8m557+UzLHJuTJ+XhFKLKeyMTWNYARsXpJ3bu2MKoLqMeo1WNmF7AtYns9ISmE5p6cu7ryQQ068coDpgzvpOliA6tTigL8b+P6sJvc6Hx7PHJmyj+Kss7tJqDwyS6g8rMXd0Cj7PC4dJpflps2+BxdzuyDLQfUT3x8ReohCYZCvnBVKKwzVq+uFXsyTb5pR2msfiJvl454/C6wsIywnN2VzVOi9rUJJy3ZflliYWW3KGdgRfcWlWmrTzbKsuovDJCRFCcFvoxyau8hDd6g17Bk0kznDEe2Jm3D7sezCn/fHJm77U345FqErCpj7uh8kS3JiwmkwglywxDK1ak5uwBE0VpwGvbFXKrvYw+I6+8oafarSkNw0Z45LFB3I4CEdqwmXlYWEyxzZWm7rKZbX5fxvwz9xv/tJhOgfCfumWEzKAsKNtr1rLsEUzyww6PelUXDcwM/CZ7H4WMUHDxgo6zi9Fo3Ca2G57vUQV/VK7F0rCx+Gy3bHfCwsOsLHsbtz8c53wF+R9+3KLznmLOghclHUUNRaZpoyPpOlwiFepGTNWjZPELi9CHL0wJ9Q0KC2J6fFBPNpYQTR6UBNDC0Za5E0dn2OvCbuscsKdvjyx7Mfg8K3J8bR7iT8mUMDm8VNmQQsyGJRjhab2ehLhUrCZ8+2/QC9FOEtaWUhrtYvErMpTyO3s/ix7DHIPyblhP7/AAvKMVq5tDXJ8rci5Rhl60bJm5AqlPqqnTcCrRuBQ5dVD7e27TC5C8JadNafcn7vy8/gdss/ZY/E3s1u5153fmygHKhHs3bkLrS59p14N4wlhV+yg7e/W9PazA1QpwlFwf2N7y84xyxyxzbkyzz1Gw1WtKW835iyr+QJC3fOOVPaJ0bXxilYhgn+UUooSz7nbK1LT5IrUzExfgxzkbB49/j2eOXEVrcXsFROokyduTEtJsPtGR01jBQ2nxBNlN7tUg6ta3iRPzf3sp2+bc8e3HsnPpR2petN5T8mQsogd1DC7M3bl5UeFpTEL1a72Yz0l819PNnrCYFG/wAfa/dazG9e/wCfZ49zKw3ywsLHJvwa3L06JOsd+cMe568GFjk6yovOnWwZ6FoHVq+xPDqTb6+otIcEzGzmze7iyHbbdOsLCf3Mp2WObexuWU3LiMttR+WOQqiGUzdnTui5dR1UmYTi1V4o/qVh3l1WfcGt2mahxTOypay9lXL3pBHuK3My6gu/END1lMmw/wCKVvizezHtd+fEv+l+bMgbvp/InTknJGXbuLafUlnelXeAHCSWO9QmBHBKy3ED6Hf2zS6iV/VB/XU5zhhmvzMVTUpYZYNTisR6pGw2uePZ/ORt8U/sZSXIwfq5XVwjuGRxamhPc3Ex/F03MSw9acQRajGzHqgp9Td0V6V16o3VGcpBgt9KKKxMSGIt+pxejPa2WqxsJ048adxB0bVTiurZaxrL61OejuJVNFoysGi1Y1rFPNabTZXGL7hv25kW1fXHKaCXrRJk/h25Z5PIpJuxEXX9S4JpnlZyURlYtQttDiCuVgSoSgxRuCyt3dRiRr0crp6UiDTZpH+jW0WlWo1WB4G0nptGBafiKSs6+jU9Rr65on0aU5ykKW30oOGKXqbcWmQVg0i41Pie/rVkVXvWwCA9waqP+L1nBjmLqZW5b2dF3H/Xdpy7a1vUWrhVkeSFEv4iNO6JW8NPYfaUZdIOuO3ToGabfsaey8h9XvJCMqOkyOq4MIrS4ltTgzpvg9SVykthDLWo6FDZuNQnjVOCSJWtS+1puvtRj4g1l9UN1ciKUdJtvVs0LbXK1jht/rtvQ47KkovpsNDVD6mr3unHYmlIIpGKWQ8KSR0VqSrbCffHaLGoxm7qdurPD2BkXbm78jJXJ82yJyktn2AXkj0zLDek2x9Ttj4lM8M2wSEq25jDZJT7JvCJlVDMlGpG603SIqjtTQwvjUtIG4FjhWYVeqBTI7EIo5Xd4i2rhLWum7d2WoC1hDpnTV2CUysxNXhN8qRSK7BupU7RNUN+rbrA+61FsuxfqKNu/J06uT9GHaTlAzk8gb1VyB6fL97Uu4sSAshJA8iqzNtD5K9Dies+GB8sywo+xaPeaB6epgTR6lC4xzyzC0VgiGPauN5B9e7ofO3K0GL7lV8wL049bWouppxG6J0SlR91cwFNrbgEZPuexI69cch6fZaeBvBc3WVem68mFCRMm8Y+MJuM0w9WGUdhf647Fp2HTsvZiLBXhyojVd8iIrHIJXFULYqrqbRNVdigUhtGHEF31F2WRRM7oOy4em2nWx0eVkd8EjYclI6kdAOX1uxsr/8AUXZA+0ii+5pNoq07Oi5ZTurkrwwbnXUNNMTILDopSdq4FJJXbIanW6akF5ICFAbxFGXUayO+IZO8FjCA8snTsojwWnH1H021uZcW68MMNqXeqY9U3DsIuqMz15NF10Jm5F4tdpyUr8iLatYPdO6BMW5gfcNcP8kPD+zUe9dn7IMOsszbhVDDqFsNah68Tf45S1hNFp7u1eUqsrlmM8DIMiqy7mF8thOmWmz7D0q2OzV+KWhiu3SsHK25qU3SK3eMBi1WwqMhWh0iuQ2Kd2K0Kdag222Sk7u7qY1qEbkTiwKPBKqAgHVFmhMetH4fxydame2PZldBnQ1wR13FwYVUNurB3YVdp9QSgIX8DKDM8Mm+CbuQttVd8KN+3LCjPYX1E9li28iM+zn2PDqKA5i0vha3cWk8PE12TRIChqUY6QctSfddmfsTolKtQLaAizsO0SEmGvHdF2d8FpzvLUfxyJaoWTZjNV8u4xuchAUaDbOoYnCzC/YV2dXK/wAXwSkDKrdjsx7ZHdQkon7bu7P25T295HNtTzLqO6bJvoNRoH0fZDTGcW1OOZpOdqZq9ec98sr5RJ1L3bUx+UQMREcOyp/p3bhjp7C0oHjrv45G/bUHd7HWfFLJHK5xytK7I5d6phvkiZCh8YWoVio2Cfc0bfK4Xz8qB+8cmB3IX7K5O4iT4UVd5zkp9Mgqm7VqG6LTItrabVJVWI9Qpt0lCWWXFNvpVjJG6dEi7q9Fvib9T81ZgaKs24po9iqtti5yeLYb7D1mVCP5PGzyehEl9PGNqUeGibCBDysQDYAYCqy4w9kvuP4Bu0TreoyXlrPeUY8o7GxfUhd2vsqlkWVS5EzaTeMpG0+aAo5JQOvYCUPK4msdXUJHUj8nRMpB3NjCLCrHHjT9pS2/3h7Ci8qRWRxObuyr2DB4pD3VY97WuwVx+IIULpuWqV9wk6s9j3ZYO7Qv8lvw8RbmuR4eKN3co8O9HqqtGMRaRPQVTR9K1MYuFIICPT5CEqNiMNOlmr6h11qp77kr9i5Oyx2YFa+3aryDBNFqcMiq1d0tyTCr/KJPyNW/9shqKZmUJbnjJwCY95RNhgQpvCZSixhLG8ZXQ2uzoC7CS6mUHyeL4PHV9UMFEK9ect0ooI8qpp8Ey0Op0JAzt5WqO69cbZBYPqSSImWE62pgWo1ZJLkml2WWn6ZYAgp2NtrQ7MrVKcscLViZPUd16J0avH9xxElXgEmrRACnn7wSOVgGQpkPLKJ2FrVobM04dSNxw49mYsoVG+GCT5hO4L1xsG1yIG76ZHvKi7HNHAIzxuXN2ytdsuiqC69DG69DAvSQL08DJgiZfbZbwZeoFPaFesFeuBfUAZPqIY+pivqYqT9Z3KSaWORo4XlFq0kkovSklUOmzxlBVk2tWkQ1zTV3UtqCBFr9ES1nXo5YqMm5lYhyOe7N3ZsCJofIEs7k5YUQbnrSFEVK8LPWtbDru7oN3K1O0EWsaiz2X1F19QdPqBp75J7puntmntmvUk6eUlvJ1uJfLHyWHTAS2OrOsafCn4jpij4oZScTTun4iuZj4pug9fjXs3GkCfjWNHxu6t8TXbTPK5LKj/bTpMSp/k0lbenZwdi7N4d1vwIOox7RRdOJvsyR14ifS9MCSOrVaq3LX9QYYpsyH0l01010101010101sXTXTWxdNdJNCugnfCeROTrPsys+0H+QH05Y5t4saYkcbSI4XZRkm7uTYdjwopPnUnLcw9QggYG06cQQvnlYLZFq0nUlKPD7FsWxdNdNbFsXTXTTgmjXTTRoYkFfKaFsO+fw+E7cx8yeas+Ex5W9CabBMVfKx0idt7O7oe6jkJlSk3hQcXfTu7y39pRy/HUJR9NL8j2LYumumumukukumumukukmiTRJokMax7mF3WxfFZ5fqt3NvJ+B7PHImJM6B1GeUcQSKWkYpgXSQBhVDYWgtEDwam4kF1pJG1PKsXyaHb32Latq2rasLatq2ratqYVtWO3LHPC/Vt3szzxz8gyjdC6Z8IH7CSEkJZXQE0+nuvQHmDS5zKDT5hec5ICG58NPlIHnk6prCwsc8csc8fh/wCf+PxD4Ub4ceQuoyQIFCyrV3lfSeGxkCvRirD0gXH7DDYr2NlrPUiCzJifUJqr078VsfxN7nTJvDdx5/wvdGn8sonyvCFCKjZ1GCqw5PQ9HaZo42iDl/8AR5/8u1J0r1aTfEdFzh1Co07WBlpT6VxBIIwX4p23LKy63Leuqy6rJ7DKPK/nsIcco/I+ebeC9w/rL5ZVy2uhZRMoQyoxVfs/DlhiHkZbR4q1P1+uXn3VNGsdWGhaevNqejOa1mp8dPDJyicDhrU8LxcQEoNUpyj62my+oU2X1SqyfVoFuaxDG/x9o4dnba4pv9heeX8fxz/qH9S7jjCE+8L7hAVCKgDLgCqVyN+H9PkhflxhrTadRlsdS9Pno8PSbZXfLevaxX1mKCxUr6XR6+s6NJWjt/bmf5C5bWCyQNXvBIn7rblUPlWgfMLe2N1J8mZMpWwXJv1/nN/KjRPtjzlMqc/TkkpPG0QKrCq9VzWj6UW6KPpitT1GPTavEGsHqFjd965kp9Mldpd+4aRZfiLfWstlgrajNGtdqjNbOTozSOwrY7MdExGC69c4ZRlDS3y1YvsM6/mebdlnkHdS828P+vN+VAR9RbMZLHIXwtE1QYTGrgoIsNpcLEqcbBEp5hrxcV8RlqU9uXcg/eZ+qNBn9RRLdBFI4PrtX1dNot0LZjjkZrmkap2vvJmKCXeFRuvp+sxbJ4bZQvw7fGaar/rZfxvYPflH+0g/HkPh/wBOWEXlmy8n+PCscx7Pw9qrzEIttoydM9OmIwXG/EasTu7yuhVct1KBtmp0ncSxgon6lPUKnpbEke1qDb6Gq18w5UMmx9DssNrVaTnU3d6J9M9Kus6Z/cyJlGLk89YhTtyFY+CFnd30S4NcKckq2BUUsrynzby60/PqaIn0NOrsx0nAR4n4tjpRW7Rzm7qYu7KkTnBR0sHmr3jq2eoE4aMf+TfruVGX9tOPp3LEe9nDC8PSs7HlkGeLVKr1L+dsNiZwp6dcG7Vb2W6/p5ULqk+JZdGi1TSNe0KTTJ8KJsm8LgKrG0Ut7i+F6N3VHnbm7Iuzx+Z9q4M0b6nMFZ4HbUI6w6rxgZBPYKZ5uRftQphaXD9YClnli9VrlXoaloagk6U12MTUylleCWZxj1rVaHprZjlxN4z0PUBsaZrcLW4bHxCtl9K4VsgxeypWbWY5o3jNQng+F5fUaBxnXGQDHaegVvUalxPoO4ibD8n5svDz7TcPPT7cDXYKNfU39VQt2JSJ+7kWFI6edO+UzrUb0FPTdI1g4x1putVrSelkOUZWpF6mhZbYWoPhbnsVtWbrw2qu5WAw+lWfT2qMnXVwe12Y61DTLm6xp9xrkHLK4QJ213jSIItfQeeAXzS4wnN9NLzwz2ksizvqIMFl/ayfy/7/APUf68Pk+OGDcpOKYhi1NTdk0hIv2gZiPVoAgd1W8/tw2P66T8ouFCclqn/ou/6tLbKj+WiK43xD99Of/wDR1kWGfW//ADwE4y8Lym9vn//EACURAAICAQQDAQEBAAMAAAAAAAABAhEQAxIgIRMwMUBBUQQUMv/aAAgBAwEBPwEr1fC+TZZd8Lx8L4Vwv2VxvjWbPvF5o+G4czyG83G4v8llFem8UVWb4WdspjJW87iyyMzeKZfF+3vheb52UfDcXm0NDSHw6wpG48pGe733yr09Flj7KK4MY8fTxjixdZ2siL2XwvF4vjZZY+NFn0Y8zrEUUOLYtM2DSwkL11iyy/bRXCyyy8tjVigxWhdlYZKhtCmKRfsooaxQysVxvLLLLxRRRtZtY9Nnjks7kXmaNptEyL9NlllljeLL4UVy+lcEULTFAqicqJTLsoooodocv9LQ5FkOd4rNDxfKy+F5eUiMCMUjoskTGhLlKKZJVhRIqvTb594r13mJHFD6HIkx+hpMaEL0oeaNpWbLL9aIoXReGTQ2XhPnJYXO+K/BeYoQjrMiS4Lk0UJfjeKzXNCF2Vl9k1isLm0Ll9xfOiiuVl80JCyxkxyoU0bkNkeTOy/deX7EJCXGRqEhYshM3cmy83m+Nl4svlWGUUVlISEXhlokTHBs8R4jw2eE2Vw3JG5MrFYrNcejrFC9qEREssdIlrJD1h6rPIxaotRM3RHTEsVZ4rJQ2/DyCd+y8dlrNZb5VhMUkeRD1UPXPIyTscexEjci0z4eQXYm0PUr+G+zs7Hpol0KTPKJ3lIa43iyhI2lDiUPKg5HgkODRZeKZsZ42bWsNj+iQ9Ns8AtGhxPEhRF0Nl0Sm/4b5Ck2jULF9IFYQxpnw3caEIZYxjIKyMaFY1ZLTHAUSKKRQ0TgbRxQmiTE2WyiiMLPH0SRJOx6aaHptG7aTe4jEjBkUbRxF1mSK5UfByGz6NFGlErLHE/pFYoomTY5NlNnjkxRaE8RTZGAkakMzdkoMqiNkRYfwrsfRZJ2Vi8XixlMsTGz+mkyhl5aFi0WS7NSA4CTxRtIxEKxE2PFIaRtQoiFn4WMfQ3xooSGSExtDo0mJkvgxYYuMokojg+EURQkMnwfBDYnihokhrnZZIop4jKmaUrJcGsN8WiY8JWRRFFobJdjjwYkLH0XWGSLJPj9KxY8WiQjRbKss+4kyyy8WXiZIjFsqhZrEuynxWEjbjon0bh5vhtscBrColhSo09U6eZLNFcGbVlYbSHrIepYp0Od5YsIsseGdEhi40x2StnYhliE6IapGSZZJjELD5LDJWMXJYooZTNrGjYSiMWbGxSGxt4oeFmExDxQh4SHlsWGxjQvSuDGSYxcNtnjJRGrNp8JyLE+EJFl4WbPp8LKsUBIaHFjiUVmhCzYneJEpIchoaFwih9EpDJNiJFoR8zARZfF4eEz6ODHuieVm9G5MlRWFmxsTouySZJMQ10PhQuhyJ2JtZk8LgiA0Jm4+lFjGy7IoRJnkY53xvCZuLRY2jcjekeWP+ktSJvQ9VDmb0bzdiyRIuyhoeFx0xjLLLHWaFiZTLaFIvMpUeU8rPMzys8jNzOzsorO02m0o3G9EpolqFm43ok0WKYpm43CIDGsWfSuCxLsaJRspixVkojibSjabUbUUVijaVm8bmymUKJSKHE2M8bPGLTFHOnjaSRfJyoczcNjfCyT51my8Xi8bV7tNix9GsWbjcXY0Sw2fRYZJ8Gyy8Xiy8WPFl+9OiEi7KJFD4seJMTNw5DZZZY82WWWWWWWX+JOiLLy0dlm43jleXEpm0kkhsvN/uTEyy8sbLHKh6qPKhTTNyJTQ5fmv3WWJljeGxs7kLSQ9OKG6JydG5m4jL33xr8FiZuGyxjtkVhkyTJFkJCqSH1miisUUUNehv33m8srMhq2SRNDVCdGnIk0IWdtnjZsZtZ2S/LZeLx8NxuNxusQyTESRMY1YkdiYu8ItojLFEkS9L9tFEiM/wDTci0MuyEcSkLsaGjUVFm4svESih9CkRmKVkhj9DF6pCWWiUcsghEnRdiQx2THQ2bmLEbIdmw1I1hOxOjdYx5vixemTwuElY+hO8J9idIbsiqw2Ps1BoZuZYmRZp4miUGQdYZY81xm6FNEXebxKW0WvE8q/he4S4WWTENkWkOd/CKKwycqJSsoaZeEIgyLxqRF0xfCeFx05XnW+EpSi7NHXsTJSoUxM1I7j/rts09GhKuLJdEtTsjJMaFCxRoSHiSNRMcqNxuKH0xCxCZFklaNSEkyLZ9Hxsb2MhK8aiNWJp3ZB9GtKjT1akact3NivEn0aq7NOVMg0z5j4dDJWbXI1NIcawymRQliyEiJqxH0WKFko0LhrGjiRrGmQ+GsP/0f8d4XomaovposWH9ETIvGr8JfcMjhZiaYyeIkz+8P/8QAKREAAgICAgICAgICAwEAAAAAAAECERASICEDMRNBIjAEUTJhFEBCcf/aAAgBAgEBPwHsTHI2LPYssSTNc1jsvCQyihorNZrFFCGhIoqixDxQhnfD2UJDKG6OxOjbGoxDzYz0KR7KoVs0NEalDXGinjsReKKNSjsQho7zWNjc2PZRRSEsez1xoSs+NCgliMRpFFFGpoaWfHRWXisMsZb4WWXjU2Q5I9mpqVjYXebPZ0UamgoIpI2PeNmLvLNj2a1jUcWhlnReNeCxRWGRstlFFliGzbCx0dFFEWLDLLLvFCQihyQkmevRbxSJ5o1KNSih47OymUUOiyhZS7GexI6KKFEoSR0sXiihLCZHDkxKxUhs2Lw2PFnZ2i+CzZeKw5f0IsoSGLNlimKZeaKxWFiLo3H3n8jsWKHwfO8UXReGhNnQkjUorklyvFYR0OkXi8IiiihxHeVmhdF8exFI1EqLR7wuNiEXxuhzQ5lsirIwHGs3hHf0ds7FZJ86Krj1muCeK4ovjKVDk2U8Ij0KQ+x8VKj/AOCGyTR1n0XworNZsoXR1zsh2Vwkxlo2QqYkhEcSXKMqPY4jRqVyost4XGijUoss94RVkVXCyTJFWaidEGIWHfODGxtDeLzWaEViilh4XCiij0R7wsseHZTNaPGLCLGuf0N8FxvFnvC7yyxMssvLIcWSGyyymQsViRWGuaHiiiuFCKKEhlYRViXFJFCjZrxY2SEsJiI2JFMRRLn7RWfZWENixX6qKossgx8WSKzRFEBFY9k4tFco+hoorjRWLKKKxRWbzR6FxZKTOyxtilZTIpkSLNkNo2NiyMbHimasix4vFFYsT4pllneLLxYmNER8JDQ0UdiiyMWQ8Uj42aCghwNDVihI9LCdC8pvsfGxrhtwooRfCjsvFFFFFNEWPg1Y4M+OR8MiPhF40RSI+iQkKNmnVmqZrQpakqmP+NL6Z8Hxq5lRJfj6F5Wjxvd9koI+IcTU9FiVnovFCHWLGyyyyxKyqJMUqPkTPfC0bo3vClEi0L0SE6FJ/Q/PKJDyKY5FWSFOXo8kd49Hj8EPs+HxttI8ipn8f/IklY3SJvs2LsZFl4rDzY2zsidHv0RTw1ZohxSE6PkFKxsnMtibExIj0KYxEYk/GpCrxqkOVnyJDkmNEPKtaF52n2R8u/on4tkeDx/+htWSkicjfsh2SSKx430PFFUWPosSFFlC6NkNCRLhv2XaHRVlC6ItkSizZEPKfIixkmWe8+H8e2fPboUk10TokSZVkU8UOJDodcNcwRY5FijYosSJCVjiS6LExlFPEWRkKZ7OkbRLE+iTLvCHjbqiDpjnQ5sbJMizYXeE1iuFMYhDkIoXRsWSFjyCVsgkaoaWUJkZCkPvCJOhnQhMeV7GMeKNWLrCICplIdFYojBFI1QoCiIpDQxdFjiOkJlZ6zHoXYl1iK1JOyeE0RY1fFsYss7Ixs1I4dZfRB4oRRTExtDSGXQ/IifYqLwyhLERCY69jk2JjVkYJj8aKoUisyGPEiMi2IihpkWz2VwhE9G9G7IyvFFEojgxsbPYo9kRjZbE8xxZaOsuFmlCovLfQ8PCTFYhWzSRBD4xPxxGvsjqx9CdlDPY4GhrWFiQm80RLLsrG1Fnscui8olSG8NmxF2XRdltCmRnY+MUmfGdIjFMqsLDRWGh3myQi2WRzaNxnZ2R6JKz0XeIobGPFEehyEIqyEa4UWJ0Pyl2Rk0KTY2yA8NFCkTLFlljZAvFWLxmgoNejV/aEl9n4yHBexrvF9DfBHrCT+iMJCgei0x4rKjYvCaUQjRQixvNj7RJdiHHNGhVFooVoUhSZHyMlOR8y/oXlivofkjIl0bYZZRQolGpBnZKVCexWe8JCiQ6JJMqixYfGZAZRqJlllpHQ8Iiy0P31n6KKGimas0YoM+Ni8MmLwSPjaHBsh40hpGpoih+xJnoid46IsYyz2UdkkR6wzsoplYujZlsimWRY64RSZojRHxmhWU2NsviyxQk36F4pULwyYv444NfZqfHfoUWjUcBwNWayHFr2eQ7ITZ7GhFo9noUhsUhPCw8wxZZeLLNmbCkbG5dljeFCRpL+zVr7Nb/APTNI/2aR+mRhRqyni4o3/otjZ5Dti6E8NYTJNFo2FTPRZEbvhHr9VcEUajm3m+Fl4eLxNFU8xGjQrFCSOkXygiuFFFFZooo1FEUSuF5v9M44oRuJl4oWUyy8JEYlFFFFcKKKKxXLr9VFYaJRNSqLExM6ZQoGpXCzx9iRRRRQkV+hIo9f9GxjNR+xMTLE3hKxeFsf8aQ/C4mli8ZGNcaLL4pYoWaLxRXF8rLGUNGiZoUK0R7FE21H5JM2Z/HW0exeNWKCNEOBVcqzXPY1y+LxXC8JcGjXCZaSG7z/G6iJiZ1erHGiV/0aWU4mxeb4LFc6/beEaocEhJFHY11lKyMdYiEVv7Iz+pDTF0SpmiPjHcTYczZltizZZWLx3+1FFGtlUeykViVJZ8Ee9mSkRZETNdXaFJwfR8t+0JqQ+iyh+Oyfia9ErQm7IMvlXFlfo1EqLxCiULRVYR0kSd4hDZj/HpCtkSIjU1sSHFErExSNkM8niTHDVkRZorgsMrN8exWSeVKiPkskPoTPI8RVkYqKJLEV2RVZjj2SJdHyURnZ7xLxqQ4akeFcEWMazXGKKHwh0PtDWJLHihr2yTOyIqI94iepHsQ/RMogIZsNJjhWe8tZjGz4uiarLxCGx/xX7PicfZ6HiiihCXRJYn36PH4/tkpFnsohGxQNBOjyf47L6L7wyascWJtEH0SNSPoq0TVYsstE458RFxapnl8X2hqiEbJeOhoTaI+Rk5m3BiIpNmnRONG1kV/ZsN4XRHyf6IyExDR/o8b/Ff6GM6RJIkhSoTsWLJK0Vjo6I/kjyQrEJUeN2Rl00zyezxDhsjywceVizBn5NdGifTJ+NRGNsVsoSF0bkJWIlZF9j6m0MbseJxJJpkJCIoZQ+i8dHhfZ5fQyJD0UqPN/keI8fpH8ldcHh4QyJ42zyuo7HkX4kj7FiBMZ4iI8ef1Z9ZRIkITIEyHsnw//8QANxAAAQMCAwYEBQMEAgMAAAAAAQACEQMhEBIxBCAiQVFhEzAycTNAQoGRI1JyFFBiwVOxgpOh/9oACAEBAAY/Av7YPm7lOcz0t1Kd2RtDeqyTPdF+gUXKj+0ErsoAQDnQFkp3TqlTXksoAjUhCsQG9F4XOE0PT3MExaEc7SR2Qh1/ZW+Y18rVTMDujlOYq5thbVAqHmyb0lOgxJTabn8ESExvVDJIT2kTKqyLotNo0TmVWZhGoQqUvhnzT8n6oRDSuPi3LroVE6oibhNq8uas7ip80wh2hTW89EGv5lFtB2qb16rMRNImEf8AhLtOiMafJ9/Ku4NCjPmUkxvSh3XCU01DY2Ka9hz0Kif4Yyy2CFBQPMGVm9KAiVpwhFv7V3RPyN/MifIF11QAIHusrgD3hOp+tnbkstVs0+qqQbC4QMSu6EWUNumuqXGuVeKGwzopiGnzB8iQPIg+lc4WWLLKyhx/uU5Y9grzdZoPvhOW6vYjstQHDkoJUaoeLZg0CeI5+WPkSB5OiEr0iUARovSoyiFOWFJAUtbkqdQgKrTSEwVwuNVh5rO0gdkS5/2QgzPzZR8oYX3i11ncimU6ug5qJFSb2K9GVZneWfkO58wRhlO85jvyntNnaS5SHiW391PzeXk3yhOikaITgN/xAP1GJ3ZR5h85x3hu8SEYjyKjZ/TNwr+YfObTHLXzIQXdNcLob7jlzOaFe0D5lzuie48zvydyFovbqs82XZOzIb9WnFpt5g81/ffvvNaOqhtxzKiZaOijNfoFld5FOp+5vzIHU+Zay4pWWix33U5CvQ1hHNoWqh5smhUhq51sdVGYKW+pl/mafv5k/wCkHNoOLfdX2QT3qBZf6Jn/ALFP9O1o/mvhfgyrtcPsmnpyVIXAFoQXB6iviGUHOdmQjmE+NCfkonDVSDDVcfdSqY3rlaq11otV6k9hMiE0MtZXdAVM1HvDXGM0rIKnj0v3YS8BZqfC7UIPrU3VajXXLUOCqw9C1f0mwkiL1XxcBX2WrV7mVfZTTdzzSuBmT2Ks6QLXCJYQY6hOpkEPbrulrRYIO8myMrVdQo0AQbylQqWQSVcK+5AWi0XCF8OVLqRATyXC4TfFpB1tVel+EAWE0wea0OQ9E1zX5qR64O4rpuf6yppUwHgara2VHZ3PbyCik0M7lUqlWoyqyp05YOhOC2l7HZeJox1RTxHNBWuU1ztT5DepQCuv5IvCla4EwrInAnDRWV1w+pMFRk9llyEAIO8Fx+yyeBl+ycHjN2TbQ0csLJhH0plQIbew8M3apzua5ZDxDkUGuFtFlaMxTjkhVmVWvBNSQYtgVmuWFBw0Kd0lQsqaO3keya4oBQNQVfXEIdECOaI6ojum7gCADQXFBz4LzoFMiV9P4XJrlIJd7LLUL2u/ivqOPhPPDiGBwkd1nniRn/tF1RzfadxzuYuo6LXmoKb+UN89VmQ7KUZTm4jA0/24TuzKDjryWaobrhOZ3ZWpW6yrvDQruJUN5DcDkz2w8TnEKrGoutTuvHZFgHPVZtSszSQeaaXmYQ7b8cgtFcROBTTiFwqOqhA72qGYpo0lMI5jAuOgVV/KVZXwhNjGo3q0ojcusvM4EoHUFRyUcidN5xUrRXavStFJ5KCpQIRWYaprx90VG+08gmjlgdnpO4jrCJKk4aIHkm03YlVP5HdF7YwNOqjLMc0yDN9525qtU5BQi13pUsK1UO9ODjgN0XUuRpUTxfuRkyispX6USr5SrtylNMw0akr9N2aMa4/zO6HBXw7KwVN3fe11w1USpbxDCJxkYRicBveqGqxtuQwF3sswomEKFThjVCmwZBzWWmIx2j+Z3QiXKxWbCWmypuOu6AuFQeS1hcJUOs5QL7mYBS04ZcAhuyVlHpUYEKEAdcqYBdx4ltThxPZCEG/TGpUOjQnu6md1q4/T2UMoj3OEZGn7ImZHRAbpUBGURgC2zgs274jfhuUhTgN4M64Brjl7rWU6BZZieIOhCeZAlOfm9OirPmMx1UvnscWUR9evtvEjUYkSuqCG67A8kQdVos0Kd0tdoU+k77YHfap5KcquIVjAKjMIVzKIZWjsi8FvEstQNe0DULhODm8mCN4qOiuriE2DKhN9t0lWWiu3AjfFQfTgce2E4BQFBUthcbJUu2dn3VqAYerVNOo8IDx3ABOyVA490adZsHos/Loqp6nfe1ZqtPxGrLT2b8rxgMuDT23fthxNX6f4QkQsqG8W9UWlTgN3VS7iKcRgJEoA08hhfpV5/khOLK/5Tz+0aJzt7RcDCfsgPBeT2CM0HfhZfDICBbr0TWvF8NV6sGqSr6KWM++Bkob0nROLdNEeqvu3VkRriSb5bwpdzTvCPMQhuZGlaq5OHpXoavQ1aBclqFqtV6lqtcSp1XpKHC78L4bgvQV8M/hCWFelaYcddjfuo8efYIUtndmzauT/AHwsoK6YDcyt1TgVrCvqmu1ZKMq+BJTy37rTc1Wq1WvkxZ56BcOyfdcGzM+6sGM+y+J/8XqDvdRWoX6tXwnK1An7rh2cfcqM+Qf4qSZxcDzxtqoQxjCZQfMl1llcJhMLA4O7qTooBxc0atRcdTv6eTp587nfDiQGN0ByR+olUXEwYuE0dRi7rCgOsVH9gjcjCDuQgOYTQ8TK4uVr8lkbqFdP6wifl7nDTfG8MJBV1bB0+pN6JpcVmDrIReU5nI2+Wn5TRSxaSVAYSiMjhHUINLcvugOaGconl/awm1KmihjApyiVsTgNTCew8io+qLKDdT+nUp+91LXtnpPyZR8seSECUGt0GOy0xq0SqdTlUaCgeiO00rt+odEYVjHsoqfqNVnfbHTDUflahesL1BX3pxI+UBQbiTyCefpmAqD/ANhylDqo+h2oRq7K23NnMI5qeVwWXRAtKjOVxj8FXqlp6FeolfUvhuKtQKNUDLHJMPbejruHyih5FhKDyIGJptP6tSya7/JVqffMsvVA80wPLmVNBUapr08zgPiU03LVewHqmvH6lI/W1DCQ5AyodY41gqe/3wBR8vviM3pTXi9N2hxsg5AYOq1D7J9R510QPddM7f8ASb2QKLeerU4tcfDqCYTavQ3QAqHJ+0oS0AOu1OpnkYTOhRHJU75g/Qo06nJS0qq3/Ff+XkkIHE77S+9MXcnlghk2G4adbjpOEQeSkacsAhg6o8w1olENMUm6DHZqg/bCqN5goTqg7oVLblvG32RamplX66ToTz1wCcPqonMPZNqgWqCVYrKbEhVOx34wC9jiUd7J9Tt7wKz5/aThbH+kom31Hco/4uTwOqInnhPqNJ1/4rh9DuJqcPutspDpmTavQwcWg+h/CVVp/VRMj2waZiFUpv8AURI7+RKsriMzZ3DgAvHdQd4Q+pSApnM//pFzjfAbjIsmh13KXBdE6hs7pqnn0Rc4yTuOYOqIfd7myCstS4a66FSmeAo0z6aoLVJ+JQdl+yb+Fk5VGlq2ih91GAvotn2kaVGZXKpT5TZM6rY67fUNUyo2/XdLfxjdbDUAyu9Mp7To3AJ4IjBrnXhU9noNJb9SytaKY7bgxGVPcImmuLhhS94EI0qHA3qpcZKF8X5qwpBo581xkCmHySei2avSPAU+NHXVak46iWhMd0MraY0q0s6HYprhyKY/WnV/2qje6jCpQceJhzNVGu31N4XKmEzhza2VSmDH+J3cg+KBZOaRBB0wCeOdJ8qjXjhqU7ohbPIlgeJW01KTfS3QeQFpEYv8amXS7UG4XiUTn6O/0nBxvjK0wgWVLZ9nM1XD9QptF5lgNuy2XaBzblKbWichlNqs9DlTdzaDTKeOhUhB/OkVs20x6m5T7rMzXBvQ2KrbOfqFkwdCtkAOUwpBy1P+0HfULEbmyj/JbQGNyjHbAdMiotzcIdg3+S2mRrQ/0ngCL74xKeEaRM0yLtVUMGW+OuABTfDbltjSJ6p3sqrT6Z0W0MN2yLLaPcoey2gcsqqzyfbByHuqXuE6P3LZf4KxThmtG5//xAAmEAADAAICAgICAwEBAQAAAAAAAREhMUFREGFxgZGhILHRwfDx/9oACAEBAAE/IZyHgRHgzfJiFTEkMVSI2Q2glrA8aFhZv4IhPQsKbeh4HuDgWvBLkbRx7JESls0YhR2GJWTwdI08EOfC7JyTprwVUTOvDBOCKOHOxIy/ghLwaFC2ZE70axJyMxGhSfIleDB7LcC9ji4LEYGISVQGQx7GNcJvwQdnSIYip5Lkk/AgMJjXhmUaTWREojnwaCQ1B8kJS32FA2EkO4KwyiY1Bgi0NK4Yqp0WLBM7rMt68Hsfod55EiZPYl2R9mIVD4NVcU1CWRi2OM8iGqqz/bGu+TwaGIgdDYp9jbUExVFyY2TEe2LTVbBweuxcC3a1jYV620NVkap6NCrIRpkbME/HhrJe8TAoRU34EuRKsiWIYbR1ET8CT7HbG8ZIJsQ0EnPY3ORHWj9Ce6uFC/QN9l0G0Mxqr2Zf6xSaFch6BhCSoUXRVhtk6OMiSiT165g218b5WBrU4sTT3BaJOW5ChbhS5QlG/E8XxISsSENC5oanyHOB2IbNGGJZ3BJ0w9lILBndIQ2hyVDfYZaq/JbbHyMyp+PRULsXM5IXt6H7UQsEdLsZfPjKrbRxVwIP+UYlYvgNRrBH0yZT3M4Vof2G7r5DFzWPgVO60wWnyYNbNBCROiF6EhKDGZqfZOTrFnFFU4fISg7ehayMrJOfFpW3PBZeCE1k/aaKyIwywukyqN0hYxSeSEhiUQRrxgxsHmJEdgPLiTi4Zla5Dz2VDzCB5ObWtdMQobtREXSMFZ+RgM2MYOtIpvBheFsbbWPJH78EghvcNvwMuBOf0G6hRDzBaKcVDX0J0hYI+dCpwT+RTZ0aiVCbFSGsPYTqEJGk+RWNXLHvf2Zj03jeQvrcpCyiBDfk5QuQGIKY8royTmXoKKWr4H9ILNHBjXWZG9dGE7PQwFSbMB5iXAlxIIuTJEzCJrIiQuwbz6KqPyIqaWxMRwQdxCJPLKbHDvXsamhDIGspeEti3jWMinkFRHjmCtK+6eTKpP0GZPRByiquGDSx+ki/V+yEholJAqd97CuaaxBMvUTwiHFlpWxCeEsCXxSg8oWCkyLZ7JeS55iG4KRRw0LJEoWzTJ6EVCYj5eNsQ6nafY6byPI8USjT8KxxPIiKhiIZ06sI+BCRx9/JSfRgxBW9CIya9ESUa0MyNsP+l1lZQQjuzyhQFpYSE4WStdGDMIaPQkWCDBZKqPoXLWR1Q0L6MJLxsNaGREvI1+CI0OJSTuhBLEJSeQ5OKMdIXvkWHaXQxVRLxhRSVa/j/TAIkiKnuuxwlJDyCT6tTkaHNt9mUeEJ5yTkmxkWRsi8fURK0auRXbknixl/JfLUIL7CydUGNQmW5hHsKHGJkmTaCSUWhq9HA60yFNPBkf5GYznJsrHI5/yi2MtY6Ma9wAvTRGD5Gc22NDMDjkXQ8syaH4ofBkaMvLT7MYGjXieEzkg2/ocWTEnxIwjdGLkqT8F20ZhrcH/A1ZviKh9tliaQiuUP8Av5M2OVTlD5keaWqOLKNDfY+CTgesmWUPIvGytexKIX8wlQlPkPQg8jXKK9MgkdZFEqLOB43lqIa9t3xawQ7vAlyXRj+Q3Ix6vrkc1vhGU7FYYyY1P6Mj+UjT0y9O3XQxJm+qS+Ls1s48c+PmIbwSRkiRTk+i1iXmBO6EoTkQbKqxmH+x+x58KQHxejWzwahnNXzTcE5Ib8lx0clnka/wAlp80XY0HxSQ9b4Ghrw4NEEjaJDJGS/RRoghHceNPxsIa6JcCHnCj2KsG02DFEwIUaHcXiR6M+jCXZ85DJX0xUataNqSfEOJdVBBTqfTorkaS0Wd6/ksNPTWRgoSvg8jWxyPYhjbN+E6/Gh9gZD0PmIbJMjVfhaFEzbEL7+BRUWweSaJ0h6FBa1CILD0d0MMpuhmDT5Mk9g9YHKWhEEYXxYLLKyE1OCLkTv8GTRcD+hc4MGbeCCLsVohiQa4IMyaCe+CtlSITOMGQy+yHc56F5NjbguSlOeiHQXJJGDn2VLLwPACVkXMK0DUg7uhxG+VCv9iHHqdH+cwtrW8KCN+VA0H2ij3EFMMfVHbd8+0MYmo1wMj8XMITys7FksfYNGUL1ogc4OjjZcIyG2KofD4U8HPgkph+S4MHsrOD0UyIoPItwW+LbRUrKAlDJ9oX27iD/ACGFoJ7GSm02i2MQmqRuBIj0TWPIOg2cluaZUXImhnrkNDwNYIIQcIt+CEkUYX4KEsC0VLY0duDE1voSJUyUbcJZMZhr2ZZdBakEcZ2nYyJnwpFNSoxDP0I6NzRqvZ0v4KssQZl5HInsJ6b8AqiX6Ap+GJp+BrwSV4ErO8HGJkQ/36oIrDJbZNqX8SEK263tfwKLXNJGTe3bx3pQQ3+YKnHAFeoWJVCU0+DQ3SBvhEkV3E/UonfBKxykRA3CBZ0ws1nsijZkJewU7EGhXcJwJFVIWEV5McBlFg1TKoWds3C/QlN6WZhkxVjVhU9MWrtxoem4Ek87IxVEYd/tzMZaONvwLyyjruq2RaNwwSIcyO4W30aEE7uwV6gpwr3BzHyxyYKhY5DFV3FG4FrrOAU1kWzhavgbPO6SqJzCDwGUeFCbcGUU0Vdgpc5EybEEQvejVogKDEVrQm9pigd2lFwZhL2xENjWBxSr2OMD0XOJ8HVQ2kVHLFgNK0Fk0VEhTOyIa+rJHbbcN9D2sAShXIS1M1ny4M5lqtDoUXFlI7PsGExQsDqK2si6NgadvEZgXHPEMrnWXYrAQhsyEEKI2WbgRELJqINsTgAUTmCgStpG65TQvqIMlozMfoRfexuCIR5mzKB6G/AO/wDhWF8hi1yKuB6mTmGsiwyiimywg9jwMEYTEokZVaKI+RTjkPwMGCW/rIT7gEZfOSF7RfUe0Rwwicu9VQYmnEVJ9DU5wZKfY0OF19aeRBgyGwlzuBem5ibbModQ59CkSyiwvLYZmFkF8zexji9D9AbA4Qg9ZbyKZ5YQ2bbKou7HuwiRInimWKYEa/gp6gyRDh4+zOGUOEVgbJqZYhmPDArVgmRyF7iNi7tjVf1KUzvZBfdRWSTI+ynsVfk7xltyKGBuBqDHnsMy5zB2enENVu+/E/2NdGeFA5GTtkF15FgsnGnQ+LpTHKmm8CxmyH0NFoYxveZRqhA0hOgtvYo9IjW04IpK4OmVNPog8byMjWuTMh7dCINU4T7IDD2QMkUZSnLJaVODXP4HDxNYrctiV+wiwOAUhpKM3EnnkSD9HYibQ5DJlc8KVF0Qbf5C0z6MU6jG4ttC1LxmLLBhsvgxCg7hlbZbdEzQfr+wiw46SSREPw4VSjUolkbPQVGRGJb98DEtwlTGRUqxshCjRCQxD1QtGWUm2ikLjZQlu/0GsxvJe1pjVgYHheNQj0LK8fpCQ9f2DnsHsSOWEWhFW7nomRknEDEP6Qc0O9mxnArlZt+G8eKSKhnVtwWEMrkyWMZ8ATgZRabmXEUmWn6lhVnIfANCaBnR0D6njz5G/gVp1bGIYU4Ans6Kx9B7sZt1s9iFO7RD/KyL4X0wpyGTTNYiRP8AHt+M0/gzbr+zxcJ6HGNzgVVleyVpOigmkKGOehxWJrMbDxSeGyJn6IRpsUuX0LOv3EuHKkYDaj9izrsNO8HTgS5eeRgSqQt7LHmpNlhXKUPQCmGRWjBSLB6wLLRJekjTTeoctU7CUwnYWjO6HiUXrOeVBEfMhjQ2S5PmiPvw9CWwjiZjFVGgn2GNCzcIguORRbhjslqZUKEzHGCyKHoqGiZw8tnpHZQVkNSw7yM7UmhLmikz5GjHTVEQJTVTZn1cNdDVfZbm0ywvsk+rJjGZGTE6DdJFSJdjlNaXHyRJCzfNFGcCopmiRVwfYzO2X5M2oVJb9jfFfOXl5MZY7mHDVFN6LoWRjFAlxwEoh/YPdU0nyjLcRlAz8BpRQyeibdGcsJGHjE5BosF4ex7WqL2w4HPZHRgbByDRrOiL6/K6EJVbpi8mGvAeCHKWjZPLNPXj3AgDyxPE8ijKvbEdVnwT2XJiNBGHI3nknoKJ/Yi6MF1z9DGRSdNzm7l1eHsOXvwEGPrHQhSPoiAdQy2wOVD4Fr0E4sJn1kw0PDZFyzKBSb9COdlUIIwtbG3yE54BauwhOzBM+AhL3AaA7XfQk/cX/QYokYGC7KbXRJxsb5dDZmkVWbWmCEmr8jNS8B0yq13YjX6ZNS4o2mPp7t4rMaLnwjRDDkExPXjDlKNjyKSpFdrRs0qCrWFo+ZBEqkvAixaRgloLZg0Prw9gIUNrTVbpVg7ySG0vshYTRwTEIS7KIRsmfRzUesx3ser2SjolBqcCx+ipvGuZSozUYqBnkWjRhwOjXOkyxaXKQ75GkFhWum0/+C06AoioWWkGf1O5YMG0eSC0+ctZAZIZQZUYntHR+qNIjpfTArQfCwiu3NH9ugtG5DR9GKYGVrSFLyITtN9jOwY1F+34smDAUJnOj3cIIR1o9aZJ5FWFpucmmYg/qhezA/gTs3Gc9YFJETeir7LAuQyNpwTqHXARKE3y/KpWEkglfEmsNacszZU32JVs+FGhtegQoHEZCcW9hZkMT/UeTKKimxhmDvRHpCYFJXMJuARXiKKCnMkrbTgxMXAuBBb4JRPkfoPD4mW2asWDFKwMkgx8XDGyJW+WSQP1ZdnpiiYZ9iGE6YBM2Ws9EYlGOuaItMRLN2LwjYQqKUlT/QwtjygQ2n+RdP7fgX+MC6P6On+JEEOI0cT1BrczPBn7BozOMjPnTiUEJNb+DMjOKH9P0Wtjx/0jBnzaglCMcCewjtFZPltNTHP2c0foj0h6lsGILukKvYi+2RKbGYPnR2SGHZsz9CJyNhsVuCWIXSo0KO7SjVmNVeRSzptpRp7Ru18OalwvkaQmtfI4Ae1Ik9nJcGJb5CG3G3uVrLE9yOEJ6vC3J2vDXpDhpZg32iJdv7mirpHIL9RVNL0NMPsiXP8AYsQcTmfdR7dG8F06D7MmU0RWxTQy0CLbIPDYYprTLVt2FEpyJM3JhlhxKkGTULYs6ODQ7FuNBZbU47Y80XvzvdKvZXuSjR9j+3jD4EjR8BWMeDMcaFnopnwFXBcgJS6Ql6V9j3kymngxlESQchXpiU8jL/pmOHQYaGRrBE4AyRWJjgZxUvkS+n/1ijulbPXAIWp1eKtPdCRRlfyW9vY/UxF6mYo2vKgvUwno830jsiNhF+TkWSTy/EbYpwrKLUILxLt2YbuI7BMSeDKkh6hW+i/cYYpbv0MLGlyN2SY5h1D9AeZhNZforRVx/wACRuEm1yMzWTYsyVGaEfZMyyLsh+gj8WV4Lx5DwKEdi+hSMS+OS+GhCeBDaYlF0GdQc+Q2Yss5NBkDwZKCHvQ15H3Aj+wwrWtCXT6HPFrsbPuS5wy8GuDIo6jog4/8MBvV9lCskaNcDnO0fA0bCQmEC8C9fFFFHXwIIEsEx/EwmE6NnyUTaY2/gofhaHJo6eDBmcou14FBmV8ESQzqqEkyBYbn0hiWOyM/VpJsdCvoNFkWUmSlvomSrwkRiIIQggkhBEycmvLYkPf5Gq/p/wAG9DWRiLrwjN12vGw0OXozbEvZwSZmNaWRQLkcRnwIqiczI7fYQUNTY0HIn+Ise5lYoMSC4r5VhBP7TOoXifwx4TyNwbxyJ/xNYxP8LicPKzPCGnpoWP4RkG45yUsXY9zgibijAlN6FTTwMW3mwHp//kiYPIk2rCdsWZPYtWe4bCTiMe0J7Xe2zLh/gWej/wAGfVfZihQ/38ajzfRlytyiXwXmRJpi2JfV4F+pB/wOHlcD8Yp8iT5+GB7JSXJoX7KCayaxyLHwvKHLiKxNip+EQvLYSV+GGP8ANTG6Ymats87/AIXYCzCh9EyV+nlDk+xhRBV7n1zqP9FBEnKPkWU/jNe0FonjXhautCyngaNM6ORZ83lI6PDFwMQzL+pNNPgDFvffgvFIMwsYlGKacw1l5MHs0ekMbPgQa4UWGxBdauBgIEV76aEtqdyP5RbITLVOMbLiRGPsdr6HWlVK4H2HwZU1zyN0n2CIJIwfirLL5Q2C0XinBzwb0w2PwadHgXPjJhrz68Wx62u0Ianyq8haGzDGvpSoM6xkZuDBQWBfoQk8LWKLDtl9tYdIVr5iDYGg56EdcoU9pic/oz2hUqQaw+/2RRxYAlsMbyJoQ8yYlNoqp0lxveJEOBmpEagdg3caZFZr+j3A7LJHpvBQo8UaqPkJZEn7CYrlHIhaHEieNQ38MJ6Yeiu4b1LwnCzJoQjy+0PQnPIbeCpkkx6JRsQmDYNjKjUBLYyNQ2ddV/pkQ2GkSDY4KuzYvYF8jbX5EQto+EYHQmdfrgr5GW1yski3xBynKZFY6320JrlNwY2JrvkfEtpYw9hseZ4HJgHXBqMRkgS+CDieCUlls9Nmb6NjEEO0ZyK2MX+AUUakoS4XZYhLbx2jkYpmmc7cm45av2PadhzaEN6eTGnonbbEuir+hjGa9DktK/kYiVbC16Mc9D4MEPkhHMIlXUZZ2WRo1EmthkTFkTyXzg6IeAkQJUm4ComIz8NoShVsakGyMFvBG8ml1ORbA3h6YeWKEnyI62eW0NtHLYkAu1ESFwcRqHiWnWy2aVimxuIlpn1k0HMDJrRuUkZj9DE180+ZgcUvJfYbbppsLazjh+BkautPpkjTaHkQ/vlP7oN6Yxi5viZVZD6phj0xy+S/Lw5F4Y01tuxJ0ztPTELWgocjPsZdWcdGDJiWuD7bJXKGhdkMSyvF4yI/LPYbb274Wxrj0LvwIpb1S2L5HcuCT5I9W2EGQwerwNHlju6waOAnt0u6oXdtdvgdSSFgSAckmtYwWiTB+yHjfvxmmNQgkx0+UVGDGM1s4EJ/hBqJYX6FVFGhgRzHkdCguUg3uuRMDvJiAFd2n40XPigVU15cMHLm3BplneTldBf0Msiib2SLSZuSCvk1IXhwPgxof8GxSIk1VMnExv8AA9E64Ov6zz8hxLLRUBVdj+DGWRCjCi+Ss5XZADsXj4GK8bYRmD+wR60L65G58k6Ebb7kTg5n1T7O8VUiT1X8CkFL6kJaSSZhIenyPYfVC69pl6yPXZEMVUvHyYs700Fwf7F5vsUxxOGvoWoY60u4hjB79ckHXelJCjILYWlEihFpE3i9HY/4DtRr7jUMgYlL0CppiuKVhD0x4wQS4PWEs1CPQrVuG436pnZGlz9CY+LOWkBi/Fib5Fn/AKM+GQrs5Z+Ds0suCXAu8zR41BiZc/8AIUCQ6fBoWJpjys3poTwXx//aAAwDAQACAAMAAAAQc8EjBoWu0BWOlvRYeMfoUmXx6ay6XjtzA3X/AGbZASiYuRH9neSekMvhmQVJo6IKjgOQ8KniMz70O7YVgj64EK6jl1NnHs9ttUgqlOAXjNWr3C2SXALFoas/HbCDeRpT2YXbto5eqV9fI7iCPofEeiBFGllL9xdhjolleau/TPGEhAaLxWCpq9LF0CW8ViemANB2inAMvB5t/nuOBt++Cau9mBLwqLXvyHseJV4m6iyc+L+f/wAuhCIVhmVWmGkZUegeC8Z+Mb7+RVrcFWyyYhJi+r+zGG/qdt8//wBphykqnd8BH1DzpwCRJyvqqP8A/wC6GYe0WIE5F+CldTRgFAiB+X//AM2dLnnLbx3HsfWhLlEnvjn73/4rCgJ5W5K+USpQIt5i2VdLoSvy83w73A7F5nMy07XUQko/02vSYrFeIUgw1IjxS5X2TMTHogWvJnzjdPf16AfvdlCQOPlwoVNJ/UOrm6bUxVe7cp12nPaDgnGzaiUwuA+NrIKhil9S4OuJOWCLhHcxTcMZILzcHBrZTlNoY16P4p1YKtSQKbv2u67oD0Re18fToQEy/wCdzLDsijqxcdEqxorAbX2dp3KSRCE1lvQogdLCZpXAuFi3te0ZuH/nrYT4ioZqmK3lBZG4vQIvyrTL1c/rBi5TW1+Mz41KB2r4SLRt1DwOyqAT6cOBe+FejxS+FkHpfk2XMNLqrydj7SJg3QlFtNa4r/Cxmhy5vhrBF2wgMiGKEFyYrkkuFGpTig/BV7xW0lonWIIzJymM7gkKbrt0WPCfPF4nyIYB8LR/UFx50s5S7sUAAo0X4S9CLb7xHIRZRT+X2w+tqj45+in8O2VCIDtfLIh/RpwtUrSfszzECiFuNhNHHmn1lfD2qgJokFgLuN0pQCjqXv2XPmJGBctkiULNhvbnldYus4MzKB/8nJxzkc8w9GsWNcc5xovasUV4cK6CEgc9YDAhFULMoqqCHLNhQvDtcDdIw88MOlVsCdN8NhgoM3Pt9nC0X3tQsSVK2H5/T/3Fuqp+YjLc8chqOkFG0K9upW2C623CQ/gxAskkQohsnNTgU7vddsXEa4ZGJl24E53JE9K5vJpGrZ7fqJmkYsfpXvbAx8KnH9GnS7BEicSN+cLwuS8UkPmOWHSqW5oxD495bTGVQcobShtjSxjoR0VZlRh8OP8AE9qMTsDB1M1r09BZI/W7HqlEHP8A1QsxMZGOmXl63In9ToaLD8dA1DluWRmbdXxGXaEbiVHAulVcbT//xAAfEQEBAQEBAQEBAQEBAQAAAAABABEhMRBBUSBhcTD/2gAIAQMBAT8QSYCSQfMfm7PsGzOoGFty3fpwRGMWh7MwZHxEEtGX+WmOfGPxetucbd+bL9X8+q2CeTbZ8vby0tPJuZGM45GzMhjWQJb5Gyb88EghJTAfZGcnqDCJ98Je9t/l1iW9uT/lcl20Pk0b8bPbAtsLzy7d+NuoUdjl/wAXVnJbn4f9JAQS85Dl45K325k8tB5bfl3cyCyMJf437vJ7Jl7BaWxu2odu7324TM2TJhRl72ElNr27ET/MG9bQJSBiE8Y7bkJeI0dJDjc+Shv3fPy78zbMt2ycsnJl2DPYPk4/N34o+2LCT+WNjkvzL/20WLJI425iPLNkCUdn21PI1Y3RpGbc3ZF4RrrF2XJfvlvx+PYLki3bEYE78bnl1aFhl5y6x/ydmf8AHwifbGxgicTkGefXPPhuuE+9lBz4j3EZhLpvxl42zf8Atwt+perhPUTFz50l2DLeWfXPLMn2w2bMlI/s4+OZ3bOQfL6L+Cf6IBCT2A6xnlkwvtjOW7ZAz8z/ABmSj8d2ELGPjUj5KnS1ieWMs3sEgsvwDyTWJjJ/iEmn9L9EGHbQiMJ7aT3kxaZR2YbYhl+d/wAeSPhEakFrBII1Px203Fs7JHLZGOP8jcjKWB34tY0iZvFuHhN5Ao7pne3nxnX4ALd8hTZHED9jCBZvby9nltou/m5LMGPhW7aMi3oTjybpIb78kGLPr2F7C5xmX4Bf8J8wfbE8gLNnk6S/2UthgLcX7JNh8CW0ZwlyYsR1uS5JexRz5aw7DDt5/kztm27ntu/M+8kt/JSJbqSmJuUhZ+Dv6ISjJ89k+ZH4IgH5/wBXtllOrXfgOx/hInsINls/wvxBy3ey2bZkjJd5YlrHLW9kuFpL/I/7LI3SFme/APgERQl22D8ukreM4/yCSR+W/X5kmsCSzZAlrdtjlsQssfEhSXPiRsPys+Gtvy4MQWayWv2QljDv+Wd7NOWfCZclZ0PjZtS1i62vhmJe+3jbtuFw/K7ZF78OxkYZ8yxlwTI3K9lYN5LSP8hzlgd+FuxpLvx2CTJS5NuW9iLksoXJ7HLRsJycg/lrsTerZ7LYkmnZ5JcGEHtvbVvLR2Cww22/GLbqMtfjSWxPbrZPIV/1aXE6vZC23liziPjTCD43Id+0sg/to2Dswh8kvkpsTUj1IHjH9LofNCfY2bjIe2fGHwYcZD4zLCWY/qeLD8u712Utjs5bnl1Jtu7bKvwFmDB+yEAWWY+3GgfCR5GnWy9fmCHZ+T8Fms+m9cEOfNkfG3t34kEk8e2Gx/LR8AHpY2jssBAclGJbsC+wbAiPY37YnWP5I8CUnD2Z6hhILH2OIlzyHuX635O36lDeZbj274yjWXkvdsHcYjtt+2hcZLy1Z578dv8AqA/E32yQbIC5+AXnQF6JOJDL/Ia/42k0RPYgls1OfNVvb2RvIDsR5G7HST6v1Lvu3QST21nIqNjzkos/s7GWjk76+F2LskUY3RBCe5DIS9ct3xAWLsHsB5aMf7bEghbDyU+wuwjLhxtHtyyGusHxhXC5mKkrxcRf+XD4vMSZloOWYWEucFWbZElxf8gd2bGCN9txJ5J/JESnwHkTAJW0x5sGfR1YLklF+R+CejZHssK+T3kwdhTL3SyeXgJHYS7LV4vcBkziNwAycyTWyZeLX4OWIL5IMo+BZLCzhCYZInt4duulgtPl1DF6SnSF7Y+Rv2Bdo5JcghtoY3LJs7bM9tnSAhkVswgWep2QGBOnL24S/PHtuSDkEgOMjkNRj7GklgVhks2Tby42lm2vZt58AZ78x35o2V4hP8+ZGOMIwXFp8k1kZjFD+SDDGtkl/wC2p5C2WkllfBEngB2/4kf2G4nyOSbF3YW625aOQ7v0YYQBshH+RxIMc+N1YfC5yBmOISwyDy3vkj+WEWc+AepBYTN+ja71mFzlrYTkIbsscbCZk23Nl9gXPfgjuQ3dtbhe5GXLexuXsxwkDAkfSJs7HYJA/JFwkRA1lrZibL2H42OT2MPbIsPZzYKEcbcJAmLOWxn5Zp8E7GXXluTIlyOyDH8LIRmfFByHxJG5bfiBvcc+DE6hvth+Nne2mxkslz5kn7dhLAxjCCD7ZCevoCIMHpIIbOyN7ItJWbJjMjZIIBg5yUSNpsfmS55avtj9jGNeRj268l7C3OytsCxuWLLlr8nGWXNt0q2p87ObcfDnZvJ6bttkYW7f92CVW0sFvZbOQxhAESBbBKvkCe2yxBsGQMG5ZJyMsjSRt7ufB/PiAsoVIISB+Rtmt+Djc8lZabGH9ZwsHs4mIRO58kIf2N8keSHpY2c+WpDsz/qFtraPfgJB+39Fs2pdoP1tJo5a+3XkTltO2QsG6tIO+fEjLS4tMHJh7yxglqwhvybAcleMp0MS7eX80F5ceWMG6Q/2JpCozm+sCKvfgMOQ7aTqTH8gnlwGcfbQhYCUG+wZ8WN3EWECEji/mW2BKgZIITywcjGMo/V+B2AJghIP2/qv+l/dbexXjA2TIGwsNmxPYFh7PJSyOLhHGGLE2x7BgpcQbGRr2ESSe9skZk/5PqMo6zrkK/Jy/DLPyaZ/7y57L/tq9kV/1Ys/liyn4FbnB1j+kI9gWC9icOzvIGwEL8A/ktRmZav2n2B+z/Fr9nSxZGDDkMWHJVsFa/fjDHtp5aI3ZiCEbZsLvy/6gE5f+Qjje/s8ux/a28YD8gPyT+E75bRG7F1gCwj2XkmzMJxG/kayQScgiRqDLvyFffjycWn7LL8HYhuu3kufKGWVbM0sPI5/9e+WE2eS+IyUL+7GWLECBZYXrW8fEX8pX9tPgHI+Q3JbcQZSYrWeuyt/3ttvw/xov7siYyG/KzkQl/5bntlsfk5IeX5zj2FJ7/qQRVraFtj5f5nXsYn5tYu2nzLPu3X/ADv+HXJ2zlpPkv5ILiHJz6E26bZO3XrfsXTlrJdtz422bs7Y/NLSV2258z6vY6fGPfmfGz/CEsONSzE9sPJ7h6hQz8hD2zW9v+ty2W7Kwv7MOS8tgfu23qH5lh/jP8jkQo5dd+C78sSRLB2E3IuSZxOvW2Psr7be2Wy7bbCS/GXPm5It+a+STAyDnxsht+bb/jbY/qAgZD8ebRZHzzM20MhnZox0Q3aMnxj4wn+IV/2WH58Wz7uMNhDsPzLA+Zd+Z/hjHLbY1LtsuchbHx4cmjE+Q9QKHPZn7IvEpcfIQjT5YflmJNvzy34wyDfsXPjbbb8PridT/YUduxBd8nHY25dnZYSvLAdmS2cht4Jbaw9hjlieyIRaexjOywg5Jls2fMsvyWQ/Nl+Fyyz4uXsYmFOkA5Fi2mEpYSfscsjLS7aE7vZo4s/yGvl4t1vfsNJ/mSvjg5CLoj/hv+HMumz8yYM7Dz5ttshm9+6WbsP5HkXLt2AHIROtzi/lhZfYzDwhfyWnbP5YMcdst+C3HvyJ3JGQg/cW/XI3uH4rBfksO2whGrcFtu2WS63ot5Y8fE35B6hAhJggGwQyMMLWa38brLX2x/J3YDuyLL0kDBT2/wCPmDILIxN+gO2736lIFuwDW14tuRVzst2AnHwtZhZynbwTXsOeFuWs7ojewBxk3yGQ1iIcuJP+RekyA/JEZads/sgWtjeJuWfy3qxK5BhSPiJuFgwmDj1h8Xk25DsJIscMDy1lgwrBez5y7NvBkcSnk6dbPG2y/Ick8trxrQELjPFjYWVbsfCyJLH8tDL2sjbabIpLgwAuXPgZ8TYN0v8A2PiJSMuWwPsZDtz1OiGeSByGawhufAZ5ETkp2T1sLQjDs0xJM5aXlrOX7JKvgTA9l2PkdYmnw83HEmWvyfCfZv2/IGMDbjiTbxeEn4NPJl6wGfBdQG3iT5b2HbDJO3qTdexC/b3AsPj9i//EACMRAQEBAQEBAQEBAAMAAwEAAAEAESExQRBRYSBxkcHR4fH/2gAIAQIBAT8QMewny18tnGa6vLiDbCGeXoWS8sGSSp5HW2BuMeQt2iVj/cEZs7NsRvfkCCZzA+SNhGpX4Lsod8tMMYPyIbbvv5vxKbOH+QM2TxapVYYReWGCeNr7YyxO2v8AIgGOoshmhy2YwU1Z2ghyB/JPk6Etn1lGG+SXsmWi163+CX/LDP8AEL7ODligkBsvJL49t2/zeXH4NzkxAiAPJcnZk0blxLP9ieJgia1kPkT7E4WAhvkf3Kfk8dkeJcwPSHPC1fYZ2EbXsOS/2BL+RIuxEHMnqdEGsBnykeWT1H1aPkqfLQ+fp/twXrtmbD4QID8vukHIjs3CNpQMOkJ8LVo5G+5c1zGS+fgJLq0WJ75f0tmeW/gLNOW2GFtt8l6fwzALB8nXkseynkrO266zPly36ZF5ahD2w8Ifbf5LOxqURBBZSDmWjt9kvshGjAfbW48nH2Q+QX0nHlt/Q/AFuI1FZgnt/CIwZY+yf7exhYwT2/ggfZNlY/l/LfwJfsx7CG6O2E8LFkvSBPYfL1rYs2w8sns7eQvttH7CTxyVcv6RN2xkuJYLmQ+QLf8Aa4QvIOaxrk7+wHG5nv5mFxcdkJmN/j5EJkx20+2fiwnsBbt08bqzjyH6lCs4g2I+Rtz20/LP7bCwbYkgyhjGxf4TqPIzpAsXZH2AeNh9sHlqeRr7YWwDbQZdsVgw/W7IjCRvZMyLk6+Qp5K+kr8vULYJ1afIidy7BySCYmd/j2N21+S6gPv56b6JC51G/LVtuMux/sA8JK5EJDF5b+IgpHyGZPqR7ti1/By/1aSNnHbvpAX24a39pyWFgly63Xtmb+KQkMUjY4m+clzkTBsRAYbm2nyXY4iJJ5fFKz6LCRF5bnytGCP/ACq72dewhKtmfyxt+oDeWuwb7+yEBKYJCA+SW+D+MPkDMqWksYhMQQO3LPy962Xkt+Ak9IGdgB5ct/5I7+yw9hZmyP42DYHl2O8nE/xaM7cSsa/gGSl9/AlhD2VazsvxBn2Vj8rfzZ47C+SclwQ9gmsIRpbEn/JTl/T8AS2LRs24gu5MBMRGNe2FyH8jYbeXdnYT7Zgy/gzC1+aTuZ1Ih5CWl2H8Lz2O3E6Lhv8AyHG0dfk9sMlPxueWrA/bfx7GPxqXSR+QSAh/Js7M5sMZ/YD5+Bk9jb5JLCT7KTtjXlu+wH2a/AUZ9gSTP+T7bPk9vERBBtgETAFiRsR+ABcLxIgl55dJdgvqsHlshByBPxfw2UlfwOMB7L+fg7ZQt5sMf+Rxs6TESHJ2Gbfki0g2OR/UGXJbS2z/AGz+wHyBnUqahySbs9gZG2HZ68tsKfINIX2IbDH8aEIMJyGzz/gTuz+HF5HbCCy3FqQmNez/ADDLLcI+W/Mg+ysh+7CMj5GRk9liZoSOR9MHyE/gNL3XNh6/j8EEeSnuwOwNWGP4J4WPUkOSl/Hv4JiSvkiXICeRtwnHM/AzU9LM6v8AuAW1pf7unZ9gy3YQ7H8ZxK/L3yXMkGyH2M+tg6w/GWfbny0bkCmkusXTf4Rhl00lPklnbchW72dIS18hSyxcnPJLcI5dfbN/OmM+2l5eqeUpbv48Z5bfI37Y+xITIIB7fwLJ5GNC0eWYfxMc4XoBlaEbYJfRDNxoCQ5LzSYc/GM75+LH8Q1uLH2/hZDtCENv5DTsK+uALaH1asIZmQLZkhAfYC42h6wObIYDxF3k8H2PYgx252F/DYt6jyxiI+5Ebp/5Dylvf8/FBPzaL4XMH9k+QbAJT8HsBl6kcLh7bYB7JuDe2/g49vCYPux8I5mW3kbewfs77Is1T+E3dtLn4YDv4ho2wfm4Mfb23gs7DjDq+5DM2LWfCwNWdceWo2kA2YTr8Mn4G5F5e5/JFtr9tujaetgOMus+WxjFnpPWQPEfOwmzk19gByce2TAcZbkpmB21YRDjv4Mu/Js8HLKeLTckmlhy15j+y87GrroyBjfQl3hAlmwGffxN2wEHy0vknRBJfsALZJkGZ2JI3+YC1JTy6QeQ3O7IeE78JPUv2dORnkft7i3PbnjJP2EHC11DjedjHy69s/4GQkLIDbTliE+RaHyNN4vd6TzkukLk/VhPJk2GWMTYS/A1yIZIfZR0n2LMn398mI+3+Z2D7H8WzDGTtqWeyDySgpx5C57YbZrBnJeHeMJ9gHtqcLE3I/7AVr5bGsF8tvJebA/yz+zhtjJa/LXq+yjW02vkaOv5Dfli0/T2VCSaSFtDyEfwT7IB9hyf9/BN8ukXxI+yrGjWGN0cgLD2QX3WDyMNbRljqyYGW2mVtoXl4lllYP6wKCdgeMLyyHJWb+FrJW2A/sEeMhxlPCb9n+X90SxiyfLGRZIfErwXbywZdjVnLQgd2Y5dPLcZIQc/CH4J9sMhRlsLnLf2Mpfi3+Ef5fQyhuyHs6/D38XU8l/km/gR8k/yPlIu3xY4cld/FhmfZV6zi8mYi5JfJ15eJpdiPkvcZ9pCMnIMC/3IfkauLAQFkvOTpDhuIZiC8P0aygZd9mPIR7FAU+DCvZQ6xTs5IzZB3sJyT63bILYPi3+xB5Ly0ljDkyaWT38LGWU5J24Sr/Uect72C7AeEjLryAS+pB5buMNi9G2eyPw5OpPx+Wny1nYcWahPZCxY329xImS1pFXMyFPesjJ75+CJAzLucuXXty5+Dm0h5sCM2zzYe2G8JCRtMQ6h8lEExZlgw38CeyrIl1d8T32e8QfSTPIddgFmyZgvkOGRLrfYb4LfAyc7DHlrzPwdbp7cWIE4iy2ZYMG68hPsQ6eMfpJnYH4QG+x9Yps4GZA6s+WiTzyUzcv2Iu4w2XCB+sptWx/Ejann4BtPYJfxA+QvFqD6XQyBKEu3L5lw8krbyUvYwQHJ60jXsQ1PL/F3ZsJ7ObsxX6EcWVteTbDPlx8h3y8eQ75CeW6ze3qMF1vmndhBl8TIVY2142Y9t7238HXZSICLPLMfktQaafkxaloR7ZsL4tPk/AvsueQ/YzPwchOxl+APlxGbkP8AIH2/otW78ssYW17ZPzDpZO9H/wAsDXk5mLP1lrVe5+B9hgvCWeQkSPIfcDjOrYCQe/mp4L/VqZYdukka9tkukj5+bjBbMsbbPIb7Ln53AfLad/Hxv7jHNtLb0Fudv2P/ANlf/Yv7KX9JFxn+kf1s+L+B1PtI6No8h4N09t/sAxE5sQdkeWFiKbB/F/OWPWPx+5rPHJVsf3P7IfI2xyx+wLz7AMhmQNgMJ7cG6/bpaPLFgzyFtb+Nlsc/DKnL7JIcZ7aE67LIJDKII/zCsKEmfmXTsfzOjl2BhZ2Yq1dgbX5BvkfgxjHLIJchLdYM9kz2z8btyUnpJvJJ15HJw5Cmy5d/HHLObGWHn5/qXZlsvZ45a/Ii0wZDe/LH5JwWEagyFbPwZYe7clhbP7eQzye+WtkEBCli4WiYZkx/bSu3Atjs4nE8hPJSFsFRBYs7Zyx+DmAkCMvYD8zWGcSvFsJCSD2AnC9huNmWh2UY5bbaEzDdeSPsQmEd3wLbH9W8xCWTTt2m2vLv5erIsLLF/Bjk6PwLMvZIP7NCQyjyICMbNgnNjJYPx20tvbMmAfbD5dQTBAX+fwamr2z5OzPsJzYEhVn4mB+S1Ow25+e+2PliRrP1Cl1nQtg/twuPlzAkPP1rH+2H47EAPxf7I8s1hsCT+WTv2Cd2DHsWz0LRv4cltv2LdgfLx4bJF4iuglnSAn42GYP5uSftu2nsb9hPtpa5awfhLlt2138X9bAhDyIm+Qz2H/QyQ4QfxI7f3QBY5J9mDyyM/wD4gHfvx/sf/SD6I8Ml2TnGPwv7y/nFH0Wj2OlufkTYsgy7H1GsnLLyxYJQtJhy3SYK8h52P9WVGc7AoVkkn74jhach8YD2A5H8w+lsPH8Zo8P2KvImk8iOZd6fVfYkTsfEdsIwvtlz+D2MnL/H4DLGfYS4yQXZv4WiPsVNwt8smwYdDa1/GyknSXo2BxZGQ15GCENWE4wOnGQunYghDbKcjN/AeTY1k7EersEkGDILKvIQnUsGzyxYIRfBgggVgyAmyhfbph5Mrwsd9m3SMAjdEQ1t7yBRL6Gf8kxf9Za2HCHZ54yZ7fCh4gQGXIRkNgchSOtgjUFkIs24nGHJN8sbScHt7iNJFmeoHBumMj8gzW+5Efpd+Sq/1YMz6szyTdE/pH8QSDM5MAQJEEcgdvQTpdgg4WyDeQ9tvJXTCksYH2FnbXgzjhlkc4J7BAZVoh/s0QdjkUZMZB5OORBHkmNiPZwctvYL/wBxfc8QZ0ZByTkgfJC0Gw/lrdtRMaNt/kKBv4ctJGIvEmAdjRG2S1RtjO8vYs7us6ut37ckkrqIBxMt2AsIPY/Jf4y630u+MNNDLV/HfyHTX+yYvqf/AMgkR7saWTkBMsiOwM1ufIy5+QA5ISixQMslfCALwFqNYD5UfMQyPnI2rKzreW/hSDBkWxwQ0h3+Xgkv5A+T+Uom3yL1koMQmm2QOSumZ/Rx/wDHGZ7YPLEm+RWgJttJ7BnbO3R7DUhJCOJNEN1+fUTLG1yILLxJ/wDT/wCYOrCTs/r4t7+HN0v6psXo+xYf8hja8XmF2JHDk3Z8lODlwB7/APjPm3rHl4/P3cJk2SVuZLBNll//xAAnEAEAAwACAgICAwEBAQEBAAABABEhMUFRYXGBkaGxwdEQ4SDw8f/aAAgBAQABPxCvxdwNHIHMrssvDOAtPCWQVaOZpVpHLn5iqym/ENIEuWaTCFDMimgqi7uW2U/EDQFL7m6hPMXj1mTBcJ4ilncAAuHkjUus6DZQgNe45b5iCzty/SEBK8vMsZvxAWF7ufdQavqUbGohU5+JaovMSjQUxBLOfcwUGeYsSnWYu3fUwbY2BfEdVyy0HuI2eIlC9GM1YJCpKdPERNpvaQNm18VERv5TIANLfXEQFlto9TI18xXkD4g0oX9SkFG3MawT1RkaJvXJG4vnuN2m5QsOfcs0XCJpQfpFDAGXUx8RAWpbVgbwDksV58QVgmeYmy1XkRB4SgLQvFwdcO2uEAQrkw9HmEULALz3DQBPJEOiO0AqLITg7nYABOj5go1f6idC+otaRGr4gNOMNUizAMOCEevVyt8A+Ypa34jXyfcsANiups1F3wS6f6SpzbKhvnqVnrlBvuvZCuKkuYbNAv46l4pKlki7ZUOAkDYKdvqAtg5LOFUXRS/ZB0p2Wj4iKOQ7CmqheSxQG15icwnhJZll8OoA3YpwhThfi5XrW2Bp37ihtDGyhsVI4vDKuW9tXIxf6RJwJc8g6Z2X9wTK/jN2JAe1TdP0VK8Ku1o5n39dLGzZkpLC/wDZQIl7iAtuHx3OJhsu2Om9yqwi8o9VMPbHNn3LqAo8sAbtrxFLXiCVhUQYxs2qpEqrBTzxKLTA8ks425URdyw6YUksjLOCUhRU6i5KrzLMBL8ReyrdIoPIX8S4wz8wdFXlgOi62wbCaI6lG+ZZ5XtzGrlFg0xx7M3WEPgB0azcGi9ZV3jKURwYZDVp9Slks3gju40eu4AapXzzK5RyD4Hf1On60Vi4KQvoBdr3cuobluUhF6rReW8xtvo1paCwNZIpxxBadJ5uXFRYwpS4IAratDG5YAzoZbxXPcaHOy9Tj2TjVTFS7iXZ3L1GE1dGuoBbgWd3DQFqbONXSuUPglD41lRht2fEUV0/UFzysfZcTQ1xHIVfcVM1fFwAXb3YlX1ES4DxONO+a7lM6+Jw6V2WgaseyWlIfLKbb6RU8e//AKjnhRbuPhQpMI92j1HUOXxFau3MHFd3DvkQP4io9Fg6YpaJTs09TAB3dHv8RIJXE1dyfuNUCX0JQygRdAmht7cjzN/u1Nl5ZctdUUeXUZm72MVv3HesgN0fE7iT5kLGI8t5AK0QxablVKb7ni4QA1DivmcirgEGqrxK7RT6jilVghvoPUQDBuXPC8+I0yv5lStvoqYMKeCEwbXZAH25YnOR8QACuoDffZBngRJQNe5wF+5RWqzuGht0BiQNDsKRQaKAaxMPGOqF+U9sunNRRI8R1w8oiRtOZYSjLqAg7hcGUsQahwyldruNMWw8KVTssJ8uAbIBJEYyrLyOEDaXtDcs4Wv2dS9kNpWv3AGzdnqeliUpK3LklHbeGMKGb8RF1WeWYG0TtqByhUFrLybFt03MOVHbSrio2Zj266iOYW5V0wZWoY7F8cRjTfvxKaAnmKY75UwBXYRd1lEGwCVBy7VTG81BMHuKovtBChfiKQao7lUHmU3lS4JvlH9StBfMtr0xzVbcKqgzqAgzBtkNI3sGHVTZkqMu3YLIiDOgsVVLDAfk4h9X1IoS/VZcrKH6yJDqT6LAlp2DUbgDasr0xkUS8TYSDsIsbUgfpYJSxaHJx/URbl05YQHTe3Dork7hVeqjwNxTFaA4moHxBe6uJ6jLWhc6VHqCfDzKQhxTGrRruw23rlwRTb5gCihHMOupShzXUDf4IwuYGQwofcJ3HJjjjxKVB9XBUJycy+yOaRiEhZ1Pj0gW0q2zcDpKkoog3JtYxWVBnAYJemUih5iUZEb5GEjTxTFnA9n5nd+NV4tXXiVIfDMf3LrVvbqD8ziVgWh4jNmnTiSxgy6UD6l+slUPvITueo+eMZUZSmFSwmMWg3iEtkaK1Yv3Lw2crKn0PubG6CWVD8QhXesFaOZq9l9PEpgaZTVs8ECFaS58hzUUZ+LmhxrnqA+SpWYPma1YncbvvmWKBztweTzEvNLZSNqADlfUVd8woNwNmRAL1lygsfqGaAWpfWGBEdcygY+ojxFekyQ/MOKvPUZsp6gd0GuJS94QeIJSIEMYYAoYZ+ZmnwlyyR6VQsP9ghSlox/kTEk1HoTuY5DK3S0H1LKGsIaYcONSWeyDjSz5XJmivGp8wQaW1NWqLtCUsvfUpxyY0KFlrsvqAUG9MDAb6nEj6iKnTccvtw8Il3ifMDWJ+IiHNYwDWruCojQdS7hwx3W/UAdvY0W0PEsTuBmkyIJ8VlSo4ZrH6UL8RnOlqKSr46lxvmLHxAuXBW0QRUddkoKD8RiUr4RC3QaqUcbBQdykzYXR1FCkcLalrR8j/wCQaALOrRxfk9RQV8GF2CGgJoRfDX+S6a72VeYhlj8IDZIMaSOdEuFtGZ4L6iYKsYwlaDqDxeI5NUyqLvmJsp1KR33AdL10k0FL6eJbqqiCkaOalLQl+IoFB8pTfNSmr58QLq+ZSXGoDByIt0hoIB5nyRjEdSI9sdqvMHqGE3HPUs72HsoTiIo2NsggHWGAKb5uWizTHxKX6vR3AaWrC8rNwPsS5dU7fMoI2Nj9/wD0Ce1lvkuARr5xTmzPDBrwhL8lTzJKyht46jbvmCCOewjRTt3UO4fuYKyvMtOYDFKY7Kc1sBosqeX2ieaIyjs2tirzhLjzM3xASqBnCOXxUVpsGw8wRjspU3PEogo0w8RDaV4REtSiMt3hOGbrv5iK2FapjVuC9fPHccNhb0E5SCdyslwXvcpnNsl7JtMLtgteoTAHQalkKJweIbV2jkjV/wDlPzDENgFvYS91qgF+U4UwbPcqUF+GDpw8xHGGHKwBUv4g7FJBlLrxFB5uooMNmASjFauKlfzIwblO1wuQBVtwWk68wNMllAX7gTr8BL2ZsC0sviW/xmi1Zg+1RygzftyLMStqwCJcAow6QkFXsAJdAygcAz2xkUr3BVZb1ESKNupVRtTqJVhCk5GaWBSjipsUcBZq4WqBj5hICKPMN/8AkjVhSeu4hbRS05IV4grJto5Li9VxGgpahUBsvouW2rD1Def1KDy9w0oLnnF7O6lBUz1KmvG41a5iKFHwiHcC8QcsY27Oo8cwL2CHANXLRbmK15IXdf8AkNle+4gtUrs8UTDzcVNjAeVcrX1LzJeg8rey+0VaQEjbFrGIAQ8MvRMFG5Ye3uXika4uIYobDKTDdHOXBaUCwuH2JtoHJ8XwlA/+WHkYPhByCQhgqKUm3vuUJOXUuCjYAZzDe8nglCQKlWqtigaK83DBXH1NoIJgS6WWSriXwlFV+4ivDNwG6yUi2eGXdKXEqwLm22X7mLHEAvOEKq0PmGUMFQ6WWdgdbzqGRbBVtrM2cnquPW6hwblYLziVqIEcCIK7tECa6Ik3BeA7i6GtIdwSj8vF+pTJsEpX1LuCw0sqvFZeX1LQ6F//AEJq6g8kEflXeD+ZyLVy79IBhN0n3K6qcLf1Kt/UC2JRVSqTkZkH6m4HCKrWTYREJbXjzEcb8S7RNjFXXU545qWXcI777jfJxEFsQhawp1sAFrOojtLKHiFKhwfWOEpwK4slAHqCxRRxUKhxcLFylzkgk7fEGVLK0XBKy1hzKNb1Qu4eKA0iuctw8gPLif1Hgp4slsIsG2ipQPcyGn/OEHFLz2v8qGu2QXBc1pqDttyUZyeYqMMhZXc9glTkuK8T4gKu5a7p4uKoO1LaVQM+DAcN9ShhhQzzO5ElqxmCNNxsLxPBKFKCeWURph2OMv0E6uiGXpEApX1AgNd7CQ0MOAcasnkbOmVWqliDi4idTh+YRCp13NuFxyv1AzFLVWceqiC2G9bMZ5WGfIqlYLQY0qaniawgudrHdjl9qtZ67GADZxUfmepCtRkWcBvYRtTKVyTQ4XCzYiFuzVnEWd5EHP6lVW5E7C4uqkiOBvllnVWR7V6ReLG+Ml4VxKKC5bUEV52NdvcvMXZzB0YfUByQAA9wKNvwwNzdzKN01PGR20OwM8stD7lFISwdDp5gArjsuXErb4gjDfZjKw/aBQ6dC7Lyue+pTr1vFX5nM1YD8Fh1sM7H7g9NFY1PMfbi0nPi6l8mPVn4ZtR7MRat/tL0QUe4o8G6Z64BLu9LwIatw0tqCcdrevVRAL1Jnm4iICOBmiNn1GWX9kEXe+JYq5UKCu+JcDGgTUyicyhzGJX4g3MPEJdVAx8waYRIgamXtAXBeYCdzVdqUZ062xGoGl6jBFqgw1djWqZv7ZeQu60v4mGniO6PMBqrg9nMQPOJ2B56uBS3hiBvz3CGIp1acCAcUlc07OWAGp5R7P8AYKFgAc3XUZQhauppeHgFQWPNdVGycKd3cK0CtEdvODdr4mRt9OqTjZUozRUcM84Qju4pVnq5a6+zOECnKyri0pPqggeYMZfpWYaRWn/MAl4FRn5I9+dQH5Gv1F+bVRZY3+IPKkkEVq1ANEucsFYopaJtaamKnqSpa74jEvuVN7I1W6lAWvuUZU3MC9hnuHZpXUzoFQ/1AhyhXqJC0KF9eI8XLoRdrE4IRw6Aqold1r0S+7vEeJqwZX6jkcRAUN9w1uegXLYFyLlECvTLjvc0RP4wCy1ThcFTCRVaNMqO4fPgfMc23WWc+pVFkbDuuIeC0gGPUbdqk4OnzOEtlERpYx5ZEMprju/5hxBB7RU3zzKM1dUUVDwFfUIsXIKfFxShOEX3HgsAjqOygKjoPUSQFNlRLTuWlxZNw2VrEdqo8xGWDmmEw8jEOSjt72XjKDg+2CRRwuAsDoPc1YYka1tl9QCnib2pW8jEdm11BYmmualVg7CIdge41shtxFim1L/Hl8ypVLIgoLoeJgUq6C4ICOKRR2PIiEtii1adMR8D0WcEOrQtcwyhXbDudeYy2pdKnU6lI/qPmTABbJx7qA8zJuGEcCdaknU0FlfmO0bvLBKuygPbBY9+IJECFfBev4iZmD0ogeg6HTLvyL6Nq/G3Mv0oVU8VUaaIK+2nTKj0krWJehaHHEuD8BdeJmsXBgBb9RKwpCWU14j/AAIS89w2QlEqlish7Bl0u4XjEs0LB+o5OAD6JdeXzBaCx6gVgAU8yrWKS02leGWF0VcVBBQAwpliEDE0uiAs6TeoBOQFFdK7MViheXFvIDYjHDVFeqhBnKBA0WOfMxfGPnYaz7L4lfVYbKWBdc3HoAt2LEy11WQmKFDW2HxTW83vYzCBUqSrgD94K/3LYgWpOVmqvfxcGzG11/cTDfWARnzWUOBDFF5XEEtID74gBbBlQOsWGOOYBfqUaFGhoJXwOFRX5hrRjRPGXkBIXW+bjmj9Q3zpThq8wxYaI8EVNkXkMO3s8RnzQ1ywaZW1bEJE7Q3jZCp3LemXzmeIVVnxKIiD87KFrql9sb6wWHm4SrvQH+YssdJHeNG4yoF7BLXFh3AIHKt6mDc2FDuG95Jk3+/EpXz+uIFhxsyY5kFWm/MJtwDlnl4D3EOgVZgt6/mVYvrb4gR1Ef6QkGHh6+LgvwI2PuJ8onMvUTPQ/wBhKLGoHsIenJfUtRU6MUa6C4IBS0lYfTBGoLqkpF/Qxt2DlWywbV9sutW0v9CMC3MB0a3ctBJK3BAYmJbVuD3pi7Fyw7JSnuFjUdNwaGeoUHb9waRC+UvLSeYb6DqXN3sYsHSGOTHjrq4ZVVdM3FHSpVyBRfIsAojLKmLQQu4WGMsnMTuxNPMTuOEJWPOTxGsMvn6lcOq2DdlA8nmBTXIxIteYuQodRxVYQ5mBEnOOmK1RmjLfMqC5dZaoC5eW+VfJeR3Sq8BCilvLUxBgcly3UQUxhRcoR5/5i3APlS7sWHwypdIRo1tRUtHIyyRYbxEK7h7V3FZsHhAxpdKvUPFLfCAdRq+nIj3J7jlxj5WcTh7eCK5lvDDW99RQPE+LgyDaGMlC3lNXOKk8COrSriHhW52dBytPcrMANS3Yg2UXZDq4IOSMVtpxKDwJUA60qzhgEvS41z5jJYuBWNS1bEy8IWkpOZYla7cPE1epQqY9y1ybaX1Ec1FPEt8AgsziB6HOZVpja4PaOYHAXiHVFAEKH1olpVDhiJtDx7lk1Ar5iARse/8AgEnS38R6agX7QAU5lYnJId3bfmWvo4h+LjwjahfFIbM7Lu/U25xtLCD4LzmlHoLQobpCGY0b5hBg3G7HgmShXyzB6K5+YoU1DnVs8SpBx2OqN6IgFR9wJCiOeajCWx4yKMiW/PqXCK9nEaMG0lAgTghsmwdfmKiJp+ql6TR9I7Sheqiwzwl8hQZhGwtD8wZEPpKHFM06hpQReZKKl8b6QwlzRTCV0PIubW8NS4ATN9y42mpx+44jjgMv9xVRPHEZGBUB5WPqQlehTM/4UXmF/wD7ioqF68Q0pSp3FpKlvIVk3JOrIRU/NGILPXURWPI9yzklyZlHZBB5lidNVB3p+Y2XyRpcajHpMDK/2TEIVzbxBmVPlkFue7O0MEBaLSXg7KoUypQ8CooNUoiKK3A0ewe4ifASuIxE6xYkcg4ln3RXxDi0838QxzVdypxbzBUKt0iSXlyyHRWviJdaHnzDK7WzVRW3q4ozsvrHRcvzccLKu3Egr0mQw9TR4yrNC3I9kMmgO5rmvxBpUgOmDaKizmDic3wwymNH5m4q8zder5lUQjLXy+YA3WtX6h7laPEeAug7lsdYV3D2rkMMqhaOdMWGlSjzSh/Ev9yH6SrKiuTiVDgfyjCED2OZWQaN9zml6ekPhe93cBb4pOokTZVuoaAAjSjC9uWRXMIchYDkjqbdnFQYel0lMNkBNHuMdt6+KisqoLWuncs0Y3cssYGRxwjzM3k4nBOpekQGFk7QgNXqzaaWXDw2UVDiXhBUwelPAXehubMSeXbSv1ExOUIEK/Uw2pZh8kvzz/wBFivw1+6IqOSPy3LNDjmVK1YdS5/BLkVXmXwODnUoA8eqzrfijT3Cq1gLCelg8EPqE1jYNI8tll+Ilw24KseYjsXAsLbtgjbARCoR3fMOFosfMEAFADmb5rtWDkNAHJL1hlXzcRQRjuVeVdEYeDxDARXI+I9JZkYu4Jum4LoGhITYyxr3HeanDLg1cR8wxLtgHqDq3dIZYqplrnuVMxX8RMF0TzDyOq8MidHHhKZeB6lNlNABBg7bnPFiNY3+CZcmtWU0II7z1bVKn6gFBTnZ/kL0Gg8juRa5lfDKv0fmoVlYFPzGJYFys+XuG0tHxCBfb6melSL+4xQN7AjPtqy8DJwlKoOWHMs4sAh8wBj0RlWbrzMi7bghYXNHQ4GZhUWZivaEMsYKS8QapGoMp5lhICxr1DV2Wn5jDs8KSjLpWEavpzDWrZcuSjTOPcMPK7b0lD1KszRcenECsKNKIAmL1mVSo3d5UrRZwj2B7qH4g4Ja7JIWxBWw3Q8y1NEKOeIILdolUWSk6QLPrDgncwgkTxHzkUiUkWniWzgFrCbKuz9qq93xF63WTodQu3qPYBO/KC/zELtfqK/j1DTA+4it4fEoULMfmWoNIUPUTisi2u5ZEfS8x3VPHF/E8mfctqhT83C6Cjn4m9GdytJVdvcGyMCunBUDm6lJkweY6Mnl6l1cfbqEbFYNfgZUfldVUpFOjh4hp8P3D0sIE7RcsEXhstQFgDa8QkEwde4bg20IUQAHPcry3lZNFaS1/wBhqTDZ7giiicVFucHKIKyjydQQcLoMiLTTh6lsobbUyyGrBkDbVAtth4EGm79QYasfSLpAeuAVZ3xLVHFsVryMu7O6IeEilwsHZDH+I+GyxpHoOsiGvGGuNYybyA8uOIlwbOA7wr5go+48Q8XRbVQjcZd62fE86TywLa1GfEQqlhDRbUTficSFbwNvkiBHyEDPBJdwzaz5pb3whdnc5k+I7vXECq69EVIFEdYy63A2l8XDVESjKnaq9jpiWOS79wLE1GAc4YK2tUzMq2NY7HobY1CFuAxEorsI7vctHZMZiHpVRk0PwYQitulxzDTXHxL0TtL6vULCJh3KDojxCKWijWnIxAjHNVf+xEeZfjqJQtpfETN8NRYzHzFQoBysEhdD2bGbYzoi7LBFS5yzEYFlD91H5DFuBL4EEyiE7tVaU+YbkZVwjIoPmFaFUYDDRw5KiscaL6jlJPfTAYLqOG4ez8uWUamKGDNWVu4xLKoA8yt8fEwBz3EVAZUJiQXovQBAKt6LjUWKQ2MsUC6GuYgGjeHmEkVWpLbLxsrZQ3DqhrvoihZZRRkA/tSiw3SK1CrYccy4WMre593TCN6iJgaAi/2nAgv8Nn1LgoleBiv/AIZi68ywsJ8oRFERoh1A8xMSTgsL/UKCpG25s83yQQGHhLi1GfrOnS8CL7qtuwTDQHG1AmvbsKo+K4jVY8xk0+LPLHzCcK+yEhYgDwhKLC6uAEXvseS0u2kowDVki3SN7b+oFHGkNkwpHfJHimfdRTh92QuqDlVAPuW8ZVmPq4+SLqiPuUj/AAhPRNQIFeMnIJjCAENnzOSRaL7lylwirbtqyY1RhJAuyLiSlJky2FQGrwO4ACkIJcRbfYS3qgs7PcIC6gQal2jQMHj8XNLlRWGX7g4Ihor/AJZn/wCF/UPIEocoVH+g/Ma6fuo7IelxOovEWUhOMiJRPaM4BkIIh4j6Mm6isefuP5F+ZZYX5dx06V4SFeqCcziNz4nov6jiYxioI/wME/aA0i6jjAKQjpN3YFfGTyhMP+pWATxf0wtaj6R1faNfzBqn5SD8EtU9hWeLWIztW7X8y6VVfEQprd/ECtAV7sZQEjIRZlMUEFv1ABo/cODfYx6Qd2+I/l9lRorPxOd04bhi1VAvSpRvqiwvEQTAKObL5+GOHQVhR3TXEaT3fKf1EOFY8g8TipfuoI13RbMVn3E/VbDpuNX05tlAcxR0DDicxpOJn3BvFTzUQ7+UCMFnqFoYH7isNxlVqpmESvULZ+VSpg+oh4yFRiXFRd9syAb25ivNdURtWVU557iK5yWL5p9wQKaPEEP8xX7+GNtu2aNI6FIyb4dlbg1uMI+VwwADwwVsKXagy8TF5i5Kg0V1HmUFW9xgUVpUDpyib54yXYpVAvupgVyvyIuZFl+0D5APKl1x+PzBWxMOZ/sGD4P+OLForVoLCDjX3hbR+JUFVeHcsArfcH0gLwIlWc+ICwgx9QMsEqCDkfMWiGzAqBut8y7qI6fpHhNHipvLvcchKeJVcKmnGxULcU4mCA1XMK4GHJ+EziINDbuZKmB8wSrAXkR+KfcxdemNgNu4xxQXcp72LUolbcrqUoqa9MZfbPCYAC00nSovEtTs3eZZPQqNNb/sUjaBLXhsIKV4qkcj5REd8UYQf7lD/C1zRWiPGc/uc4hXZlS1+I93Z/wJzTUUmAeQg5fP1ChdXLeqYb9xNCvuVVSq7lT3M3I9RhdK9wqmj5gHRh6lOzYDWkR0/iKVqUsFUV7YcRP3cyD92FoD9SmVpcrIl9ZkLZL1S9EaaUXHbHiyKM7GRux5IZYseZUByuCLNsTb6hYTjBNUDyTLnkDZVZLg7l+YkqkW+YXYA6quVT8ghFZTBwRKcDRFBo/xF11Qujl/zCIMa9OqjonPMHdFyq2SwLIPAZ7lQsOOYp4DHGGTDAItLUu5gFkcKS2CGEIXmH8odXMMbVztXYWu5C9N0dy9BFX4jFv0l05uCEYNaaVcssVvyRK67AUZUI5rYlqHKUL2QsXL0wEbXuMANjFXKPVQh6e6hhWADJeANldzuU7uVCAvFrYXJ9M5TDeFhFbpdK3ML5Jc/FRPUNgp5lsqyr3Sysk6Cr5wlSBKoqmATi5Xoln+QL4h4pWuKllEKQo42BrIB0v3Bp69Qg3KFjuAFIUk2aTWU5LP7isjkHEbX3Q3k/4IJux6d4+ZVR/UVPMttQAH1Hl8xU0w5fafUAK9zEVKEgjY+I/kiF1/ErKVeRooJV7KcsV6laYHqBA0tyDQFvOtTySNNvuGe9zTbKYmKCz/APMLxk0+XET9jm5qBHgpK0/9letWR3cc/wBQ6Z9p9Xcq02b7/wCFIBfiGObGuYGHiOOaQH5TkoS33ACryD/7mHuXZ/sRHIrYra4HxCPIKuIS8XUtZVEo5YlSoC1eMaerg7XEVOFx5Z8oq1aiu8U5Gjd8bDOmnuGDf1LPpLisOsVqnCqgdrVnJMlWFwaEUvIhEDUBAonGEhwXi3P4icF1DiqF/EG2REfUyKH5RNeO/ueyu0rYH7SoLElQi2V+f5Sh6qXAlflRKaKv4mV48WI3NG9mIC6vcUmzvsvK2+D5l2HgApTYkcDqCK4lGEfm4c0yyP4RX4Yw3KkXFgnsWUKNH3FyIjzk105iUVwJQQftBS0w9kV96vcYU2t58R33sZU5mZNH2zx2xLCqU+EULbDiU2GumVF1rJ4kFvmHP/BFWEvAawSXkr6wV+IYfldwXYS6OWH/AO8VOJKLxen5m7wn8ijv/UZ0XdojXvxL87F055j4objTPu1UEXecUX6YZrrQf1HbG8jSN6mPGqgUDPff6lg7pdYsEgxvmWYU/wDnPBBeamcR4dxN7imh5HhjArDPlQZmwO355lGMpSw42Vkqt7m091CaYdxlw6eIdnYgdWmy6w4u0tYDdogmTHcqOgapye4KKAnRNkhzD3R6hYle0inQ3QSp1L0q4fmKEE9hGj8rLgGLbtrsYW4IdGD/ADBWAUFfBc4MKmPjiImO1GMpcwGZrQetO4n6oAgwhEwq8HqnDNaq6fmEk6Uc5DRqUWLCC/h0MPzAo3YHH6hYjY6IxhaTuMQTAR9NTywWnjuYLYinqUci3CmMts+3oyxAoaJhjqW95SwkGjTAFVd+4WzA8Mah7lU8QEbqWnygTXf/AAXS0Q5+EPJUUr2bL8zdVtejiGAk/h6j2tFVxVoNHiH2j5IJ+U70qB3gSwL6h9DZaeoIgEqI50B8RVWlfwxVZXtnZ+SaUBFHw7MP8weCos9QQ7hr8QUFyoBCledQOPKGmlhS/iLl4zl+fuC8e7jBh1lrDI3c9O3W7F5Bwpz3L5X8pG6r5jUWy7T4YW7oB2CBRtE5dgFurV6qEu8ymodMWXZxsxlSvE2rYPJKHXFxbTgpCuAEX3EpfP8AwKrqqgv2OYxWcR+iM+NQLPEp4H5A6n8bTQE25d2CKLHk8y+pBLE4hitRnhHch5KmrlEAeF25RIFpMEQRxegCI9AZxDtimz9rgltVrN7VN7aCfuMB/ciYfWf4hMoSx6dIyQC8/C/i0pjOc9xp7zbXQsAYIe73/lHwEYU+DiPUqlLl+S6W+Ip4KjVVWH3cqKNfFAX7Il7iMAzL1Qa/mAmFMr2lxjgyBfZKgU3AaqVXPM5PUWm6mTbnFZA2mhaZQzgxfpneXK03H7Q6lkORNlReV3G379QjQeMiAlFVUpZqgDp4i2+WXhTvEGIPJMvGy8EDh8Smy4XYHlCBBU5ISDvQZrpAs6RN7ORTXuXj6hircnzKqVRV2eYjb1aHW5F+bSibnZxSD/cBdShXbyxLrQ1yb2EVL4IfywRoJg4oa+rmGhQHxUY9ZF7A1/EPEmjquIppVP5jilFvmA3s3s5Av2xVXGu1t59NykXp24PzKLU08w+eS4bvPcqTFOJny/Ex0aeYoPcu4trLnwSEhXeJ4YFWNmBeyppWVdHUuN2N9zmvIQ2zLIeGmMsWWIxQHNwuECFhdV/METC0q0H5hBaSNAdyxJbc5nz1MVKsgqG7mHzxccHl34jMapXox/qHg11+qjAFrbjDlYrUM0C7g9PmK9GRrbN+ix/EMS4tsgVNB3TdQ9xMNjrghOImhS/ghZJEs4LcZQojT1p+wSzwVvN8fzDjQtAFvdk/iKsj6loY5avNE9Q8seGiA+TSP8JhPHRmiB6S9K/MtWFEfiURko6QX/INGmB5ByfqIXMuAll/EVETeWGZvAcKuc5xzsESuHO5cjIqrFq4L+oaVS/Jw/hIeBNncAex/wDJV6QKMuYpVMprK0fMpmvT1434hjuY5v2xVUTuwYb5X6ghUpwi1C1H8To6sy6wCqp3L7AD9L8EZc0wqoQBGi6x/aXDCfltV/ct13l7llg7q5eC67IGuTIfQLhFEy4KHefMcMtDwXj5NhnyxwygXKxJBrdnkD9xD1knONytQqHhAV+bIBdvy+HIxei+h2N2lqdVY/awthko7aJ62e+QeZqmv4lkE7PtQn8sfoVekaH8Rzuie2xQrNdkt0i/3zo9EG0qoNy257slrt16gxgacouojtgmlDSMsscYXEgRT3m3oL/qWQJdKqV/UBVawfOw8VJGJTIFXCfU1sdBSTA8eI2i1fuCy+Ui7xAucJAdtxCZKhp2wly5y5QJqavxAdpi9QMTzGHOudqN9p5h4QVqyt8RWDatzy0cbxFtkE6Mh6FOI8wqkTabVEttV6XecyyKa2F6/wBJvCkL149XKBiFC7/8JUaYA8vEfiB4sHbg9Psci+TeFv8A6hL7RVLqoap0HHEG8Rle3bP2wuQD841v4nIYDzIxZYp6TRsUbQiZf3KAE7nBGD3RO9hF9F1/UMEGPBFQn4DShBAMh0uQZ/UN02oq0V9sAKh4a4OEvBMTJxTL+WdIxlqmJoq2cX5/UT16BhXEK8fAAG/uFAqd7srK6JAdbDxCQBUkuDU5E/TI/QTT6iFRkIJbOahAu9h78xJTXWwFNMtg06fMbbekVVU1DjkOIynbYbB2kfmHJlL6+dibN7V/qNVvdHu5XWdLkglhdmvpjUqxS4O4+HhxDQuEYKP7MaPe/wCoI+mzi+0/0u2rqA0LEbmGKYfuI0pDSfMMaKd+nX8wHZHQe9j+MtfbFtxAnI3LX0y4tcxDEUvP/P/Z"}], "issuer": "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU", "hash": "7zPfFwyXGZKYMmLTzVynFGVEwLDfj48a5E3EA2RWtifq", "signature": "CP4p0+Nby/Fs7exp/mlLnOAu8iMJwyLRW6UHkZXZX8q8WbqxHYBRo2uzpcFAT0zEYSig7j3HqYdeoA3MpU1BCQ=="} diff --git a/duniter4j-es-subscription/LICENSE.txt b/duniter4j-es-subscription/LICENSE.txt deleted file mode 100644 index 94a9ed024d3859793618152ea559a168bbcbb5e2..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/LICENSE.txt +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - <one line to give the program's name and a brief idea of what it does.> - Copyright (C) <year> <name of author> - - 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/>. - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - <program> Copyright (C) <year> <name of author> - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -<http://www.gnu.org/licenses/>. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -<http://www.gnu.org/philosophy/why-not-lgpl.html>. diff --git a/duniter4j-es-subscription/pom.xml b/duniter4j-es-subscription/pom.xml deleted file mode 100644 index 7ec70c5ec4957fd43e0518ecdfc311f42ec1f15f..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/pom.xml +++ /dev/null @@ -1,190 +0,0 @@ -<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> - <modelVersion>4.0.0</modelVersion> - - <parent> - <groupId>org.duniter</groupId> - <artifactId>duniter4j</artifactId> - <version>1.0.4-SNAPSHOT</version> - </parent> - - <artifactId>duniter4j-es-subscription</artifactId> - <packaging>jar</packaging> - <name>Duniter4j :: ElasticSearch Subscription plugin</name> - <description>A ElasticSearch plugin that can send email notifications to end users</description> - - <properties> - - <!-- i18n configuration --> - <i18n.bundleOutputName>duniter4j-es-subscription-i18n</i18n.bundleOutputName> - <i18n.generateCsvFile>true</i18n.generateCsvFile> - <i18n.bundleCsvFile> - ${maven.gen.dir}/resources/META-INF/${i18n.bundleOutputName}.csv - </i18n.bundleCsvFile> - <config.i18nBundleName>${i18n.bundleOutputName}</config.i18nBundleName> - </properties> - - <dependencies> - <dependency> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-core</artifactId> - <version>${project.version}</version> - </dependency> - <dependency> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-user</artifactId> - <version>${project.version}</version> - </dependency> - - <!-- Elastic Search --> - <dependency> - <groupId>org.elasticsearch</groupId> - <artifactId>elasticsearch</artifactId> - <scope>provided</scope> - </dependency> - - <dependency> - <groupId>org.antlr</groupId> - <artifactId>stringtemplate</artifactId> - <version>${stringtemplate.version}</version> - <scope>provided</scope> - </dependency> - - <!-- Unit test --> - <dependency> - <groupId>junit</groupId> - <artifactId>junit</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>org.slf4j</groupId> - <artifactId>slf4j-api</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>org.slf4j</groupId> - <artifactId>slf4j-log4j12</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>log4j</groupId> - <artifactId>log4j</artifactId> - <optional>true</optional> - <scope>test</scope> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna-platform</artifactId> - <scope>test</scope> - <exclusions> - <exclusion> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - </exclusion> - </exclusions> - </dependency> - <dependency> - <groupId>org.elasticsearch.plugin</groupId> - <artifactId>mapper-attachments</artifactId> - <version>${elasticsearch.version}</version> - <scope>test</scope> - </dependency> - - <!-- Websocket (need for test) --> - <dependency> - <groupId>javax.websocket</groupId> - <artifactId>javax.websocket-api</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-server</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>org.glassfish.tyrus</groupId> - <artifactId>tyrus-container-grizzly-server</artifactId> - <scope>test</scope> - </dependency> - </dependencies> - - <build> - <resources> - <resource> - <directory>src/main/filtered-resources</directory> - <filtering>true</filtering> - <includes> - <include>*.config</include> - <include>**/*.properties</include> - </includes> - </resource> - <resource> - <directory>src/main/resources</directory> - <filtering>false</filtering> - </resource> - </resources> - - <plugins> - <plugin> - <groupId>org.nuiton.i18n</groupId> - <artifactId>i18n-maven-plugin</artifactId> - - <executions> - <execution> - <id>scan-sources</id> - <configuration> - <entries> - <entry> - <specificGoal>parserValidation</specificGoal> - <basedir>${maven.src.dir}/main/java/</basedir> - <includes> - <param>**/**-validation.xml</param> - </includes> - </entry> - </entries> - </configuration> - <goals> - <goal>parserJava</goal> - <goal>parserValidation</goal> - <goal>gen</goal> - </goals> - </execution> - <execution> - <id>make-bundle</id> - <goals> - <goal>bundle</goal> - </goals> - </execution> - </executions> - </plugin> - - <plugin> - <artifactId>maven-assembly-plugin</artifactId> - <executions> - <execution> - <id>assembly-plugin</id> - <phase>package</phase> - <goals> - <goal>single</goal> - </goals> - <configuration> - <attach>true</attach> - <appendAssemblyId>false</appendAssemblyId> - <finalName>${project.artifactId}-${project.version}</finalName> - <descriptors> - <descriptor> - ${basedir}/src/main/assembly/plugin.xml - </descriptor> - </descriptors> - <skipAssembly>${assembly.skip}</skipAssembly> - </configuration> - </execution> - </executions> - </plugin> - </plugins> - </build> -</project> diff --git a/duniter4j-es-subscription/src/license/THIRD-PARTY.properties b/duniter4j-es-subscription/src/license/THIRD-PARTY.properties deleted file mode 100644 index acee82b8c584abca4af608a2dd8a9357a79943a7..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/license/THIRD-PARTY.properties +++ /dev/null @@ -1,37 +0,0 @@ -# Generated by org.codehaus.mojo.license.AddThirdPartyMojo -#------------------------------------------------------------------------------- -# Already used licenses in project : -# - ASL, version 2 -# - Apache 2.0 -# - Apache License 2.0 -# - Apache License Version 2.0 -# - BSD License -# - BSD licence -# - Bouncy Castle Licence -# - CC0 1.0 Universal -# - CDDL -# - CDDL+GPL -# - COMMON DEVELOPMENT AND DISTRIBUTION LICENSE (CDDL) Version 1.0 -# - Common Development and Distribution License (CDDL) v1.0 -# - Dual license consisting of the CDDL v1.1 and GPL v2 -# - Eclipse Public License 1.0 -# - GPLv2+CE -# - General Public License (GPL) v3 -# - Indiana University Extreme! Lab Software License, vesion 1.1.1 -# - LGPL, version 2.1 -# - Lesser General Public License (LGPL) v 3.0 -# - Lesser General Public License (LPGL) -# - Lesser General Public License (LPGL) v 2.1 -# - Lesser General Public License (LPGL) version 3.0 -# - MIT License -# - Mozilla Public License 1.1 (MPL 1.1) -# - New BSD License -# - Public Domain, per Creative Commons CC0 -# - The Apache Software License, Version 2.0 -#------------------------------------------------------------------------------- -# Please fill the missing licenses for dependencies : -# -# -#Tue Nov 21 18:04:41 CET 2017 -com.googlecode.juniversalchardet--juniversalchardet--1.0.3=Mozilla Public License 1.1 (MPL 1.1) -org.antlr--antlr-runtime--3.3=BSD License diff --git a/duniter4j-es-subscription/src/main/assembly/plugin.xml b/duniter4j-es-subscription/src/main/assembly/plugin.xml deleted file mode 100644 index 004bbfa77a7e12db6bae7fc242ffbaec788f73d0..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/assembly/plugin.xml +++ /dev/null @@ -1,44 +0,0 @@ -<?xml version="1.0"?> -<assembly> - <id>plugin</id> - - - <formats> - <format>zip</format> - </formats> - - <includeBaseDirectory>false</includeBaseDirectory> - - <dependencySets> - <dependencySet> - <outputDirectory>/</outputDirectory> - <useProjectArtifact>true</useProjectArtifact> - <useTransitiveFiltering>true</useTransitiveFiltering> - <excludes> - <exclude>org.duniter:duniter4j-es-core</exclude> - <exclude>org.duniter:duniter4j-es-user</exclude> - <exclude>org.elasticsearch:elasticsearch</exclude> - <exclude>net.java.dev.jna:jna</exclude> - <exclude>com.fasterxml.jackson.core:jackson-core</exclude> - <exclude>log4j:log4j</exclude> - </excludes> - </dependencySet> - </dependencySets> - - <fileSets> - <fileSet> - <includes> - <include>LICENSE</include> - </includes> - </fileSet> - - <fileSet> - <directory>target/classes</directory> - <outputDirectory/> - <includes> - <include>plugin-descriptor.properties</include> - <include>plugin-security.policy</include> - </includes> - </fileSet> - </fileSets> -</assembly> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/filtered-resources/log4j.properties b/duniter4j-es-subscription/src/main/filtered-resources/log4j.properties deleted file mode 100644 index 7b6667b1facc361ed8b1993869f728e2c01f1799..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/filtered-resources/log4j.properties +++ /dev/null @@ -1,32 +0,0 @@ - -# Global logging configuration -log4j.rootLogger=ERROR, stdout, file -#log4j.rootLogger=ERROR, stdout - -# Console output -log4j.appender.stdout=org.apache.log4j.ConsoleAppender -log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p %m%n - -# Duniter4j levels -log4j.logger.org.duniter=INFO -#log4j.logger.org.duniter.core.client=DEBUG -#log4j.logger.org.duniter.core.client.service=DEBUG -log4j.logger.org.duniter.elasticsearch=DEBUG - -# Other frameworks levels -log4j.logger.org.nuiton.util=WARN -log4j.logger.org.nuiton.config=WARN -log4j.logger.org.nuiton.converter=WARN -log4j.logger.org.nuiton.i18n=ERROR -log4j.logger.org.elasticsearch=WARN -#log4j.logger.org.elasticsearch=INFO - -log4j.appender.file=org.apache.log4j.RollingFileAppender -log4j.appender.file.file=${duniter4j.log.file} -log4j.appender.file.MaxFileSize=10MB -log4j.appender.file.MaxBackupIndex=4 - -log4j.appender.file.layout=org.apache.log4j.PatternLayout -log4j.appender.file.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m%n - diff --git a/duniter4j-es-subscription/src/main/filtered-resources/plugin-descriptor.properties b/duniter4j-es-subscription/src/main/filtered-resources/plugin-descriptor.properties deleted file mode 100644 index 93ed139ad0a6f61620c3d7e45a81f88c0960cb46..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/filtered-resources/plugin-descriptor.properties +++ /dev/null @@ -1,9 +0,0 @@ -name=subscription -description=Plugin for Gchange API -version=${project.version} -site=false -jvm=true -classname=org.duniter.elasticsearch.subscription.Plugin -java.version=1.8 -elasticsearch.version=2.4.6 -isolated=false diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/Plugin.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/Plugin.java deleted file mode 100644 index d53caf2c56362b20fb64f3ccad6ba7a97d533220..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/Plugin.java +++ /dev/null @@ -1,90 +0,0 @@ -package org.duniter.elasticsearch.subscription; - -/* - * #%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 com.google.common.collect.Lists; -import org.duniter.elasticsearch.subscription.dao.DaoModule; -import org.duniter.elasticsearch.subscription.rest.RestModule; -import org.duniter.elasticsearch.subscription.service.ServiceModule; -import org.duniter.elasticsearch.subscription.synchro.SynchroModule; -import org.elasticsearch.common.component.LifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Module; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; - -import java.util.Collection; - -public class Plugin extends org.elasticsearch.plugins.Plugin { - - private ESLogger logger ; - - private boolean enable; - - @Inject public Plugin(Settings settings) { - this.enable = settings.getAsBoolean("subscription.enabled", true); - this.logger = Loggers.getLogger(Plugin.class.getName(), settings, new String[0]); - } - - @Override - public String name() { - return "duniter4j-es-subscription"; - } - - @Override - public String description() { - return "Duniter Subscription Plugin"; - } - - @Override - public Collection<Module> nodeModules() { - Collection<Module> modules = Lists.newArrayList(); - if (!enable) { - logger .warn(description() + " has been disabled."); - return modules; - } - modules.add(new DaoModule()); - modules.add(new ServiceModule()); - modules.add(new RestModule()); - modules.add(new SynchroModule()); - - return modules; - } - - @Override - public Collection<Class<? extends LifecycleComponent>> nodeServices() { - Collection<Class<? extends LifecycleComponent>> components = Lists.newArrayList(); - if (!enable) { - return components; - } - components.add(PluginSettings.class); - components.add(PluginInit.class); - return components; - } - - /* -- protected methods -- */ - - -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginInit.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginInit.java deleted file mode 100644 index b4f5c004e26b3d4f50c06283ecdc71167758c350..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginInit.java +++ /dev/null @@ -1,111 +0,0 @@ -package org.duniter.elasticsearch.subscription; - -/* - * #%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.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.service.SubscriptionService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; - -/** - * Created by blavenie on 17/06/16. - */ -public class PluginInit extends AbstractLifecycleComponent<PluginInit> { - - private final PluginSettings pluginSettings; - private final ThreadPool threadPool; - private final Injector injector; - private final ESLogger logger; - - @Inject - public PluginInit(Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) { - super(settings); - this.logger = Loggers.getLogger("duniter.subscription", settings, new String[0]); - this.pluginSettings = pluginSettings; - this.threadPool = threadPool; - this.injector = injector; - } - - @Override - protected void doStart() { - threadPool.scheduleOnClusterReady(() -> { - createIndices(); - - // Waiting cluster back to GREEN or YELLOW state, before synchronizePeer - threadPool.scheduleOnClusterReady(this::doAfterStart); - }); - } - - @Override - protected void doStop() { - - } - - @Override - protected void doClose() { - - } - - protected void createIndices() { - - boolean reloadIndices = pluginSettings.reloadIndices(); - - if (reloadIndices) { - if (logger.isInfoEnabled()) { - logger.info("Reloading indices..."); - } - injector.getInstance(SubscriptionIndexDao.class) - .deleteIndex() - .createIndexIfNotExists(); - - if (logger.isInfoEnabled()) { - logger.info("Reloading indices [OK]"); - } - } - else { - if (logger.isDebugEnabled()) { - logger.debug("Checking indices..."); - } - injector.getInstance(SubscriptionIndexDao.class).createIndexIfNotExists(); - - if (logger.isDebugEnabled()) { - logger.debug("Checking indices [OK]"); - } - } - } - - protected void doAfterStart() { - - // Start subscription services - if (pluginSettings.enableSubscription()) { - injector.getInstance(SubscriptionService.class) - .startScheduling(); - } - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java deleted file mode 100644 index 2a1d1eb2c6bf855df948e64a5b8d4500039733dd..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java +++ /dev/null @@ -1,222 +0,0 @@ -package org.duniter.elasticsearch.subscription; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.util.crypto.KeyPair; -import org.elasticsearch.common.component.*; -import org.elasticsearch.common.inject.Inject; - -/** - * Access to configuration options - * @author Benoit Lavenier <benoit.lavenier@e-is.pro> - * @since 1.0 - */ -public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { - - private org.duniter.elasticsearch.user.PluginSettings delegate; - - private static PluginSettings instance; - - public static final PluginSettings instance() { - return instance; - } - - @Inject - public PluginSettings(org.elasticsearch.common.settings.Settings settings, - org.duniter.elasticsearch.user.PluginSettings delegate) { - super(settings); - this.delegate = delegate; - - // Add i18n bundle name - delegate.addI18nBundleName(getI18nBundleName()); - - instance = this; - } - - @Override - protected void doStart() { - - } - - @Override - protected void doStop() { - - } - - @Override - protected void doClose() { - - } - - public org.duniter.elasticsearch.user.PluginSettings getDelegate() { - return delegate; - } - - - public boolean enableSubscription() { - return settings.getAsBoolean("duniter.subscription.enable", Boolean.TRUE); - } - - public String getCesiumUrl() { - return this.settings.get("duniter.subscription.email.cesium.url", "https://g1.duniter.fr"); - } - - /** - * Should email subscription be send at startup ? - * @return - */ - public boolean isEmailSubscriptionsExecuteAtStartup() { - return settings.getAsBoolean("duniter.subscription.email.atStartup", false); - } - - /** - * Should email subscription execute as DEBUG mode ? - * @return - */ - public boolean isEmailSubscriptionsDebug() { - return settings.getAsBoolean("duniter.subscription.email.debug", false); - } - - - /** - * Day of the week to trigger weekly email subscription (default: 2 = monday) - * @return - */ - public int getEmailSubscriptionsExecuteDayOfWeek() { - return settings.getAsInt("duniter.subscription.email.dayOfWeek", 2); - } - - /** - * Hour in day to trigger daily email subscription (default: 3 AM) - * @return - */ - public int getEmailSubscriptionsExecuteHour() { - return settings.getAsInt("duniter.subscription.email.hourOfDay", 3); - } - - /* -- delegate methods -- */ - - - public boolean reloadIndices() { - return delegate.reloadAllIndices(); - } - - public boolean enableSynchro() { - return delegate.enableSynchro(); - } - - public int getSynchroTimeOffset() { - return delegate.getSynchroTimeOffset(); - } - - public boolean getMailEnable() { - return delegate.getMailEnable(); - } - - public String getMailSmtpHost() { - return delegate.getMailSmtpHost(); - } - - public int getMailSmtpPort() { - return delegate.getMailSmtpPort(); - } - - public String getMailSmtpUsername() { - return delegate.getMailSmtpUsername(); - } - - public String getMailSmtpPassword() { - return delegate.getMailSmtpPassword(); - } - - public String getMailAdmin() { - return delegate.getMailAdmin(); - } - - public String getMailFrom() { - return delegate.getMailFrom(); - } - - public String getMailSubjectPrefix() { - return delegate.getMailSubjectPrefix(); - } - - public String getClusterName() { - return delegate.getClusterName(); - } - - public String getNodeBmaHost() { - return delegate.getNodeBmaHost(); - } - - public int getNodeBmaPort() { - return delegate.getNodeBmaPort(); - } - - public int getIndexBulkSize() { - return delegate.getIndexBulkSize(); - } - - public boolean enableBlockchainSync() { - return delegate.enableBlockchainSync(); - } - - public String getKeyringSalt() { - return delegate.getKeyringSalt(); - } - - public String getKeyringPassword() { - return delegate.getKeyringPassword(); - } - - public String getKeyringPublicKey() { - return delegate.getKeyringPublicKey(); - } - - public String getKeyringSecretKey() { - return delegate.getKeyringSecretKey(); - } - - public String getDefaultStringAnalyzer() { - return delegate.getDefaultStringAnalyzer(); - } - - public KeyPair getNodeKeypair() { - return delegate.getNodeKeypair(); - } - - public boolean isRandomNodeKeypair() { - return delegate.isRandomNodeKeypair(); - } - - public String getNodePubkey() { - return delegate.getNodePubkey(); - } - - /* -- protected methods -- */ - - protected String getI18nBundleName() { - return "duniter4j-es-subscription-i18n"; - } -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/AbstractSubscriptionIndexTypeDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/AbstractSubscriptionIndexTypeDao.java deleted file mode 100644 index 525f7f0e049e5a816017d5235fb76fa18b4d440a..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/AbstractSubscriptionIndexTypeDao.java +++ /dev/null @@ -1,63 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.local.LocalEntity; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.ObjectUtils; -import org.duniter.elasticsearch.subscription.PluginSettings; -import org.elasticsearch.search.SearchHit; - -import java.io.IOException; -import java.util.function.Function; -import java.util.function.Predicate; - -/** - * Created by Benoit on 30/03/2015. - */ -public abstract class AbstractSubscriptionIndexTypeDao<T extends AbstractSubscriptionIndexTypeDao> extends org.duniter.elasticsearch.dao.AbstractIndexTypeDao<T> implements SubscriptionIndexTypeDao<T> { - - protected PluginSettings pluginSettings; - - public AbstractSubscriptionIndexTypeDao(String index, String type, PluginSettings pluginSettings) { - super(index, type); - this.pluginSettings = pluginSettings; - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public void checkSameDocumentIssuer(String id, String expectedIssuer) { - String issuer = getMandatoryFieldsById(id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); - if (!ObjectUtils.equals(expectedIssuer, issuer)) { - throw new TechnicalException("Not same issuer"); - } - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java deleted file mode 100644 index bbf8795883ac9e43da55c6c31ba4a870cb666d1d..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java +++ /dev/null @@ -1,44 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao; - -/* - * #%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.subscription.dao.execution.SubscriptionExecutionDao; -import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDaoImpl; -import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; -import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDaoImpl; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class DaoModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // Subscription index - bind(SubscriptionIndexDao.class).to(SubscriptionIndexDaoImpl.class).asEagerSingleton(); - - // Subscription types - bind(SubscriptionRecordDao.class).to(SubscriptionRecordDaoImpl.class).asEagerSingleton(); - bind(SubscriptionExecutionDao.class).to(SubscriptionExecutionDaoImpl.class).asEagerSingleton(); - } - -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDao.java deleted file mode 100644 index ecdcded3b0de9d28963c769f15e3abf6043a2234..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDao.java +++ /dev/null @@ -1,36 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao; - -/*- - * #%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.elasticsearch.dao.IndexDao; -import org.duniter.elasticsearch.dao.IndexTypeDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface SubscriptionIndexDao extends IndexDao<SubscriptionIndexDao> { - String INDEX = "subscription"; - String CATEGORY_TYPE = "category"; - - SubscriptionIndexDao register(IndexTypeDao<?> indexTypeDao); -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDaoImpl.java deleted file mode 100644 index b93ec9437bc19b40432727690edc14cb2bed3ad7..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDaoImpl.java +++ /dev/null @@ -1,122 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao; - -/*- - * #%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 com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexDao; -import org.duniter.elasticsearch.dao.IndexTypeDao; -import org.duniter.elasticsearch.dao.handler.AddSequenceAttributeHandler; -import org.duniter.elasticsearch.subscription.PluginSettings; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; -import java.util.ArrayList; -import java.util.List; - -/** - * Created by blavenie on 03/04/17. - */ -public class SubscriptionIndexDaoImpl extends AbstractIndexDao<SubscriptionIndexDao> implements SubscriptionIndexDao { - - - private static final String CATEGORIES_BULK_CLASSPATH_FILE = "subscription-categories-bulk-insert.json"; - - private PluginSettings pluginSettings; - private List<IndexTypeDao<?>> indexTypeDaos = new ArrayList<>(); - - @Inject - public SubscriptionIndexDaoImpl(PluginSettings pluginSettings) { - super(SubscriptionIndexDao.INDEX); - - this.pluginSettings = pluginSettings; - } - - public SubscriptionIndexDao register(IndexTypeDao<?> indexTypeDao) { - indexTypeDaos.add(indexTypeDao); - return this; - } - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", INDEX)); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - indexTypeDaos.forEach(indexTypeDao -> createIndexRequestBuilder.addMapping(indexTypeDao.getType(), indexTypeDao.createTypeMapping())); - createIndexRequestBuilder.addMapping(CATEGORY_TYPE, createCategoryTypeMapping()); - createIndexRequestBuilder.execute().actionGet(); - - // Fill categories - fillCategories(); - } - - - - protected XContentBuilder createCategoryTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(CATEGORY_TYPE) - .startObject("properties") - - // name - .startObject("name") - .field("type", "string") - .field("analyzer", pluginSettings.getDefaultStringAnalyzer()) - .endObject() - - // parent - .startObject("parent") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), CATEGORY_TYPE, ioe.getMessage()), ioe); - } - } - - protected void fillCategories() { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s/%s] Fill data", getIndex(), CATEGORY_TYPE)); - } - - // Insert categories - client.bulkFromClasspathFile(CATEGORIES_BULK_CLASSPATH_FILE, getIndex(), CATEGORY_TYPE, - // Add order attribute - new AddSequenceAttributeHandler("order", "\\{.*\"name\".*\\}", 1)); - } -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java deleted file mode 100644 index 512ded888e5d842ed7aa0d52cad17d5fa3792d54..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java +++ /dev/null @@ -1,43 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.elasticsearch.dao.IndexTypeDao; - -/** - * Created by Benoit on 30/03/2015. - */ -public interface SubscriptionIndexTypeDao<T extends SubscriptionIndexTypeDao> extends IndexTypeDao<T> { - - String create(final String json); - - void create(final String json, boolean wait); - - void update(final String id, final String json); - - void update(final String id, final String json, boolean wait); - - void checkSameDocumentIssuer(String id, String expectedIssuer); - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java deleted file mode 100644 index 2ea19c36c847303570e74abb91aa3a45b2f817c3..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDao.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao.execution; - -/*- - * #%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.elasticsearch.subscription.dao.SubscriptionIndexTypeDao; -import org.duniter.elasticsearch.subscription.model.SubscriptionExecution; -import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; - -import java.util.List; - -/** - * Created by blavenie on 03/04/17. - */ -public interface SubscriptionExecutionDao<T extends SubscriptionIndexTypeDao> extends SubscriptionIndexTypeDao<T> { - - String TYPE = "execution"; - - SubscriptionExecution getLastExecution(SubscriptionRecord record); - - SubscriptionExecution getLastExecution(String recipient, String subscriptionType, String recordId); - - Long getLastExecutionTime(String recipient, String subscriptionType, String recordId); - - Long getLastExecutionTime(SubscriptionRecord record); -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java deleted file mode 100644 index 52802c017df98bf81e7f3920f7ecb7a8e4c9f1c7..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/execution/SubscriptionExecutionDaoImpl.java +++ /dev/null @@ -1,190 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao.execution; - -/*- - * #%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 com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.subscription.PluginSettings; -import org.duniter.elasticsearch.subscription.dao.AbstractSubscriptionIndexTypeDao; -import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.model.SubscriptionExecution; -import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; -import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; -import org.elasticsearch.action.index.IndexRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.action.update.UpdateRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.SearchHit; -import org.elasticsearch.search.sort.SortOrder; - -import java.io.IOException; -import java.util.Arrays; -import java.util.List; -import java.util.Objects; -import java.util.stream.Collectors; - -/** - * Created by blavenie on 03/04/17. - */ -public class SubscriptionExecutionDaoImpl extends AbstractSubscriptionIndexTypeDao<SubscriptionExecutionDaoImpl> implements SubscriptionExecutionDao<SubscriptionExecutionDaoImpl> { - - @Inject - public SubscriptionExecutionDaoImpl(PluginSettings pluginSettings, SubscriptionIndexDao indexDao) { - super(SubscriptionIndexDao.INDEX, TYPE, pluginSettings); - - indexDao.register(this); - } - - @Override - public SubscriptionExecution getLastExecution(SubscriptionRecord record) { - Preconditions.checkNotNull(record); - Preconditions.checkNotNull(record.getIssuer()); - Preconditions.checkNotNull(record.getType()); - Preconditions.checkNotNull(record.getId()); - - return getLastExecution(record.getIssuer(), record.getType(), record.getId()); - } - - @Override - public SubscriptionExecution getLastExecution(String recipient, String recordType, String recordId) { - - BoolQueryBuilder query = QueryBuilders.boolQuery() - .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECIPIENT, recipient)) - .must(QueryBuilders.termsQuery(SubscriptionExecution.PROPERTY_RECORD_TYPE, recordType)) - .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECORD_ID, recordId)); - - SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) - .setTypes(SubscriptionExecutionDao.TYPE) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) - .setQuery(query) - .setFetchSource(true) - .setFrom(0).setSize(1) - .addSort(SubscriptionExecution.PROPERTY_TIME, SortOrder.DESC) - .get(); - - if (response.getHits().getTotalHits() == 0) return null; - - SearchHit hit = response.getHits().getHits()[0]; - return client.readSourceOrNull(hit, SubscriptionExecution.class); - } - - @Override - public Long getLastExecutionTime(SubscriptionRecord record) { - Preconditions.checkNotNull(record); - Preconditions.checkNotNull(record.getIssuer()); - Preconditions.checkNotNull(record.getType()); - Preconditions.checkNotNull(record.getId()); - - return getLastExecutionTime(record.getIssuer(), record.getType(), record.getId()); - } - - @Override - public Long getLastExecutionTime(String recipient, String recordType, String recordId) { - - BoolQueryBuilder query = QueryBuilders.boolQuery() - .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECIPIENT, recipient)) - .must(QueryBuilders.termQuery(SubscriptionExecution.PROPERTY_RECORD_ID, recordId)) - .must(QueryBuilders.termsQuery(SubscriptionExecution.PROPERTY_RECORD_ID, recordType)); - - SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) - .setTypes(SubscriptionExecutionDao.TYPE) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) - .setQuery(query) - .addField(SubscriptionExecution.PROPERTY_TIME) - .setFrom(0).setSize(1) - .addSort(SubscriptionExecution.PROPERTY_TIME, SortOrder.DESC) - .get(); - - if (response.getHits().getTotalHits() == 0) return null; - SearchHit hit = response.getHits().getHits()[0]; - return hit.field(SubscriptionExecution.PROPERTY_TIME).getValue(); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(getType()) - .startObject("properties") - - // issuer - .startObject(SubscriptionExecution.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // recipient - .startObject(SubscriptionExecution.PROPERTY_RECIPIENT) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // record type - .startObject(SubscriptionExecution.PROPERTY_RECORD_TYPE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // record id - .startObject(SubscriptionExecution.PROPERTY_RECORD_ID) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // time - .startObject(SubscriptionExecution.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - // hash - .startObject(SubscriptionExecution.PROPERTY_HASH) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // signature - .startObject(SubscriptionExecution.PROPERTY_SIGNATURE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java deleted file mode 100644 index 6b414bfed93287aad3f73472e756fd8b0522d428..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao.record; - -/*- - * #%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.elasticsearch.subscription.dao.SubscriptionIndexTypeDao; -import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; - -import java.util.List; - -/** - * Created by blavenie on 03/04/17. - */ -public interface SubscriptionRecordDao<T extends SubscriptionIndexTypeDao> extends SubscriptionIndexTypeDao<T> { - - String TYPE = "record"; - - List<SubscriptionRecord> getSubscriptions(int from, int size, String recipient, String... types); -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java deleted file mode 100644 index 7ad0fe863fab6aabff575f8a94e74db422a52d22..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java +++ /dev/null @@ -1,173 +0,0 @@ -package org.duniter.elasticsearch.subscription.dao.record; - -/*- - * #%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.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.CollectionUtils; -import org.duniter.elasticsearch.subscription.PluginSettings; -import org.duniter.elasticsearch.subscription.dao.AbstractSubscriptionIndexTypeDao; -import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; -import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.SearchHit; - -import java.io.IOException; -import java.util.Arrays; -import java.util.List; -import java.util.Objects; -import java.util.stream.Collectors; - -/** - * Created by blavenie on 03/04/17. - */ -public class SubscriptionRecordDaoImpl extends AbstractSubscriptionIndexTypeDao<SubscriptionRecordDaoImpl> implements SubscriptionRecordDao<SubscriptionRecordDaoImpl> { - - @Inject - public SubscriptionRecordDaoImpl(PluginSettings pluginSettings, SubscriptionIndexDao indexDao) { - super(SubscriptionIndexDao.INDEX, TYPE, pluginSettings); - - indexDao.register(this); - } - - @Override - public List<SubscriptionRecord> getSubscriptions(int from, int size, String recipient, String... types) { - - BoolQueryBuilder query = QueryBuilders.boolQuery() - .must(QueryBuilders.termQuery(SubscriptionRecord.PROPERTY_RECIPIENT, recipient)); - if (CollectionUtils.isNotEmpty(types)) { - query.must(QueryBuilders.termsQuery(SubscriptionRecord.PROPERTY_TYPE, types)); - } - - SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) - .setTypes(SubscriptionRecordDao.TYPE) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) - .setQuery(query) - .setFetchSource(true) - .setFrom(from).setSize(size) - .get(); - - return Arrays.asList(response.getHits().getHits()).stream() - .map(this::toSubscription) - .filter(Objects::nonNull) - .collect(Collectors.toList()); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(getType()) - .startObject("properties") - - // version - .startObject(SubscriptionRecord.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // type - .startObject(SubscriptionRecord.PROPERTY_TYPE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // issuer - .startObject(SubscriptionRecord.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // recipient - .startObject(SubscriptionRecord.PROPERTY_RECIPIENT) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // time - .startObject(SubscriptionRecord.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - // nonce - .startObject(SubscriptionRecord.PROPERTY_NONCE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // issuerContent - .startObject(SubscriptionRecord.PROPERTY_ISSUER_CONTENT) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // receiver content - .startObject(SubscriptionRecord.PROPERTY_RECIPIENT_CONTENT) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // hash - .startObject(SubscriptionRecord.PROPERTY_HASH) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // signature - .startObject(SubscriptionRecord.PROPERTY_SIGNATURE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } - - protected SubscriptionRecord toSubscription(SearchHit searchHit) { - - SubscriptionRecord record = null; - - if (SubscriptionRecordDao.TYPE.equals(searchHit.getType())) { - record = client.readSourceOrNull(searchHit, EmailSubscription.class); - } - - if (record != null) { - record.setId(searchHit.getId()); - } - - return record; - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Protocol.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Protocol.java deleted file mode 100644 index ea9d484afe92a2e81ed9c2c32cde18a6730b9d90..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Protocol.java +++ /dev/null @@ -1,33 +0,0 @@ -package org.duniter.elasticsearch.subscription.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% - */ - -/** - * Created by blavenie on 31/03/16. - */ -public interface Protocol { - - String VERSION = "1"; - - String EMAIL_API = "EMAIL_API"; -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java deleted file mode 100644 index 63cd57f4298e2e8b15170666a0accfc5b3e90421..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionExecution.java +++ /dev/null @@ -1,78 +0,0 @@ -package org.duniter.elasticsearch.subscription.model; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.annotation.JsonIgnore; -import org.duniter.core.client.model.elasticsearch.Record; - -/** - * Created by blavenie on 01/12/16. - */ -public class SubscriptionExecution extends Record { - - public static final String PROPERTY_RECIPIENT = "recipient"; - - public static final String PROPERTY_RECORD_TYPE = "recordType"; - - public static final String PROPERTY_RECORD_ID = "recordId"; - - private String recipient; - private String recordType; - private String recordId; - - private SubscriptionRecord record; - - public String getRecipient() { - return recipient; - } - - public void setRecipient(String recipient) { - this.recipient = recipient; - } - - public String getRecordType() { - return recordType; - } - - public void setRecordType(String recordType) { - this.recordType = recordType; - } - - public String getRecordId() { - return recordId; - } - - public void setRecordId(String recordId) { - this.recordId = recordId; - } - - @JsonIgnore - public SubscriptionRecord getRecord() { - return record; - } - - @JsonIgnore - public void setRecord(SubscriptionRecord record) { - this.record = record; - } -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionRecord.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionRecord.java deleted file mode 100644 index 6a5081cc9edb6ff254eb2e5dbd9bc5335cdd4733..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/SubscriptionRecord.java +++ /dev/null @@ -1,101 +0,0 @@ -package org.duniter.elasticsearch.subscription.model; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.annotation.JsonIgnore; -import org.duniter.core.client.model.elasticsearch.Record; - -/** - * Created by blavenie on 01/12/16. - */ -public class SubscriptionRecord<T> extends Record{ - - public static final String PROPERTY_RECIPIENT = "recipient"; - - public static final String PROPERTY_NONCE = "nonce"; - - public static final String PROPERTY_RECIPIENT_CONTENT = "recipientContent"; - - public static final String PROPERTY_ISSUER_CONTENT = "issuerContent"; - - public static final String PROPERTY_CONTENT = "content"; - - public static final String PROPERTY_TYPE = "type"; - - private String recipient; - private String nonce; - private String recipientContent; - private String issuerContent; - private String type; - private T content; - - public String getRecipient() { - return recipient; - } - - public void setRecipient(String recipient) { - this.recipient = recipient; - } - - public String getNonce() { - return nonce; - } - - public void setNonce(String nonce) { - this.nonce = nonce; - } - - public String getRecipientContent() { - return recipientContent; - } - - public void setRecipientContent(String recipientContent) { - this.recipientContent = recipientContent; - } - - public String getIssuerContent() { - return issuerContent; - } - - public void setIssuerContent(String issuerContent) { - this.issuerContent = issuerContent; - } - - public String getType() { - return type; - } - - public void setType(String type) { - this.type = type; - } - - @JsonIgnore - public T getContent() { - return content; - } - - @JsonIgnore - public void setContent(T content) { - this.content = content; - } -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java deleted file mode 100644 index 781c7778e0329dd320c98186e57358c0bf247549..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java +++ /dev/null @@ -1,98 +0,0 @@ -package org.duniter.elasticsearch.subscription.model.email; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; - -/** - * Created by blavenie on 01/12/16. - */ -public class EmailSubscription extends SubscriptionRecord<EmailSubscription.Content> { - - public static final String TYPE = "email"; - - public static Content newContent() { - return new EmailSubscription.Content(); - } - - public enum Frequency { - daily, - weekly - } - - public static class Content { - - public static final String PROPERTY_EMAIL = "email"; - public static final String PROPERTY_FREQUENCY = "frequency"; - public static final String PROPERTY_LOCALE = "locale"; - public static final String PROPERTY_INCLUDES = "includes"; - public static final String PROPERTY_EXCLUDES = "excludes"; - - private String email; - private String[] includes; - private String[] excludes; - private String locale; - private Frequency frequency; - - public String getEmail() { - return email; - } - - public void setEmail(String email) { - this.email = email; - } - - public String[] getIncludes() { - return includes; - } - - public void setIncludes(String[] includes) { - this.includes = includes; - } - - public String[] getExcludes() { - return excludes; - } - - public void setExcludes(String[] excludes) { - this.excludes = excludes; - } - - public String getLocale() { - return locale; - } - - public void setLocale(String locale) { - this.locale = locale; - } - - public Frequency getFrequency() { - return frequency; - } - - public void setFrequency(Frequency frequency) { - this.frequency = frequency; - } - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/RestModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/RestModule.java deleted file mode 100644 index 3c9c969efbe9dddb73d87216a3d45d1d245d7a63..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/RestModule.java +++ /dev/null @@ -1,47 +0,0 @@ -package org.duniter.elasticsearch.subscription.rest; - -/* - * #%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.subscription.rest.execution.RestSubscriptionExecutionGetAction; -import org.duniter.elasticsearch.subscription.rest.record.RestSubscriptionCategoryGetAction; -import org.duniter.elasticsearch.subscription.rest.record.RestSubscriptionRecordIndexAction; -import org.duniter.elasticsearch.subscription.rest.record.RestSubscriptionRecordUpdateAction; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class RestModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // Subscription category - bind(RestSubscriptionCategoryGetAction.class).asEagerSingleton(); - - // Subscription execution - bind(RestSubscriptionExecutionGetAction.class).asEagerSingleton(); - - // Subscription record - bind(RestSubscriptionRecordIndexAction.class).asEagerSingleton(); - bind(RestSubscriptionRecordUpdateAction.class).asEagerSingleton(); - - } -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/execution/RestSubscriptionExecutionGetAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/execution/RestSubscriptionExecutionGetAction.java deleted file mode 100644 index f0f5771be6a1c4a07e14eac1041d014383528a87..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/execution/RestSubscriptionExecutionGetAction.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.duniter.elasticsearch.subscription.rest.execution; - -/* - * #%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.security.RestSecurityController; -import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDao; -import org.elasticsearch.common.inject.Inject; - -public class RestSubscriptionExecutionGetAction { - - @Inject - public RestSubscriptionExecutionGetAction(RestSecurityController securityController) { - // Add security rule to enable access on /subscription/execution - // only on search POST request (need by synchro) - securityController.allowPostSearchIndexType(SubscriptionIndexDao.INDEX, SubscriptionExecutionDao.TYPE); - } - -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionCategoryGetAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionCategoryGetAction.java deleted file mode 100644 index 010c12cec297d6f93ddb1dc08ff88f58f075dfb0..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionCategoryGetAction.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.subscription.rest.record; - -/* - * #%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.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -public class RestSubscriptionCategoryGetAction { - - @Inject - public RestSubscriptionCategoryGetAction(RestSecurityController securityController) { - // Add security rule for category - securityController.allowIndexType(RestRequest.Method.GET, SubscriptionIndexDao.INDEX, SubscriptionIndexDao.CATEGORY_TYPE); - } - -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordIndexAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordIndexAction.java deleted file mode 100644 index 8a4bb1a6037d68a6bd6809f9b2f7752a5aab593e..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.subscription.rest.record; - -/* - * #%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.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.subscription.service.SubscriptionService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestSubscriptionRecordIndexAction extends AbstractRestPostIndexAction { - - - @Inject - public RestSubscriptionRecordIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - SubscriptionService service) { - super(settings, controller, client, securityController, - SubscriptionIndexDao.INDEX, SubscriptionRecordDao.TYPE, - json -> service.create(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordUpdateAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordUpdateAction.java deleted file mode 100644 index 2a2659578808e5cababc1446d6ef75e9855b36d0..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.subscription.rest.record; - -/* - * #%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.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; -import org.duniter.elasticsearch.subscription.service.SubscriptionService; -import org.duniter.elasticsearch.rest.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestSubscriptionRecordUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestSubscriptionRecordUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - SubscriptionService service) { - super(settings, controller, client, securityController, - SubscriptionIndexDao.INDEX, SubscriptionRecordDao.TYPE, - (id, json) -> service.update(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/AbstractService.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/AbstractService.java deleted file mode 100644 index a900a2b283f8d414186542b8a876a1f2a4343baa..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/AbstractService.java +++ /dev/null @@ -1,53 +0,0 @@ -package org.duniter.elasticsearch.subscription.service; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.subscription.PluginSettings; - -/** - * Created by blavenie on 10/01/17. - */ -public abstract class AbstractService extends org.duniter.elasticsearch.user.service.AbstractService { - - protected PluginSettings pluginSettings; - - public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings) { - this(loggerName, client, pluginSettings, null); - } - - public AbstractService(Duniter4jClient client, PluginSettings pluginSettings) { - this(client, pluginSettings, null); - } - - public AbstractService(Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - this("duniter.subscription", client, pluginSettings, cryptoService); - } - - public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - super(loggerName, client, pluginSettings.getDelegate(), cryptoService); - this.pluginSettings = pluginSettings; - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/ServiceModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/ServiceModule.java deleted file mode 100644 index 72c4223c016d15db4720c8fd9f00af2028a65434..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/ServiceModule.java +++ /dev/null @@ -1,34 +0,0 @@ -package org.duniter.elasticsearch.subscription.service; - -/* - * #%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.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class ServiceModule extends AbstractModule implements Module { - - @Override protected void configure() { - // Subscription services - bind(SubscriptionService.class).asEagerSingleton(); - } -} \ No newline at end of file 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 deleted file mode 100644 index 5f93f052166cc1abf1767bf27afacf380c409d42..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java +++ /dev/null @@ -1,455 +0,0 @@ -package org.duniter.elasticsearch.subscription.service; - -/* - * #%L - * Duniter4j :: 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 com.google.common.collect.ImmutableSet; -import org.duniter.core.client.model.ModelUtils; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.DateUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.core.util.crypto.CryptoUtils; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.subscription.PluginSettings; -import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDao; -import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; -import org.duniter.elasticsearch.subscription.model.SubscriptionExecution; -import org.duniter.elasticsearch.subscription.model.SubscriptionRecord; -import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; -import org.duniter.elasticsearch.subscription.util.stringtemplate.DateRenderer; -import org.duniter.elasticsearch.subscription.util.stringtemplate.StringRenderer; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.service.AdminService; -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; -import org.stringtemplate.v4.ST; -import org.stringtemplate.v4.STGroup; -import org.stringtemplate.v4.STGroupDir; - -import java.text.SimpleDateFormat; -import java.util.*; -import java.util.concurrent.TimeUnit; -import java.util.stream.Collectors; - -/** - * Created by Benoit on 30/03/2015. - */ -public class SubscriptionService extends AbstractService { - - private SubscriptionRecordDao subscriptionRecordDao; - private SubscriptionExecutionDao subscriptionExecutionDao; - private ThreadPool threadPool; - private MailService mailService; - private AdminService adminService; - private UserEventService userEventService; - private UserService userService; - private String emailSubjectPrefix; - private STGroup templates; - - @Inject - public SubscriptionService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - SubscriptionRecordDao subscriptionRecordDao, - SubscriptionExecutionDao subscriptionExecutionDao, - ThreadPool threadPool, - MailService mailService, - AdminService adminService, - UserService userService, - UserEventService userEventService) { - super("duniter.subscription", client, settings, cryptoService); - this.subscriptionRecordDao = subscriptionRecordDao; - this.subscriptionExecutionDao = subscriptionExecutionDao; - this.threadPool = threadPool; - this.mailService = mailService; - this.adminService = adminService; - this.userService = userService; - this.userEventService = userEventService; - this.emailSubjectPrefix = pluginSettings.getMailSubjectPrefix().trim(); - 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) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a subscription from issuer [%s]", issuer.substring(0, 8))); - } - - return subscriptionRecordDao.create(json); - } - - public void update(String id, String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check same document issuer - subscriptionRecordDao.checkSameDocumentIssuer(id, issuer); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Updating subscription [%s] from issuer [%s]", id, issuer.substring(0, 8))); - } - - subscriptionRecordDao.update(id, json); - } - - public SubscriptionService startScheduling() { - if (!pluginSettings.getMailEnable()) { - logger.warn(I18n.t("duniter4j.es.subscription.error.mailDisabling")); - return this; - } - - // for DEBUG only: execute a fake job every minute, to test scheduler - if (logger.isDebugEnabled()) { - threadPool.scheduleAtFixedRate( - () -> logger.debug("Scheduled fake task successfully executed - scheduled every [1 min]"), - 20 * 1000 /* startScheduling in 20s */, - 60 * 1000 /* every 1 min */, - TimeUnit.MILLISECONDS); - } - - // Email subscriptions - { - if (logger.isInfoEnabled()) { - Calendar cal = new GregorianCalendar(); - cal.setTimeInMillis(0); - cal.set(Calendar.DAY_OF_WEEK, pluginSettings.getEmailSubscriptionsExecuteDayOfWeek()); - String dayOfWeek = new SimpleDateFormat("EEE").format(cal.getTime()); - logger.warn(I18n.t("duniter4j.es.subscription.email.start", pluginSettings.getEmailSubscriptionsExecuteHour(), dayOfWeek)); - } - - // Execution at startup (or DEBUG mode) - if (pluginSettings.isEmailSubscriptionsExecuteAtStartup() || pluginSettings.isEmailSubscriptionsDebug()) { - threadPool.schedule( - () -> executeEmailSubscriptions(EmailSubscription.Frequency.daily), - new TimeValue(20, TimeUnit.SECONDS) /* after 20s */ - ); - } - - // Daily execution - threadPool.scheduleAtFixedRate( - () -> executeEmailSubscriptions(EmailSubscription.Frequency.daily), - DateUtils.delayBeforeHour(pluginSettings.getEmailSubscriptionsExecuteHour()), - DateUtils.DAY_DURATION_IN_MILLIS, - TimeUnit.MILLISECONDS); - - // Weekly execution - threadPool.scheduleAtFixedRate( - () -> executeEmailSubscriptions(EmailSubscription.Frequency.weekly), - DateUtils.delayBeforeDayAndHour(pluginSettings.getEmailSubscriptionsExecuteDayOfWeek(), pluginSettings.getEmailSubscriptionsExecuteHour()), - 7 * DateUtils.DAY_DURATION_IN_MILLIS, - TimeUnit.MILLISECONDS); - } - return this; - } - - public void executeEmailSubscriptions(final EmailSubscription.Frequency frequency) { - - long now = System.currentTimeMillis(); - logger.info(String.format("Executing %s email subscription...", frequency.name())); - - final String senderPubkey = pluginSettings.getNodePubkey(); - - int from = 0; - int size = 10; - boolean hasMore = true; - long executionCount=0; - while (hasMore) { - List<SubscriptionRecord> subscriptions = subscriptionRecordDao.getSubscriptions(from, size, senderPubkey, EmailSubscription.TYPE); - - // Get profiles titles, for issuers and the sender - Set<String> issuers = subscriptions.stream() - .map(SubscriptionRecord::getIssuer) - .distinct() - .collect(Collectors.toSet()); - final Map<String, String> profileTitles = userService.getProfileTitles( - ImmutableSet.<String>builder().addAll(issuers).add(senderPubkey).build()); - final String senderName = (profileTitles != null && profileTitles.containsKey(senderPubkey)) ? profileTitles.get(senderPubkey) : - ModelUtils.minifyPubkey(senderPubkey); - - executionCount += subscriptions.stream() - .map(record -> decryptEmailSubscription((EmailSubscription)record)) - .filter(record -> (record != null && record.getContent().getFrequency() == frequency)) - .map(record -> processEmailSubscription(record, senderPubkey, senderName, profileTitles)) - .filter(Objects::nonNull) - .map(this::saveExecution) - .count(); - - hasMore = CollectionUtils.size(subscriptions) >= size; - from += size; - } - - logger.info(String.format("Executing %s email subscription... [OK] emails sent [%s] (in %s ms)", - frequency.name(), executionCount, System.currentTimeMillis()-now)); - - } - - /* -- protected methods -- */ - - protected EmailSubscription decryptEmailSubscription(EmailSubscription subscription) { - Preconditions.checkNotNull(subscription); - Preconditions.checkNotNull(subscription.getId()); - - if (StringUtils.isBlank(subscription.getRecipientContent()) || StringUtils.isBlank(subscription.getNonce()) || - StringUtils.isBlank(subscription.getIssuer())) { - logger.error(String.format("Invalid subscription [%s]. Missing field 'recipientContent', 'nonce' or 'issuer'.", subscription.getId())); - return null; - } - - String jsonContent; - try { - jsonContent = cryptoService.openBox(subscription.getRecipientContent(), - CryptoUtils.decodeBase58(subscription.getNonce()), - CryptoUtils.decodeBase58(subscription.getIssuer()), - pluginSettings.getNodeKeypair().getSecKey() - ); - } catch(Exception e) { - logger.error(String.format("Could not decrypt email subscription content for subscription [%s]", subscription.getId())); - return null; - } - - try { - EmailSubscription.Content content = getObjectMapper().readValue(jsonContent, EmailSubscription.Content.class); - subscription.setContent(content); - } catch(Exception e) { - logger.error(String.format("Could not parse email subscription content [%s]: %s", jsonContent, e.getMessage())); - return null; - } - - return subscription; - } - - protected SubscriptionExecution processEmailSubscription(final EmailSubscription subscription, - final String senderPubkey, - final String senderName, - final Map<String, String> profileTitles) { - Preconditions.checkNotNull(subscription); - - boolean debug = pluginSettings.isEmailSubscriptionsDebug(); - if (subscription.getContent() != null && subscription.getContent().getEmail() != null) { - if (debug) { - logger.info(String.format("Processing email subscription to [%s - %s] on account [%s]", - senderName, - subscription.getContent().getEmail(), - ModelUtils.minifyPubkey(subscription.getIssuer()))); - } - else { - logger.info(String.format("Processing email subscription [%s] on account [%s]", - subscription.getId(), - ModelUtils.minifyPubkey(subscription.getIssuer()))); - } - } - else { - logger.warn(String.format("Processing email subscription [%s] - no email found in subscription content: skipping", subscription.getId())); - return null; - } - - SubscriptionExecution lastExecution = subscriptionExecutionDao.getLastExecution(subscription); - Long lastExecutionTime; - - if (lastExecution != null) { - lastExecutionTime = lastExecution.getTime(); - } - // If first email execution: only send event from the last 7 days. - else { - Calendar defaultDateLimit = new GregorianCalendar(); - defaultDateLimit.setTimeInMillis(System.currentTimeMillis()); - defaultDateLimit.add(Calendar.DAY_OF_YEAR, - 7); - defaultDateLimit.set(Calendar.HOUR_OF_DAY, 0); - defaultDateLimit.set(Calendar.MINUTE, 0); - defaultDateLimit.set(Calendar.SECOND, 0); - defaultDateLimit.set(Calendar.MILLISECOND, 0); - lastExecutionTime = defaultDateLimit.getTimeInMillis() / 1000; - } - - // Get last user events - String[] includes = subscription.getContent() == null ? null : subscription.getContent().getIncludes(); - String[] excludes = subscription.getContent() == null ? null : subscription.getContent().getExcludes(); - List<UserEvent> userEvents = userEventService.getUserEvents(subscription.getIssuer(), lastExecutionTime, includes, excludes); - - if (CollectionUtils.isEmpty(userEvents)) return null; // no events: stop here - - // Get user locale - String[] localParts = subscription.getContent() != null && subscription.getContent().getLocale() != null ? - subscription.getContent().getLocale().split("-") : new String[]{"en", "GB"}; - Locale issuerLocale = localParts.length >= 2 ? new Locale(localParts[0].toLowerCase(), localParts[1].toUpperCase()) : new Locale(localParts[0].toLowerCase()); - - - - // Compute text content - final String text = fillTemplate( - templates.getInstanceOf("text_email"), - subscription, - senderPubkey, - senderName, - profileTitles, - userEvents, - pluginSettings.getCesiumUrl()) - .render(issuerLocale); - - // Compute HTML content - final String html = fillTemplate( - templates.getInstanceOf("html_email_content"), - subscription, - senderPubkey, - senderName, - profileTitles, - userEvents, - pluginSettings.getCesiumUrl()) - .render(issuerLocale); - - final String object = emailSubjectPrefix + I18n.t("duniter4j.es.subscription.email.subject", userEvents.size()); - if (pluginSettings.isEmailSubscriptionsDebug()) { - logger.info(String.format("---- Email to send (debug mode) ------\nTo:%s\nObject: %s\nText content:\n%s", - subscription.getContent().getEmail(), - object, - text)); - } - else { - // Schedule email sending - threadPool.schedule(() -> mailService.sendHtmlEmailWithText( - object, - text, - "<body>" + html + "</body>", - subscription.getContent().getEmail())); - } - - // Compute last time (should be the first one, as events are sorted in DESC order) - Long lastEventTime = userEvents.get(0).getTime(); - if (lastExecution == null) { - lastExecution = new SubscriptionExecution(); - lastExecution.setRecipient(subscription.getIssuer()); - lastExecution.setRecordType(subscription.getType()); - lastExecution.setRecordId(subscription.getId()); - } - lastExecution.setTime(lastEventTime); - - - return lastExecution; - } - - - public static ST fillTemplate(ST template, - EmailSubscription subscription, - String senderPubkey, - String senderName, - Map<String, String> issuerProfilNames, - List<UserEvent> userEvents, - String cesiumSiteUrl) { - String issuerName = issuerProfilNames != null && issuerProfilNames.containsKey(subscription.getIssuer()) ? - issuerProfilNames.get(subscription.getIssuer()) : - ModelUtils.minifyPubkey(subscription.getIssuer()); - - // Remove comma (to avoid to be used as many args in the i18n_args template) - issuerName = issuerName.replaceAll("[, ]+", " "); - senderName = StringUtils.isNotBlank(senderName) ? senderName.replaceAll("[, ]+", " ") : senderName; - - try { - // Compute body - template.add("url", cesiumSiteUrl); - template.add("issuerPubkey", subscription.getIssuer()); - template.add("issuerName", issuerName); - template.add("senderPubkey", senderPubkey); - template.add("senderName", senderName); - userEvents.forEach(userEvent -> { - String description = userEvent.getParams() != null ? - I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase(), userEvent.getParams()) : - I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase()); - template.addAggr("events.{description, time}", new Object[]{ - description, - new Date(userEvent.getTime() * 1000) - }); - }); - - return template; - - } - catch (Exception e) { - throw new TechnicalException(e); - } - } - - protected SubscriptionExecution saveExecution(SubscriptionExecution execution) { - Preconditions.checkNotNull(execution); - Preconditions.checkNotNull(execution.getRecipient()); - Preconditions.checkNotNull(execution.getRecordType()); - Preconditions.checkNotNull(execution.getRecordId()); - - // Update issuer - execution.setIssuer(pluginSettings.getNodePubkey()); - - // Fill hash + signature - String json = toJson(execution, true/*skip hash and signature*/); - execution.setHash(cryptoService.hash(json)); - execution.setSignature(cryptoService.sign(json, pluginSettings.getNodeKeypair().getSecKey())); - - if (execution.getId() == null) { - subscriptionExecutionDao.create(toJson(execution), false/*not wait*/); - } - else { - subscriptionExecutionDao.update(execution.getId(), toJson(execution), false/*not wait*/); - } - return execution; - } - - private String toJson(Record record) { - return toJson(record, false); - } - - private String toJson(Record record, boolean cleanHashAndSignature) { - Preconditions.checkNotNull(record); - try { - String json = getObjectMapper().writeValueAsString(record); - if (cleanHashAndSignature) { - json = PARSER_SIGNATURE.removeFromJson(json); - json = PARSER_HASH.removeFromJson(json); - } - return json; - } catch(JsonProcessingException e) { - throw new TechnicalException("Unable to serialize object " + record.getClass().getName(), e); - } - } - - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroModule.java deleted file mode 100644 index 2196d42f92bde1819b62b51e1236962cc78d0177..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroModule.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.subscription.synchro; - -/* - * #%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.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class SynchroModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // Subscription - bind(SynchroSubscriptionRecordAction.class).asEagerSingleton(); - bind(SynchroSubscriptionExecutionIndexAction.class).asEagerSingleton(); - - } - -} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroSubscriptionExecutionIndexAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroSubscriptionExecutionIndexAction.java deleted file mode 100644 index 389aca62002e0c20b2d75c7e898d0a39a7c735cd..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroSubscriptionExecutionIndexAction.java +++ /dev/null @@ -1,55 +0,0 @@ -package org.duniter.elasticsearch.subscription.synchro; - -/*- - * #%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.bma.EndpointApi; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.dao.execution.SubscriptionExecutionDao; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroSubscriptionExecutionIndexAction extends AbstractSynchroAction { - - @Inject - public SynchroSubscriptionExecutionIndexAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(SubscriptionIndexDao.INDEX, SubscriptionExecutionDao.TYPE, client, pluginSettings.getDelegate(), - cryptoService, threadPool); - - - synchroService.register(this); - } - - @Override - public EndpointApi getEndPointApi() { - return EndpointApi.ES_SUBSCRIPTION_API; - } -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroSubscriptionRecordAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroSubscriptionRecordAction.java deleted file mode 100644 index 20ba40e804218da68ff39c0a6b5cd3abe3f4d062..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/synchro/SynchroSubscriptionRecordAction.java +++ /dev/null @@ -1,56 +0,0 @@ -package org.duniter.elasticsearch.subscription.synchro; - -/*- - * #%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.bma.EndpointApi; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; -import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroSubscriptionRecordAction extends AbstractSynchroAction { - - @Inject - public SynchroSubscriptionRecordAction(Duniter4jClient client, - PluginSettings pluginSettings, - ThreadPool threadPool, - CryptoService cryptoService, - SynchroService synchroService) { - super(SubscriptionIndexDao.INDEX, SubscriptionRecordDao.TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - synchroService.register(this); - } - - @Override - public EndpointApi getEndPointApi() { - return EndpointApi.ES_SUBSCRIPTION_API; - } - -} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/DateRenderer.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/DateRenderer.java deleted file mode 100644 index 9fb695d17a0636af3a21bfc5d6fc1bb43bca2ad9..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/DateRenderer.java +++ /dev/null @@ -1,65 +0,0 @@ -package org.duniter.elasticsearch.subscription.util.stringtemplate; - -/*- - * #%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.*; - -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-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java deleted file mode 100644 index fdbda8bdad3ba14d55f3e2bc0dc31f4cb20e4288..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/StringRenderer.java +++ /dev/null @@ -1,54 +0,0 @@ -package org.duniter.elasticsearch.subscription.util.stringtemplate; - -/*- - * #%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.duniter.core.util.StringUtils; -import org.nuiton.i18n.I18n; -import org.stringtemplate.v4.AttributeRenderer; - -import java.util.Locale; - -/** - * Add format capabilities: i18n, pubkey - * Created by blavenie on 10/04/17. - */ -public class StringRenderer extends org.stringtemplate.v4.StringRenderer{ - - @Override - public String toString(Object o, String formatString, Locale locale) { - return formatString == null ? (String)o : - (formatString.equals("pubkey") ? ModelUtils.minifyPubkey((String)o) : - (formatString.startsWith("i18n") ? toI18nString(o, formatString, locale) : - super.toString(o, formatString, locale))); - } - - protected String toI18nString(Object key, String formatString, Locale locale) { - String[] params = formatString.startsWith("i18n:") ? formatString.substring(5).split(",") : null; - if (CollectionUtils.isNotEmpty(params)) { - return I18n.l(locale, key.toString(), params); - } - return I18n.l(locale, key.toString()); - } -} diff --git a/duniter4j-es-subscription/src/main/misc/index.sh b/duniter4j-es-subscription/src/main/misc/index.sh deleted file mode 100755 index 02b66934dd45e338506a7adc1e25500fabac9710..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/misc/index.sh +++ /dev/null @@ -1,14 +0,0 @@ -#!/bin/sh - -curl -X POST '192.168.0.28:9200/places' -d '{ - "mappings": { - "place": { - "properties": { - "id": {"type": "double"}, - "name": {"type": "string"}, - "type": {"type": "string"}, - "location": {"type": "geo_point"} - } - } - } -}' \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties deleted file mode 100644 index 3277f18a7ba82c3dca3b0711603f4c4197cd0492..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties +++ /dev/null @@ -1,15 +0,0 @@ -duniter4j.es.subscription.email.footer.disableHelp=You can disable this email notification service in the page %2$% (%1$s). -duniter4j.es.subscription.email.footer.sendBy=This email has sent you the Cesium+ node of %2$s (%1$s). -duniter4j.es.subscription.email.hello=Hello %s\! -duniter4j.es.subscription.email.html.footer.disableHelp=You can disable this email notification service in <a href\="%s">online services</a> page. -duniter4j.es.subscription.email.html.footer.sendBy=This email has sent you the Cesium+ node of <a href\="%s">%s</a>.. -duniter4j.es.subscription.email.html.hello=Hello <b>%s</b>\! -duniter4j.es.subscription.email.html.pubkey=Public key\: <a href\="%s">%s</a> -duniter4j.es.subscription.email.html.unreadCount=You received <b>%s new notifications</b>. -duniter4j.es.subscription.email.notificationsDivider=Notifications list\: -duniter4j.es.subscription.email.openCesium=Open Cesium+ -duniter4j.es.subscription.email.pubkey=Public key\: %2$s (%1$s) -duniter4j.es.subscription.email.start=Email subscriptions\: daily mailing [at %1$s\:00] and weekly [on %2$s at %1$s\:00] -duniter4j.es.subscription.email.subject=You received %s new notifications -duniter4j.es.subscription.email.unreadCount=You received %s new notifications. -duniter4j.es.subscription.error.mailDisabling=Unable to process email subscriptions\: Email sending is disabled in the configuration diff --git a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties deleted file mode 100644 index 28faba1c1aefaca12ba5fcd2c051d9d857535925..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties +++ /dev/null @@ -1,15 +0,0 @@ -duniter4j.es.subscription.email.footer.disableHelp=Vous pouvez désactiver ce service de notification par email, dans la rubrique "Services en ligne" de Cesium+ (%s). -duniter4j.es.subscription.email.footer.sendBy=Cet email vous a été envoyé le noeud Cesium+ de %2$s (%1$s). -duniter4j.es.subscription.email.hello=Bonjour %s \! -duniter4j.es.subscription.email.html.footer.disableHelp=Vous pouvez désactiver ce service de notification par email, dans <a href\="%s">la rubrique services en ligne</a> de Cesium+. -duniter4j.es.subscription.email.html.footer.sendBy=Cet email vous a été envoyé depuis le noeud Cesium+ de <a href\="%1$s">%2$s</a>. -duniter4j.es.subscription.email.html.hello=Bonjour <b>%s</b> \! -duniter4j.es.subscription.email.html.pubkey=Clé publique \: <a href\="%s">%s</a> -duniter4j.es.subscription.email.html.unreadCount=Vous avez <b>%s notifications</b> non lues. -duniter4j.es.subscription.email.notificationsDivider=Liste des notifications \: -duniter4j.es.subscription.email.openCesium=Ouvrir Cesium+ -duniter4j.es.subscription.email.pubkey=Clé publique \: %2$s (%1$s) -duniter4j.es.subscription.email.start=Abonnement email\: envoi quotidien [à %1$s\:00] et hebdomadaire [le %2$s à %1$s\:00] -duniter4j.es.subscription.email.subject=%s nouvelles notifications non lues -duniter4j.es.subscription.email.unreadCount=Vous avez %s notifications non lues. -duniter4j.es.subscription.error.mailDisabling=Impossible de traiter les abonnements email\: la fonction d'envoi d'email est désactivée dans la configuration diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/cesium_logo.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/cesium_logo.st deleted file mode 100644 index 83cf445c1c6d63f4dfb4612f81a06253aff3769b..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/cesium_logo.st +++ /dev/null @@ -1,6 +0,0 @@ -cesium_logo(url, data) ::= << -$if(data)$<img height="144" width="144" src="data:image/png;base64,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"> -$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/org/duniter/elasticsearch/subscription/templates/css.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css.st deleted file mode 100644 index 30a453a57d32d5e0454ee6fb2328e570ffb69a36..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css.st +++ /dev/null @@ -1,10738 +0,0 @@ -css() ::= << - html, body, div, span, applet, object, iframe, - h1, h2, h3, h4, h5, h6, p, blockquote, pre, - a, abbr, acronym, address, big, cite, code, - del, dfn, em, img, ins, kbd, q, s, samp, - small, strike, strong, sub, sup, tt, var, - b, i, u, center, - dl, dt, dd, ol, ul, li, - fieldset, form, label, legend, - table, caption, tbody, tfoot, thead, tr, th, td, - article, aside, canvas, details, embed, fieldset, - figure, figcaption, footer, header, hgroup, - menu, nav, output, ruby, section, summary, - time, mark, audio, video { - margin: 0; - padding: 0; - border: 0; - vertical-align: baseline; - font: inherit; - font-size: 100%; } - - /** - * Hide the `template` element in IE, Safari, and Firefox < 22. - */ - [hidden], - template { - display: none; } - - script { - display: none !important; } - - /* ========================================================================== - Base - ========================================================================== */ - /** - * 1. Set default font family to sans-serif. - * 2. Prevent iOS text size adjust after orientation change, without disabling - * user zoom. - */ - html { - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - font-family: sans-serif; - /* 1 */ - -webkit-text-size-adjust: 100%; - -ms-text-size-adjust: 100%; - /* 2 */ - -webkit-text-size-adjust: 100%; - /* 2 */ } - - /** - * Remove default margin. - */ - body { - margin: 0; - line-height: 1; } - - /** - * Remove default outlines. - */ - a, - button, - :focus, - a:focus, - button:focus, - a:active, - a:hover { - outline: 0; } - - /* * - * Remove tap highlight color - */ - a { - -webkit-user-drag: none; - -webkit-tap-highlight-color: transparent; - -webkit-tap-highlight-color: transparent; } - a[href]:hover { - cursor: pointer; } - - /* ========================================================================== - Typography - ========================================================================== */ - /** - * Address style set to `bolder` in Firefox 4+, Safari 5, and Chrome. - */ - b, - strong { - font-weight: bold; } - - /** - * Address styling not present in Safari 5 and Chrome. - */ - dfn { - font-style: italic; } - - /** - * Address differences between Firefox and other browsers. - */ - hr { - -moz-box-sizing: content-box; - box-sizing: content-box; - height: 0; } - - /** - * Correct font family set oddly in Safari 5 and Chrome. - */ - code, - kbd, - pre, - samp { - font-size: 1em; - font-family: monospace, serif; } - - /** - * Improve readability of pre-formatted text in all browsers. - */ - pre { - white-space: pre-wrap; } - - /** - * Set consistent quote types. - */ - q { - quotes: "\201C" "\201D" "\2018" "\2019"; } - - /** - * Address inconsistent and variable font size in all browsers. - */ - small { - font-size: 80%; } - - /** - * Prevent `sub` and `sup` affecting `line-height` in all browsers. - */ - sub, - sup { - position: relative; - vertical-align: baseline; - font-size: 75%; - line-height: 0; } - - sup { - top: -0.5em; } - - sub { - bottom: -0.25em; } - - /** - * Define consistent border, margin, and padding. - */ - fieldset { - margin: 0 2px; - padding: 0.35em 0.625em 0.75em; - border: 1px solid #c0c0c0; } - - /** - * 1. Correct `color` not being inherited in IE 8/9. - * 2. Remove padding so people aren't caught out if they zero out fieldsets. - */ - legend { - padding: 0; - /* 2 */ - border: 0; - /* 1 */ } - - /** - * 1. Correct font family not being inherited in all browsers. - * 2. Correct font size not being inherited in all browsers. - * 3. Address margins set differently in Firefox 4+, Safari 5, and Chrome. - * 4. Remove any default :focus styles - * 5. Make sure webkit font smoothing is being inherited - * 6. Remove default gradient in Android Firefox / FirefoxOS - */ - button, - input, - select, - textarea { - margin: 0; - /* 3 */ - font-size: 100%; - /* 2 */ - font-family: inherit; - /* 1 */ - outline-offset: 0; - /* 4 */ - outline-style: none; - /* 4 */ - outline-width: 0; - /* 4 */ - -webkit-font-smoothing: inherit; - /* 5 */ - background-image: none; - /* 6 */ } - - /** - * Address Firefox 4+ setting `line-height` on `input` using `importnt` in - * the UA stylesheet. - */ - button, - input { - line-height: normal; } - - /** - * Address inconsistent `text-transform` inheritance for `button` and `select`. - * All other form control elements do not inherit `text-transform` values. - * Correct `button` style inheritance in Chrome, Safari 5+, and IE 8+. - * Correct `select` style inheritance in Firefox 4+ and Opera. - */ - button, - select { - text-transform: none; } - - /** - * 1. Avoid the WebKit bug in Android 4.0.* where (2) destroys native `audio` - * and `video` controls. - * 2. Correct inability to style clickable `input` types in iOS. - * 3. Improve usability and consistency of cursor style between image-type - * `input` and others. - */ - button, - html input[type="button"], - input[type="reset"], - input[type="submit"] { - cursor: pointer; - /* 3 */ - -webkit-appearance: button; - /* 2 */ } - - /** - * Re-set default cursor for disabled elements. - */ - button[disabled], - html input[disabled] { - cursor: default; } - - /** - * Remove inner padding and border in Firefox 4+. - */ - button::-moz-focus-inner, - input::-moz-focus-inner { - padding: 0; - border: 0; } - - img { - -webkit-user-drag: none; } - - /** - * Scaffolding - * -------------------------------------------------- - */ - *, - *:before, - *:after { - -webkit-box-sizing: border-box; - -moz-box-sizing: border-box; - box-sizing: border-box; } - - html { - overflow: hidden; - -ms-touch-action: pan-y; - touch-action: pan-y; } - - body, - .ionic-body { - -webkit-touch-callout: none; - -webkit-font-smoothing: antialiased; - font-smoothing: antialiased; - -webkit-text-size-adjust: none; - -moz-text-size-adjust: none; - text-size-adjust: none; - -webkit-tap-highlight-color: transparent; - -webkit-tap-highlight-color: transparent; - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - top: 0; - right: 0; - bottom: 0; - left: 0; - overflow: hidden; - margin: 0; - padding: 0; - color: #000; - word-wrap: break-word; - font-size: 14px; - font-family: -apple-system; - font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; - line-height: 20px; - text-rendering: optimizeLegibility; - -webkit-backface-visibility: hidden; - -webkit-user-drag: none; - -ms-content-zooming: none; } - - .content { - position: relative; } - - /* If you change these, change platform.scss as well */ - .has-header { - top: 44px; } - - .no-header { - top: 0; } - - .has-subheader { - top: 88px; } - - .has-tabs-top { - top: 93px; } - - .has-header.has-subheader.has-tabs-top { - top: 137px; } - - .has-footer { - bottom: 44px; } - - .has-subfooter { - bottom: 88px; } - - .has-tabs, - .bar-footer.has-tabs { - bottom: 49px; } - .has-tabs.pane, - .bar-footer.has-tabs.pane { - bottom: 49px; - height: auto; } - - .bar-subfooter.has-tabs { - bottom: 93px; } - - .has-footer.has-tabs { - bottom: 93px; } - - /** - * Typography - * -------------------------------------------------- - */ - p { - margin: 0 0 10px; } - - small { - font-size: 85%; } - - cite { - font-style: normal; } - - .text-left { - text-align: left; } - - .text-right { - text-align: right; } - - .text-center, .item.large-button-bar { - text-align: center; } - - h1, h2, h3, h4, h5, h6, - .h1, .h2, .h3, .h4, .h5, .h6 { - color: #000; - font-weight: 500; - font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; - line-height: 1.2; } - h1 small, h2 small, h3 small, h4 small, h5 small, h6 small, - .h1 small, .h2 small, .h3 small, .h4 small, .h5 small, .h6 small { - font-weight: normal; - line-height: 1; } - - h1, .h1, - h2, .h2, - h3, .h3 { - margin-top: 20px; - margin-bottom: 10px; } - h1:first-child, .h1:first-child, - h2:first-child, .h2:first-child, - h3:first-child, .h3:first-child { - margin-top: 0; } - h1 + h1, h1 + .h1, - h1 + h2, h1 + .h2, - h1 + h3, h1 + .h3, .h1 + h1, .h1 + .h1, - .h1 + h2, .h1 + .h2, - .h1 + h3, .h1 + .h3, - h2 + h1, - h2 + .h1, - h2 + h2, - h2 + .h2, - h2 + h3, - h2 + .h3, .h2 + h1, .h2 + .h1, - .h2 + h2, .h2 + .h2, - .h2 + h3, .h2 + .h3, - h3 + h1, - h3 + .h1, - h3 + h2, - h3 + .h2, - h3 + h3, - h3 + .h3, .h3 + h1, .h3 + .h1, - .h3 + h2, .h3 + .h2, - .h3 + h3, .h3 + .h3 { - margin-top: 10px; } - - h4, .h4, - h5, .h5, - h6, .h6 { - margin-top: 10px; - margin-bottom: 10px; } - - h1, .h1 { - font-size: 36px; } - - h2, .h2 { - font-size: 30px; } - - h3, .h3 { - font-size: 24px; } - - h4, .h4 { - font-size: 18px; } - - h5, .h5 { - font-size: 14px; } - - h6, .h6 { - font-size: 12px; } - - h1 small, .h1 small { - font-size: 24px; } - - h2 small, .h2 small { - font-size: 18px; } - - h3 small, .h3 small, - h4 small, .h4 small { - font-size: 14px; } - - dl { - margin-bottom: 20px; } - - dt, - dd { - line-height: 1.42857; } - - dt { - font-weight: bold; } - - blockquote { - margin: 0 0 20px; - padding: 10px 20px; - border-left: 5px solid gray; } - blockquote p { - font-weight: 300; - font-size: 17.5px; - line-height: 1.25; } - blockquote p:last-child { - margin-bottom: 0; } - blockquote small { - display: block; - line-height: 1.42857; } - blockquote small:before { - content: '\2014 \00A0'; } - - q:before, - q:after, - blockquote:before, - blockquote:after { - content: ""; } - - address { - display: block; - margin-bottom: 20px; - font-style: normal; - line-height: 1.42857; } - - a { - color: #387ef5; } - - a.subdued { - padding-right: 10px; - color: #888; - text-decoration: none; } - a.subdued:hover { - text-decoration: none; } - a.subdued:last-child { - padding-right: 0; } - - /** - * Bar (Headers and Footers) - * -------------------------------------------------- - */ - .bar { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - position: absolute; - right: 0; - left: 0; - z-index: 9; - -webkit-box-sizing: border-box; - -moz-box-sizing: border-box; - box-sizing: border-box; - padding: 5px; - width: 100%; - height: 44px; - border-width: 0; - border-style: solid; - border-top: 1px solid transparent; - border-bottom: 1px solid #ddd; - background-color: white; - /* border-width: 1px will actually create 2 device pixels on retina */ - /* this nifty trick sets an actual 1px border on hi-res displays */ - background-size: 0; } - @media (min--moz-device-pixel-ratio: 1.5), (-webkit-min-device-pixel-ratio: 1.5), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi), (min-resolution: 1.5dppx) { - .bar { - border: none; - background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); - background-position: bottom; - background-size: 100% 1px; - background-repeat: no-repeat; } } - .bar.bar-clear { - border: none; - background: none; - color: #fff; } - .bar.bar-clear .button { - color: #fff; } - .bar.bar-clear .title { - color: #fff; } - .bar.item-input-inset .item-input-wrapper { - margin-top: -1px; } - .bar.item-input-inset .item-input-wrapper input { - padding-left: 8px; - width: 94%; - height: 28px; - background: transparent; } - .bar.bar-light { - border-color: #ddd; - background-color: white; - background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); - color: #444; } - .bar.bar-light .title { - color: #444; } - .bar.bar-light.bar-footer { - background-image: linear-gradient(180deg, #ddd, #ddd 50%, transparent 50%); } - .bar.bar-stable { - border-color: #b2b2b2; - background-color: #f8f8f8; - background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); - color: #444; } - .bar.bar-stable .title { - color: #444; } - .bar.bar-stable.bar-footer { - background-image: linear-gradient(180deg, #b2b2b2, #b2b2b2 50%, transparent 50%); } - .bar.bar-positive { - border-color: #0c60ee; - background-color: #387ef5; - background-image: linear-gradient(0deg, #0c60ee, #0c60ee 50%, transparent 50%); - color: #fff; } - .bar.bar-positive .title { - color: #fff; } - .bar.bar-positive.bar-footer { - background-image: linear-gradient(180deg, #0c60ee, #0c60ee 50%, transparent 50%); } - .bar.bar-calm { - border-color: #0a9dc7; - background-color: #11c1f3; - background-image: linear-gradient(0deg, #0a9dc7, #0a9dc7 50%, transparent 50%); - color: #fff; } - .bar.bar-calm .title { - color: #fff; } - .bar.bar-calm.bar-footer { - background-image: linear-gradient(180deg, #0a9dc7, #0a9dc7 50%, transparent 50%); } - .bar.bar-assertive { - border-color: #e42112; - background-color: #ef473a; - background-image: linear-gradient(0deg, #e42112, #e42112 50%, transparent 50%); - color: #fff; } - .bar.bar-assertive .title { - color: #fff; } - .bar.bar-assertive.bar-footer { - background-image: linear-gradient(180deg, #e42112, #e42112 50%, transparent 50%); } - .bar.bar-balanced { - border-color: #28a54c; - background-color: #33cd5f; - background-image: linear-gradient(0deg, #28a54c, #28a54c 50%, transparent 50%); - color: #fff; } - .bar.bar-balanced .title { - color: #fff; } - .bar.bar-balanced.bar-footer { - background-image: linear-gradient(180deg, #28a54c, #28a54c 50%, transparent 50%); } - .bar.bar-energized { - border-color: #e6b500; - background-color: #ffc900; - background-image: linear-gradient(0deg, #e6b500, #e6b500 50%, transparent 50%); - color: #fff; } - .bar.bar-energized .title { - color: #fff; } - .bar.bar-energized.bar-footer { - background-image: linear-gradient(180deg, #e6b500, #e6b500 50%, transparent 50%); } - .bar.bar-royal { - border-color: #6b46e5; - background-color: #886aea; - background-image: linear-gradient(0deg, #6b46e5, #6b46e5 50%, transparent 50%); - color: #fff; } - .bar.bar-royal .title { - color: #fff; } - .bar.bar-royal.bar-footer { - background-image: linear-gradient(180deg, #6b46e5, #6b46e5 50%, transparent 50%); } - .bar.bar-dark { - border-color: #111; - background-color: #444444; - background-image: linear-gradient(0deg, #111, #111 50%, transparent 50%); - color: #fff; } - .bar.bar-dark .title { - color: #fff; } - .bar.bar-dark.bar-footer { - background-image: linear-gradient(180deg, #111, #111 50%, transparent 50%); } - .bar .title { - display: block; - position: absolute; - top: 0; - right: 0; - left: 0; - z-index: 0; - overflow: hidden; - margin: 0 10px; - min-width: 30px; - height: 43px; - text-align: center; - text-overflow: ellipsis; - white-space: nowrap; - font-size: 17px; - font-weight: 500; - line-height: 44px; } - .bar .title.title-left { - text-align: left; } - .bar .title.title-right { - text-align: right; } - .bar .title a { - color: inherit; } - .bar .button, .bar button { - z-index: 1; - padding: 0 8px; - min-width: initial; - min-height: 31px; - font-weight: 400; - font-size: 13px; - line-height: 32px; } - .bar .button.button-icon:before, - .bar .button .icon:before, - .bar .button .icon-help:before, - .bar .button .icon-alert:before, - .bar .button #menu .footer .icon-help:before, #menu .footer - .bar .button .icon-help:before, .bar .button.icon:before, .bar .button.icon-help:before, .bar .button.icon-alert:before, .bar #menu .footer .button.icon-help:before, #menu .footer .bar .button.icon-help:before, .bar .button.icon-left:before, .bar .button.icon-right:before, .bar button.button-icon:before, - .bar button .icon:before, - .bar button .icon-help:before, - .bar button .icon-alert:before, - .bar button #menu .footer .icon-help:before, #menu .footer - .bar button .icon-help:before, .bar button.icon:before, .bar button.icon-help:before, .bar button.icon-alert:before, .bar #menu .footer button.icon-help:before, #menu .footer .bar button.icon-help:before, .bar button.icon-left:before, .bar button.icon-right:before { - padding-right: 2px; - padding-left: 2px; - font-size: 20px; - line-height: 32px; } - .bar .button.button-icon, .bar button.button-icon { - font-size: 17px; } - .bar .button.button-icon .icon:before, .bar .button.button-icon .icon-help:before, .bar .button.button-icon .icon-alert:before, .bar .button.button-icon #menu .footer .icon-help:before, #menu .footer .bar .button.button-icon .icon-help:before, .bar .button.button-icon:before, .bar .button.button-icon.icon-left:before, .bar .button.button-icon.icon-right:before, .bar button.button-icon .icon:before, .bar button.button-icon .icon-help:before, .bar button.button-icon .icon-alert:before, .bar button.button-icon #menu .footer .icon-help:before, #menu .footer .bar button.button-icon .icon-help:before, .bar button.button-icon:before, .bar button.button-icon.icon-left:before, .bar button.button-icon.icon-right:before { - vertical-align: top; - font-size: 32px; - line-height: 32px; } - .bar .button.button-clear, .bar .button.button-text, .bar button.button-clear, .bar button.button-text { - padding-right: 2px; - padding-left: 2px; - font-weight: 300; - font-size: 17px; } - .bar .button.button-clear .icon:before, .bar .button.button-text .icon:before, .bar .button.button-clear .icon-help:before, .bar .button.button-text .icon-help:before, .bar .button.button-clear .icon-alert:before, .bar .button.button-text .icon-alert:before, .bar .button.button-clear #menu .footer .icon-help:before, #menu .footer .bar .button.button-clear .icon-help:before, .bar .button.button-text #menu .footer .icon-help:before, #menu .footer .bar .button.button-text .icon-help:before, .bar .button.button-clear.icon:before, .bar .button.icon.button-text:before, .bar .button.button-text.icon-help:before, .bar .button.button-text.icon-alert:before, .bar #menu .footer .button.button-text.icon-help:before, #menu .footer .bar .button.button-text.icon-help:before, .bar .button.button-clear.icon-help:before, .bar .button.button-clear.icon-alert:before, .bar #menu .footer .button.button-clear.icon-help:before, #menu .footer .bar .button.button-clear.icon-help:before, .bar .button.button-clear.icon-left:before, .bar .button.icon-left.button-text:before, .bar .button.button-clear.icon-right:before, .bar .button.icon-right.button-text:before, .bar button.button-clear .icon:before, .bar button.button-text .icon:before, .bar button.button-clear .icon-help:before, .bar button.button-text .icon-help:before, .bar button.button-clear .icon-alert:before, .bar button.button-text .icon-alert:before, .bar button.button-clear #menu .footer .icon-help:before, #menu .footer .bar button.button-clear .icon-help:before, .bar button.button-text #menu .footer .icon-help:before, #menu .footer .bar button.button-text .icon-help:before, .bar button.button-clear.icon:before, .bar button.icon.button-text:before, .bar button.button-text.icon-help:before, .bar button.button-text.icon-alert:before, .bar #menu .footer button.button-text.icon-help:before, #menu .footer .bar button.button-text.icon-help:before, .bar button.button-clear.icon-help:before, .bar button.button-clear.icon-alert:before, .bar #menu .footer button.button-clear.icon-help:before, #menu .footer .bar button.button-clear.icon-help:before, .bar button.button-clear.icon-left:before, .bar button.icon-left.button-text:before, .bar button.button-clear.icon-right:before, .bar button.icon-right.button-text:before { - font-size: 32px; - line-height: 32px; } - .bar .button.back-button, .bar button.back-button { - display: block; - margin-right: 5px; - padding: 0; - white-space: nowrap; - font-weight: 400; } - .bar .button.back-button.active, .bar .button.back-button.activated, .bar button.back-button.active, .bar button.back-button.activated { - opacity: 0.2; } - .bar .button-bar > .button, - .bar .buttons > .button { - min-height: 31px; - line-height: 32px; } - .bar .button-bar + .button, - .bar .button + .button-bar { - margin-left: 5px; } - .bar .buttons, - .bar .buttons.primary-buttons, - .bar .buttons.secondary-buttons { - display: inherit; } - .bar .buttons span { - display: inline-block; } - .bar .buttons-left span { - margin-right: 5px; - display: inherit; } - .bar .buttons-right span { - margin-left: 5px; - display: inherit; } - .bar .title + .button:last-child, - .bar > .button + .button:last-child, - .bar > .button.pull-right, .popover-helptip - .bar > .button.icon.icon-right, .popover-helptip - .bar > .button.icon-right.icon-help, .popover-helptip - .bar > .button.icon-right.icon-alert, .popover-helptip #menu .footer - .bar > .button.icon-right.icon-help, #menu .footer .popover-helptip - .bar > .button.icon-right.icon-help, .popover-helptip - .bar > .button.icon.icon-center, .popover-helptip - .bar > .button.icon-center.icon-help, .popover-helptip - .bar > .button.icon-center.icon-alert, .popover-helptip #menu .footer - .bar > .button.icon-center.icon-help, #menu .footer .popover-helptip - .bar > .button.icon-center.icon-help, .popover-helptip - .bar > .button.icon.icon-bottom-right, .popover-helptip - .bar > .button.icon-bottom-right.icon-help, .popover-helptip - .bar > .button.icon-bottom-right.icon-alert, .popover-helptip #menu .footer - .bar > .button.icon-bottom-right.icon-help, #menu .footer .popover-helptip - .bar > .button.icon-bottom-right.icon-help, .popover-helptip - .bar > .button.icon.icon-bottom-center, .popover-helptip - .bar > .button.icon-bottom-center.icon-help, .popover-helptip - .bar > .button.icon-bottom-center.icon-alert, .popover-helptip #menu .footer - .bar > .button.icon-bottom-center.icon-help, #menu .footer .popover-helptip - .bar > .button.icon-bottom-center.icon-help, - .bar .buttons.pull-right, - .bar .popover-helptip .buttons.icon.icon-right, .popover-helptip - .bar .buttons.icon.icon-right, - .bar .popover-helptip .buttons.icon-right.icon-help, .popover-helptip - .bar .buttons.icon-right.icon-help, - .bar .popover-helptip .buttons.icon-right.icon-alert, .popover-helptip - .bar .buttons.icon-right.icon-alert, - .bar .popover-helptip #menu .footer .buttons.icon-right.icon-help, .popover-helptip #menu .footer - .bar .buttons.icon-right.icon-help, - .bar #menu .footer .popover-helptip .buttons.icon-right.icon-help, #menu .footer .popover-helptip - .bar .buttons.icon-right.icon-help, - .bar .popover-helptip .buttons.icon.icon-center, .popover-helptip - .bar .buttons.icon.icon-center, - .bar .popover-helptip .buttons.icon-center.icon-help, .popover-helptip - .bar .buttons.icon-center.icon-help, - .bar .popover-helptip .buttons.icon-center.icon-alert, .popover-helptip - .bar .buttons.icon-center.icon-alert, - .bar .popover-helptip #menu .footer .buttons.icon-center.icon-help, .popover-helptip #menu .footer - .bar .buttons.icon-center.icon-help, - .bar #menu .footer .popover-helptip .buttons.icon-center.icon-help, #menu .footer .popover-helptip - .bar .buttons.icon-center.icon-help, - .bar .popover-helptip .buttons.icon.icon-bottom-right, .popover-helptip - .bar .buttons.icon.icon-bottom-right, - .bar .popover-helptip .buttons.icon-bottom-right.icon-help, .popover-helptip - .bar .buttons.icon-bottom-right.icon-help, - .bar .popover-helptip .buttons.icon-bottom-right.icon-alert, .popover-helptip - .bar .buttons.icon-bottom-right.icon-alert, - .bar .popover-helptip #menu .footer .buttons.icon-bottom-right.icon-help, .popover-helptip #menu .footer - .bar .buttons.icon-bottom-right.icon-help, - .bar #menu .footer .popover-helptip .buttons.icon-bottom-right.icon-help, #menu .footer .popover-helptip - .bar .buttons.icon-bottom-right.icon-help, - .bar .popover-helptip .buttons.icon.icon-bottom-center, .popover-helptip - .bar .buttons.icon.icon-bottom-center, - .bar .popover-helptip .buttons.icon-bottom-center.icon-help, .popover-helptip - .bar .buttons.icon-bottom-center.icon-help, - .bar .popover-helptip .buttons.icon-bottom-center.icon-alert, .popover-helptip - .bar .buttons.icon-bottom-center.icon-alert, - .bar .popover-helptip #menu .footer .buttons.icon-bottom-center.icon-help, .popover-helptip #menu .footer - .bar .buttons.icon-bottom-center.icon-help, - .bar #menu .footer .popover-helptip .buttons.icon-bottom-center.icon-help, #menu .footer .popover-helptip - .bar .buttons.icon-bottom-center.icon-help, - .bar .title + .buttons { - position: absolute; - top: 5px; - right: 5px; - bottom: 5px; } - - .platform-android .nav-bar-has-subheader .bar { - background-image: none; } - - .platform-android .bar .back-button .icon:before, .platform-android .bar .back-button .icon-help:before, .platform-android .bar .back-button .icon-alert:before, .platform-android .bar .back-button #menu .footer .icon-help:before, #menu .footer .platform-android .bar .back-button .icon-help:before { - font-size: 24px; } - - .platform-android .bar .title { - font-size: 19px; - line-height: 44px; } - - .bar-light .button { - border-color: #ddd; - background-color: white; - color: #444; } - .bar-light .button:hover { - color: #444; - text-decoration: none; } - .bar-light .button.active, .bar-light .button.activated { - border-color: #ccc; - background-color: #fafafa; } - .bar-light .button.button-clear, .bar-light .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #444; - font-size: 17px; } - .bar-light .button.button-icon { - border-color: transparent; - background: none; } - - .bar-stable .button { - border-color: #b2b2b2; - background-color: #f8f8f8; - color: #444; } - .bar-stable .button:hover { - color: #444; - text-decoration: none; } - .bar-stable .button.active, .bar-stable .button.activated { - border-color: #a2a2a2; - background-color: #e5e5e5; } - .bar-stable .button.button-clear, .bar-stable .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #444; - font-size: 17px; } - .bar-stable .button.button-icon { - border-color: transparent; - background: none; } - - .bar-positive .button { - border-color: #0c60ee; - background-color: #387ef5; - color: #fff; } - .bar-positive .button:hover { - color: #fff; - text-decoration: none; } - .bar-positive .button.active, .bar-positive .button.activated { - border-color: #0c60ee; - background-color: #0c60ee; } - .bar-positive .button.button-clear, .bar-positive .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-positive .button.button-icon { - border-color: transparent; - background: none; } - - .bar-calm .button { - border-color: #0a9dc7; - background-color: #11c1f3; - color: #fff; } - .bar-calm .button:hover { - color: #fff; - text-decoration: none; } - .bar-calm .button.active, .bar-calm .button.activated { - border-color: #0a9dc7; - background-color: #0a9dc7; } - .bar-calm .button.button-clear, .bar-calm .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-calm .button.button-icon { - border-color: transparent; - background: none; } - - .bar-assertive .button { - border-color: #e42112; - background-color: #ef473a; - color: #fff; } - .bar-assertive .button:hover { - color: #fff; - text-decoration: none; } - .bar-assertive .button.active, .bar-assertive .button.activated { - border-color: #e42112; - background-color: #e42112; } - .bar-assertive .button.button-clear, .bar-assertive .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-assertive .button.button-icon { - border-color: transparent; - background: none; } - - .bar-balanced .button { - border-color: #28a54c; - background-color: #33cd5f; - color: #fff; } - .bar-balanced .button:hover { - color: #fff; - text-decoration: none; } - .bar-balanced .button.active, .bar-balanced .button.activated { - border-color: #28a54c; - background-color: #28a54c; } - .bar-balanced .button.button-clear, .bar-balanced .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-balanced .button.button-icon { - border-color: transparent; - background: none; } - - .bar-energized .button { - border-color: #e6b500; - background-color: #ffc900; - color: #fff; } - .bar-energized .button:hover { - color: #fff; - text-decoration: none; } - .bar-energized .button.active, .bar-energized .button.activated { - border-color: #e6b500; - background-color: #e6b500; } - .bar-energized .button.button-clear, .bar-energized .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-energized .button.button-icon { - border-color: transparent; - background: none; } - - .bar-royal .button { - border-color: #6b46e5; - background-color: #886aea; - color: #fff; } - .bar-royal .button:hover { - color: #fff; - text-decoration: none; } - .bar-royal .button.active, .bar-royal .button.activated { - border-color: #6b46e5; - background-color: #6b46e5; } - .bar-royal .button.button-clear, .bar-royal .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-royal .button.button-icon { - border-color: transparent; - background: none; } - - .bar-dark .button { - border-color: #111; - background-color: #444444; - color: #fff; } - .bar-dark .button:hover { - color: #fff; - text-decoration: none; } - .bar-dark .button.active, .bar-dark .button.activated { - border-color: #000; - background-color: #262626; } - .bar-dark .button.button-clear, .bar-dark .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #fff; - font-size: 17px; } - .bar-dark .button.button-icon { - border-color: transparent; - background: none; } - - .bar-header { - top: 0; - border-top-width: 0; - border-bottom-width: 1px; } - .bar-header.has-tabs-top { - border-bottom-width: 0px; - background-image: none; } - - .tabs-top .bar-header { - border-bottom-width: 0px; - background-image: none; } - - .bar-footer { - bottom: 0; - border-top-width: 1px; - border-bottom-width: 0; - background-position: top; - height: 44px; } - .bar-footer.item-input-inset { - position: absolute; } - .bar-footer .title { - height: 43px; - line-height: 44px; } - - .bar-tabs { - padding: 0; } - - .bar-subheader { - top: 44px; - height: 44px; } - .bar-subheader .title { - height: 43px; - line-height: 44px; } - - .bar-subfooter { - bottom: 44px; - height: 44px; } - .bar-subfooter .title { - height: 43px; - line-height: 44px; } - - .nav-bar-block { - position: absolute; - top: 0; - right: 0; - left: 0; - z-index: 9; } - - .bar .back-button.hide, - .bar .buttons .hide { - display: none; } - - .nav-bar-tabs-top .bar { - background-image: none; } - - /** - * Tabs - * -------------------------------------------------- - * A navigation bar with any number of tab items supported. - */ - .tabs { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-direction: normal; - -webkit-box-orient: horizontal; - -webkit-flex-direction: horizontal; - -moz-flex-direction: horizontal; - -ms-flex-direction: horizontal; - flex-direction: horizontal; - -webkit-box-pack: center; - -ms-flex-pack: center; - -webkit-justify-content: center; - -moz-justify-content: center; - justify-content: center; - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - border-color: #b2b2b2; - background-color: #f8f8f8; - background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); - color: #444; - position: absolute; - bottom: 0; - z-index: 5; - width: 100%; - height: 49px; - border-style: solid; - border-top-width: 1px; - background-size: 0; - line-height: 49px; } - .tabs .tab-item .badge { - background-color: #444; - color: #f8f8f8; } - @media (min--moz-device-pixel-ratio: 1.5), (-webkit-min-device-pixel-ratio: 1.5), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi), (min-resolution: 1.5dppx) { - .tabs { - padding-top: 2px; - border-top: none !important; - border-bottom: none; - background-position: top; - background-size: 100% 1px; - background-repeat: no-repeat; } } - - /* Allow parent element of tabs to define color, or just the tab itself */ - .tabs-light > .tabs, - .tabs.tabs-light { - border-color: #ddd; - background-color: #fff; - background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); - color: #444; } - .tabs-light > .tabs .tab-item .badge, - .tabs.tabs-light .tab-item .badge { - background-color: #444; - color: #fff; } - - .tabs-stable > .tabs, - .tabs.tabs-stable { - border-color: #b2b2b2; - background-color: #f8f8f8; - background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); - color: #444; } - .tabs-stable > .tabs .tab-item .badge, - .tabs.tabs-stable .tab-item .badge { - background-color: #444; - color: #f8f8f8; } - - .tabs-positive > .tabs, - .tabs.tabs-positive { - border-color: #0c60ee; - background-color: #387ef5; - background-image: linear-gradient(0deg, #0c60ee, #0c60ee 50%, transparent 50%); - color: #fff; } - .tabs-positive > .tabs .tab-item .badge, - .tabs.tabs-positive .tab-item .badge { - background-color: #fff; - color: #387ef5; } - - .tabs-calm > .tabs, - .tabs.tabs-calm { - border-color: #0a9dc7; - background-color: #11c1f3; - background-image: linear-gradient(0deg, #0a9dc7, #0a9dc7 50%, transparent 50%); - color: #fff; } - .tabs-calm > .tabs .tab-item .badge, - .tabs.tabs-calm .tab-item .badge { - background-color: #fff; - color: #11c1f3; } - - .tabs-assertive > .tabs, - .tabs.tabs-assertive { - border-color: #e42112; - background-color: #ef473a; - background-image: linear-gradient(0deg, #e42112, #e42112 50%, transparent 50%); - color: #fff; } - .tabs-assertive > .tabs .tab-item .badge, - .tabs.tabs-assertive .tab-item .badge { - background-color: #fff; - color: #ef473a; } - - .tabs-balanced > .tabs, - .tabs.tabs-balanced { - border-color: #28a54c; - background-color: #33cd5f; - background-image: linear-gradient(0deg, #28a54c, #28a54c 50%, transparent 50%); - color: #fff; } - .tabs-balanced > .tabs .tab-item .badge, - .tabs.tabs-balanced .tab-item .badge { - background-color: #fff; - color: #33cd5f; } - - .tabs-energized > .tabs, - .tabs.tabs-energized { - border-color: #e6b500; - background-color: #ffc900; - background-image: linear-gradient(0deg, #e6b500, #e6b500 50%, transparent 50%); - color: #fff; } - .tabs-energized > .tabs .tab-item .badge, - .tabs.tabs-energized .tab-item .badge { - background-color: #fff; - color: #ffc900; } - - .tabs-royal > .tabs, - .tabs.tabs-royal { - border-color: #6b46e5; - background-color: #886aea; - background-image: linear-gradient(0deg, #6b46e5, #6b46e5 50%, transparent 50%); - color: #fff; } - .tabs-royal > .tabs .tab-item .badge, - .tabs.tabs-royal .tab-item .badge { - background-color: #fff; - color: #886aea; } - - .tabs-dark > .tabs, - .tabs.tabs-dark { - border-color: #111; - background-color: #444; - background-image: linear-gradient(0deg, #111, #111 50%, transparent 50%); - color: #fff; } - .tabs-dark > .tabs .tab-item .badge, - .tabs.tabs-dark .tab-item .badge { - background-color: #fff; - color: #444; } - - .tabs-striped .tabs { - background-color: white; - background-image: none; - border: none; - border-bottom: 1px solid #ddd; - padding-top: 2px; } - - .tabs-striped .tab-item.tab-item-active, .tabs-striped .tab-item.active, .tabs-striped .tab-item.activated { - margin-top: -2px; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #444; } - .tabs-striped .tab-item.tab-item-active .badge, .tabs-striped .tab-item.active .badge, .tabs-striped .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-light .tabs { - background-color: #fff; } - - .tabs-striped.tabs-light .tab-item { - color: rgba(68, 68, 68, 0.4); - opacity: 1; } - .tabs-striped.tabs-light .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-light .tab-item.tab-item-active, .tabs-striped.tabs-light .tab-item.active, .tabs-striped.tabs-light .tab-item.activated { - margin-top: -2px; - color: #444; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #444; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-stable .tabs { - background-color: #f8f8f8; } - - .tabs-striped.tabs-stable .tab-item { - color: rgba(68, 68, 68, 0.4); - opacity: 1; } - .tabs-striped.tabs-stable .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-stable .tab-item.tab-item-active, .tabs-striped.tabs-stable .tab-item.active, .tabs-striped.tabs-stable .tab-item.activated { - margin-top: -2px; - color: #444; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #444; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-positive .tabs { - background-color: #387ef5; } - - .tabs-striped.tabs-positive .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-positive .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-positive .tab-item.tab-item-active, .tabs-striped.tabs-positive .tab-item.active, .tabs-striped.tabs-positive .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-calm .tabs { - background-color: #11c1f3; } - - .tabs-striped.tabs-calm .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-calm .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-calm .tab-item.tab-item-active, .tabs-striped.tabs-calm .tab-item.active, .tabs-striped.tabs-calm .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-assertive .tabs { - background-color: #ef473a; } - - .tabs-striped.tabs-assertive .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-assertive .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-assertive .tab-item.tab-item-active, .tabs-striped.tabs-assertive .tab-item.active, .tabs-striped.tabs-assertive .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-balanced .tabs { - background-color: #33cd5f; } - - .tabs-striped.tabs-balanced .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-balanced .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-balanced .tab-item.tab-item-active, .tabs-striped.tabs-balanced .tab-item.active, .tabs-striped.tabs-balanced .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-energized .tabs { - background-color: #ffc900; } - - .tabs-striped.tabs-energized .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-energized .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-energized .tab-item.tab-item-active, .tabs-striped.tabs-energized .tab-item.active, .tabs-striped.tabs-energized .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-royal .tabs { - background-color: #886aea; } - - .tabs-striped.tabs-royal .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-royal .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-royal .tab-item.tab-item-active, .tabs-striped.tabs-royal .tab-item.active, .tabs-striped.tabs-royal .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-dark .tabs { - background-color: #444; } - - .tabs-striped.tabs-dark .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-dark .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-dark .tab-item.tab-item-active, .tabs-striped.tabs-dark .tab-item.active, .tabs-striped.tabs-dark .tab-item.activated { - margin-top: -2px; - color: #fff; - border-style: solid; - border-width: 2px 0 0 0; - border-color: #fff; } - - .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-striped.tabs-background-light .tabs { - background-color: #fff; - background-image: none; } - - .tabs-striped.tabs-background-stable .tabs { - background-color: #f8f8f8; - background-image: none; } - - .tabs-striped.tabs-background-positive .tabs { - background-color: #387ef5; - background-image: none; } - - .tabs-striped.tabs-background-calm .tabs { - background-color: #11c1f3; - background-image: none; } - - .tabs-striped.tabs-background-assertive .tabs { - background-color: #ef473a; - background-image: none; } - - .tabs-striped.tabs-background-balanced .tabs { - background-color: #33cd5f; - background-image: none; } - - .tabs-striped.tabs-background-energized .tabs { - background-color: #ffc900; - background-image: none; } - - .tabs-striped.tabs-background-royal .tabs { - background-color: #886aea; - background-image: none; } - - .tabs-striped.tabs-background-dark .tabs { - background-color: #444; - background-image: none; } - - .tabs-striped.tabs-color-light .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-light .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-light .tab-item.tab-item-active, .tabs-striped.tabs-color-light .tab-item.active, .tabs-striped.tabs-color-light .tab-item.activated { - margin-top: -2px; - color: #fff; - border: 0 solid #fff; - border-top-width: 2px; } - .tabs-striped.tabs-color-light .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-light .tab-item.active .badge, .tabs-striped.tabs-color-light .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-stable .tab-item { - color: rgba(248, 248, 248, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-stable .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-stable .tab-item.tab-item-active, .tabs-striped.tabs-color-stable .tab-item.active, .tabs-striped.tabs-color-stable .tab-item.activated { - margin-top: -2px; - color: #f8f8f8; - border: 0 solid #f8f8f8; - border-top-width: 2px; } - .tabs-striped.tabs-color-stable .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-stable .tab-item.active .badge, .tabs-striped.tabs-color-stable .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-positive .tab-item { - color: rgba(56, 126, 245, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-positive .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-positive .tab-item.tab-item-active, .tabs-striped.tabs-color-positive .tab-item.active, .tabs-striped.tabs-color-positive .tab-item.activated { - margin-top: -2px; - color: #387ef5; - border: 0 solid #387ef5; - border-top-width: 2px; } - .tabs-striped.tabs-color-positive .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-positive .tab-item.active .badge, .tabs-striped.tabs-color-positive .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-calm .tab-item { - color: rgba(17, 193, 243, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-calm .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-calm .tab-item.tab-item-active, .tabs-striped.tabs-color-calm .tab-item.active, .tabs-striped.tabs-color-calm .tab-item.activated { - margin-top: -2px; - color: #11c1f3; - border: 0 solid #11c1f3; - border-top-width: 2px; } - .tabs-striped.tabs-color-calm .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-calm .tab-item.active .badge, .tabs-striped.tabs-color-calm .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-assertive .tab-item { - color: rgba(239, 71, 58, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-assertive .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-assertive .tab-item.tab-item-active, .tabs-striped.tabs-color-assertive .tab-item.active, .tabs-striped.tabs-color-assertive .tab-item.activated { - margin-top: -2px; - color: #ef473a; - border: 0 solid #ef473a; - border-top-width: 2px; } - .tabs-striped.tabs-color-assertive .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-assertive .tab-item.active .badge, .tabs-striped.tabs-color-assertive .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-balanced .tab-item { - color: rgba(51, 205, 95, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-balanced .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-balanced .tab-item.tab-item-active, .tabs-striped.tabs-color-balanced .tab-item.active, .tabs-striped.tabs-color-balanced .tab-item.activated { - margin-top: -2px; - color: #33cd5f; - border: 0 solid #33cd5f; - border-top-width: 2px; } - .tabs-striped.tabs-color-balanced .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-balanced .tab-item.active .badge, .tabs-striped.tabs-color-balanced .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-energized .tab-item { - color: rgba(255, 201, 0, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-energized .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-energized .tab-item.tab-item-active, .tabs-striped.tabs-color-energized .tab-item.active, .tabs-striped.tabs-color-energized .tab-item.activated { - margin-top: -2px; - color: #ffc900; - border: 0 solid #ffc900; - border-top-width: 2px; } - .tabs-striped.tabs-color-energized .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-energized .tab-item.active .badge, .tabs-striped.tabs-color-energized .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-royal .tab-item { - color: rgba(136, 106, 234, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-royal .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-royal .tab-item.tab-item-active, .tabs-striped.tabs-color-royal .tab-item.active, .tabs-striped.tabs-color-royal .tab-item.activated { - margin-top: -2px; - color: #886aea; - border: 0 solid #886aea; - border-top-width: 2px; } - .tabs-striped.tabs-color-royal .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-royal .tab-item.active .badge, .tabs-striped.tabs-color-royal .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-striped.tabs-color-dark .tab-item { - color: rgba(68, 68, 68, 0.4); - opacity: 1; } - .tabs-striped.tabs-color-dark .tab-item .badge { - opacity: 0.4; } - .tabs-striped.tabs-color-dark .tab-item.tab-item-active, .tabs-striped.tabs-color-dark .tab-item.active, .tabs-striped.tabs-color-dark .tab-item.activated { - margin-top: -2px; - color: #444; - border: 0 solid #444; - border-top-width: 2px; } - .tabs-striped.tabs-color-dark .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-dark .tab-item.active .badge, .tabs-striped.tabs-color-dark .tab-item.activated .badge { - top: 2px; - opacity: 1; } - - .tabs-background-light .tabs, - .tabs-background-light > .tabs { - background-color: #fff; - background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); - border-color: #ddd; } - - .tabs-background-stable .tabs, - .tabs-background-stable > .tabs { - background-color: #f8f8f8; - background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); - border-color: #b2b2b2; } - - .tabs-background-positive .tabs, - .tabs-background-positive > .tabs { - background-color: #387ef5; - background-image: linear-gradient(0deg, #0c60ee, #0c60ee 50%, transparent 50%); - border-color: #0c60ee; } - - .tabs-background-calm .tabs, - .tabs-background-calm > .tabs { - background-color: #11c1f3; - background-image: linear-gradient(0deg, #0a9dc7, #0a9dc7 50%, transparent 50%); - border-color: #0a9dc7; } - - .tabs-background-assertive .tabs, - .tabs-background-assertive > .tabs { - background-color: #ef473a; - background-image: linear-gradient(0deg, #e42112, #e42112 50%, transparent 50%); - border-color: #e42112; } - - .tabs-background-balanced .tabs, - .tabs-background-balanced > .tabs { - background-color: #33cd5f; - background-image: linear-gradient(0deg, #28a54c, #28a54c 50%, transparent 50%); - border-color: #28a54c; } - - .tabs-background-energized .tabs, - .tabs-background-energized > .tabs { - background-color: #ffc900; - background-image: linear-gradient(0deg, #e6b500, #e6b500 50%, transparent 50%); - border-color: #e6b500; } - - .tabs-background-royal .tabs, - .tabs-background-royal > .tabs { - background-color: #886aea; - background-image: linear-gradient(0deg, #6b46e5, #6b46e5 50%, transparent 50%); - border-color: #6b46e5; } - - .tabs-background-dark .tabs, - .tabs-background-dark > .tabs { - background-color: #444; - background-image: linear-gradient(0deg, #111, #111 50%, transparent 50%); - border-color: #111; } - - .tabs-color-light .tab-item { - color: rgba(255, 255, 255, 0.4); - opacity: 1; } - .tabs-color-light .tab-item .badge { - opacity: 0.4; } - .tabs-color-light .tab-item.tab-item-active, .tabs-color-light .tab-item.active, .tabs-color-light .tab-item.activated { - color: #fff; - border: 0 solid #fff; } - .tabs-color-light .tab-item.tab-item-active .badge, .tabs-color-light .tab-item.active .badge, .tabs-color-light .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-stable .tab-item { - color: rgba(248, 248, 248, 0.4); - opacity: 1; } - .tabs-color-stable .tab-item .badge { - opacity: 0.4; } - .tabs-color-stable .tab-item.tab-item-active, .tabs-color-stable .tab-item.active, .tabs-color-stable .tab-item.activated { - color: #f8f8f8; - border: 0 solid #f8f8f8; } - .tabs-color-stable .tab-item.tab-item-active .badge, .tabs-color-stable .tab-item.active .badge, .tabs-color-stable .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-positive .tab-item { - color: rgba(56, 126, 245, 0.4); - opacity: 1; } - .tabs-color-positive .tab-item .badge { - opacity: 0.4; } - .tabs-color-positive .tab-item.tab-item-active, .tabs-color-positive .tab-item.active, .tabs-color-positive .tab-item.activated { - color: #387ef5; - border: 0 solid #387ef5; } - .tabs-color-positive .tab-item.tab-item-active .badge, .tabs-color-positive .tab-item.active .badge, .tabs-color-positive .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-calm .tab-item { - color: rgba(17, 193, 243, 0.4); - opacity: 1; } - .tabs-color-calm .tab-item .badge { - opacity: 0.4; } - .tabs-color-calm .tab-item.tab-item-active, .tabs-color-calm .tab-item.active, .tabs-color-calm .tab-item.activated { - color: #11c1f3; - border: 0 solid #11c1f3; } - .tabs-color-calm .tab-item.tab-item-active .badge, .tabs-color-calm .tab-item.active .badge, .tabs-color-calm .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-assertive .tab-item { - color: rgba(239, 71, 58, 0.4); - opacity: 1; } - .tabs-color-assertive .tab-item .badge { - opacity: 0.4; } - .tabs-color-assertive .tab-item.tab-item-active, .tabs-color-assertive .tab-item.active, .tabs-color-assertive .tab-item.activated { - color: #ef473a; - border: 0 solid #ef473a; } - .tabs-color-assertive .tab-item.tab-item-active .badge, .tabs-color-assertive .tab-item.active .badge, .tabs-color-assertive .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-balanced .tab-item { - color: rgba(51, 205, 95, 0.4); - opacity: 1; } - .tabs-color-balanced .tab-item .badge { - opacity: 0.4; } - .tabs-color-balanced .tab-item.tab-item-active, .tabs-color-balanced .tab-item.active, .tabs-color-balanced .tab-item.activated { - color: #33cd5f; - border: 0 solid #33cd5f; } - .tabs-color-balanced .tab-item.tab-item-active .badge, .tabs-color-balanced .tab-item.active .badge, .tabs-color-balanced .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-energized .tab-item { - color: rgba(255, 201, 0, 0.4); - opacity: 1; } - .tabs-color-energized .tab-item .badge { - opacity: 0.4; } - .tabs-color-energized .tab-item.tab-item-active, .tabs-color-energized .tab-item.active, .tabs-color-energized .tab-item.activated { - color: #ffc900; - border: 0 solid #ffc900; } - .tabs-color-energized .tab-item.tab-item-active .badge, .tabs-color-energized .tab-item.active .badge, .tabs-color-energized .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-royal .tab-item { - color: rgba(136, 106, 234, 0.4); - opacity: 1; } - .tabs-color-royal .tab-item .badge { - opacity: 0.4; } - .tabs-color-royal .tab-item.tab-item-active, .tabs-color-royal .tab-item.active, .tabs-color-royal .tab-item.activated { - color: #886aea; - border: 0 solid #886aea; } - .tabs-color-royal .tab-item.tab-item-active .badge, .tabs-color-royal .tab-item.active .badge, .tabs-color-royal .tab-item.activated .badge { - opacity: 1; } - - .tabs-color-dark .tab-item { - color: rgba(68, 68, 68, 0.4); - opacity: 1; } - .tabs-color-dark .tab-item .badge { - opacity: 0.4; } - .tabs-color-dark .tab-item.tab-item-active, .tabs-color-dark .tab-item.active, .tabs-color-dark .tab-item.activated { - color: #444; - border: 0 solid #444; } - .tabs-color-dark .tab-item.tab-item-active .badge, .tabs-color-dark .tab-item.active .badge, .tabs-color-dark .tab-item.activated .badge { - opacity: 1; } - - ion-tabs.tabs-color-active-light .tab-item { - color: #444; } - ion-tabs.tabs-color-active-light .tab-item.tab-item-active, ion-tabs.tabs-color-active-light .tab-item.active, ion-tabs.tabs-color-active-light .tab-item.activated { - color: #fff; } - - ion-tabs.tabs-striped.tabs-color-active-light .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-light .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-light .tab-item.activated { - border-color: #fff; - color: #fff; } - - ion-tabs.tabs-color-active-stable .tab-item { - color: #444; } - ion-tabs.tabs-color-active-stable .tab-item.tab-item-active, ion-tabs.tabs-color-active-stable .tab-item.active, ion-tabs.tabs-color-active-stable .tab-item.activated { - color: #f8f8f8; } - - ion-tabs.tabs-striped.tabs-color-active-stable .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-stable .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-stable .tab-item.activated { - border-color: #f8f8f8; - color: #f8f8f8; } - - ion-tabs.tabs-color-active-positive .tab-item { - color: #444; } - ion-tabs.tabs-color-active-positive .tab-item.tab-item-active, ion-tabs.tabs-color-active-positive .tab-item.active, ion-tabs.tabs-color-active-positive .tab-item.activated { - color: #387ef5; } - - ion-tabs.tabs-striped.tabs-color-active-positive .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-positive .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-positive .tab-item.activated { - border-color: #387ef5; - color: #387ef5; } - - ion-tabs.tabs-color-active-calm .tab-item { - color: #444; } - ion-tabs.tabs-color-active-calm .tab-item.tab-item-active, ion-tabs.tabs-color-active-calm .tab-item.active, ion-tabs.tabs-color-active-calm .tab-item.activated { - color: #11c1f3; } - - ion-tabs.tabs-striped.tabs-color-active-calm .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-calm .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-calm .tab-item.activated { - border-color: #11c1f3; - color: #11c1f3; } - - ion-tabs.tabs-color-active-assertive .tab-item { - color: #444; } - ion-tabs.tabs-color-active-assertive .tab-item.tab-item-active, ion-tabs.tabs-color-active-assertive .tab-item.active, ion-tabs.tabs-color-active-assertive .tab-item.activated { - color: #ef473a; } - - ion-tabs.tabs-striped.tabs-color-active-assertive .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-assertive .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-assertive .tab-item.activated { - border-color: #ef473a; - color: #ef473a; } - - ion-tabs.tabs-color-active-balanced .tab-item { - color: #444; } - ion-tabs.tabs-color-active-balanced .tab-item.tab-item-active, ion-tabs.tabs-color-active-balanced .tab-item.active, ion-tabs.tabs-color-active-balanced .tab-item.activated { - color: #33cd5f; } - - ion-tabs.tabs-striped.tabs-color-active-balanced .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-balanced .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-balanced .tab-item.activated { - border-color: #33cd5f; - color: #33cd5f; } - - ion-tabs.tabs-color-active-energized .tab-item { - color: #444; } - ion-tabs.tabs-color-active-energized .tab-item.tab-item-active, ion-tabs.tabs-color-active-energized .tab-item.active, ion-tabs.tabs-color-active-energized .tab-item.activated { - color: #ffc900; } - - ion-tabs.tabs-striped.tabs-color-active-energized .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-energized .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-energized .tab-item.activated { - border-color: #ffc900; - color: #ffc900; } - - ion-tabs.tabs-color-active-royal .tab-item { - color: #444; } - ion-tabs.tabs-color-active-royal .tab-item.tab-item-active, ion-tabs.tabs-color-active-royal .tab-item.active, ion-tabs.tabs-color-active-royal .tab-item.activated { - color: #886aea; } - - ion-tabs.tabs-striped.tabs-color-active-royal .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-royal .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-royal .tab-item.activated { - border-color: #886aea; - color: #886aea; } - - ion-tabs.tabs-color-active-dark .tab-item { - color: #fff; } - ion-tabs.tabs-color-active-dark .tab-item.tab-item-active, ion-tabs.tabs-color-active-dark .tab-item.active, ion-tabs.tabs-color-active-dark .tab-item.activated { - color: #444; } - - ion-tabs.tabs-striped.tabs-color-active-dark .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-dark .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-dark .tab-item.activated { - border-color: #444; - color: #444; } - - .tabs-top.tabs-striped { - padding-bottom: 0; } - .tabs-top.tabs-striped .tab-item { - background: transparent; - -webkit-transition: color .1s ease; - -moz-transition: color .1s ease; - -ms-transition: color .1s ease; - -o-transition: color .1s ease; - transition: color .1s ease; } - .tabs-top.tabs-striped .tab-item.tab-item-active, .tabs-top.tabs-striped .tab-item.active, .tabs-top.tabs-striped .tab-item.activated { - margin-top: 1px; - border-width: 0px 0px 2px 0px !important; - border-style: solid; } - .tabs-top.tabs-striped .tab-item.tab-item-active > .badge, .tabs-top.tabs-striped .tab-item.tab-item-active > i, .tabs-top.tabs-striped .tab-item.active > .badge, .tabs-top.tabs-striped .tab-item.active > i, .tabs-top.tabs-striped .tab-item.activated > .badge, .tabs-top.tabs-striped .tab-item.activated > i { - margin-top: -1px; } - .tabs-top.tabs-striped .tab-item .badge { - -webkit-transition: color .2s ease; - -moz-transition: color .2s ease; - -ms-transition: color .2s ease; - -o-transition: color .2s ease; - transition: color .2s ease; } - .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.tab-item-active .tab-title, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.tab-item-active i, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.active .tab-title, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.active i, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.activated .tab-title, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.activated i { - display: block; - margin-top: -1px; } - .tabs-top.tabs-striped.tabs-icon-left .tab-item { - margin-top: 1px; } - .tabs-top.tabs-striped.tabs-icon-left .tab-item.tab-item-active .tab-title, .tabs-top.tabs-striped.tabs-icon-left .tab-item.tab-item-active i, .tabs-top.tabs-striped.tabs-icon-left .tab-item.active .tab-title, .tabs-top.tabs-striped.tabs-icon-left .tab-item.active i, .tabs-top.tabs-striped.tabs-icon-left .tab-item.activated .tab-title, .tabs-top.tabs-striped.tabs-icon-left .tab-item.activated i { - margin-top: -0.1em; } - - /* Allow parent element to have tabs-top */ - /* If you change this, change platform.scss as well */ - .tabs-top > .tabs, - .tabs.tabs-top { - top: 44px; - padding-top: 0; - background-position: bottom; - border-top-width: 0; - border-bottom-width: 1px; } - .tabs-top > .tabs .tab-item.tab-item-active .badge, .tabs-top > .tabs .tab-item.active .badge, .tabs-top > .tabs .tab-item.activated .badge, - .tabs.tabs-top .tab-item.tab-item-active .badge, - .tabs.tabs-top .tab-item.active .badge, - .tabs.tabs-top .tab-item.activated .badge { - top: 4%; } - - .tabs-top ~ .bar-header { - border-bottom-width: 0; } - - .tab-item { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - display: block; - overflow: hidden; - max-width: 150px; - height: 100%; - color: inherit; - text-align: center; - text-decoration: none; - text-overflow: ellipsis; - white-space: nowrap; - font-weight: 400; - font-size: 14px; - font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; - opacity: 0.7; } - .tab-item:hover { - cursor: pointer; } - .tab-item.tab-hidden { - display: none; } - - .tabs-item-hide > .tabs, - .tabs.tabs-item-hide { - display: none; } - - .tabs-icon-top > .tabs .tab-item, - .tabs-icon-top.tabs .tab-item, - .tabs-icon-bottom > .tabs .tab-item, - .tabs-icon-bottom.tabs .tab-item { - font-size: 10px; - line-height: 14px; } - - .tab-item .icon, .tab-item .icon-help, .tab-item .icon-alert, .tab-item #menu .footer .icon-help, #menu .footer .tab-item .icon-help { - display: block; - margin: 0 auto; - height: 32px; - font-size: 32px; } - - .tabs-icon-left.tabs .tab-item, - .tabs-icon-left > .tabs .tab-item, - .tabs-icon-right.tabs .tab-item, - .tabs-icon-right > .tabs .tab-item { - font-size: 10px; } - .tabs-icon-left.tabs .tab-item .icon, .tabs-icon-left.tabs .tab-item .icon-help, .tabs-icon-left.tabs .tab-item .icon-alert, .tabs-icon-left.tabs .tab-item #menu .footer .icon-help, #menu .footer .tabs-icon-left.tabs .tab-item .icon-help, .tabs-icon-left.tabs .tab-item .tab-title, - .tabs-icon-left > .tabs .tab-item .icon, - .tabs-icon-left > .tabs .tab-item .icon-help, - .tabs-icon-left > .tabs .tab-item .icon-alert, - .tabs-icon-left > .tabs .tab-item #menu .footer .icon-help, #menu .footer - .tabs-icon-left > .tabs .tab-item .icon-help, - .tabs-icon-left > .tabs .tab-item .tab-title, - .tabs-icon-right.tabs .tab-item .icon, - .tabs-icon-right.tabs .tab-item .icon-help, - .tabs-icon-right.tabs .tab-item .icon-alert, - .tabs-icon-right.tabs .tab-item #menu .footer .icon-help, #menu .footer - .tabs-icon-right.tabs .tab-item .icon-help, - .tabs-icon-right.tabs .tab-item .tab-title, - .tabs-icon-right > .tabs .tab-item .icon, - .tabs-icon-right > .tabs .tab-item .icon-help, - .tabs-icon-right > .tabs .tab-item .icon-alert, - .tabs-icon-right > .tabs .tab-item #menu .footer .icon-help, #menu .footer - .tabs-icon-right > .tabs .tab-item .icon-help, - .tabs-icon-right > .tabs .tab-item .tab-title { - display: inline-block; - vertical-align: top; - margin-top: -.1em; } - .tabs-icon-left.tabs .tab-item .icon:before, .tabs-icon-left.tabs .tab-item .icon-help:before, .tabs-icon-left.tabs .tab-item .icon-alert:before, .tabs-icon-left.tabs .tab-item #menu .footer .icon-help:before, #menu .footer .tabs-icon-left.tabs .tab-item .icon-help:before, .tabs-icon-left.tabs .tab-item .tab-title:before, - .tabs-icon-left > .tabs .tab-item .icon:before, - .tabs-icon-left > .tabs .tab-item .icon-help:before, - .tabs-icon-left > .tabs .tab-item .icon-alert:before, - .tabs-icon-left > .tabs .tab-item #menu .footer .icon-help:before, #menu .footer - .tabs-icon-left > .tabs .tab-item .icon-help:before, - .tabs-icon-left > .tabs .tab-item .tab-title:before, - .tabs-icon-right.tabs .tab-item .icon:before, - .tabs-icon-right.tabs .tab-item .icon-help:before, - .tabs-icon-right.tabs .tab-item .icon-alert:before, - .tabs-icon-right.tabs .tab-item #menu .footer .icon-help:before, #menu .footer - .tabs-icon-right.tabs .tab-item .icon-help:before, - .tabs-icon-right.tabs .tab-item .tab-title:before, - .tabs-icon-right > .tabs .tab-item .icon:before, - .tabs-icon-right > .tabs .tab-item .icon-help:before, - .tabs-icon-right > .tabs .tab-item .icon-alert:before, - .tabs-icon-right > .tabs .tab-item #menu .footer .icon-help:before, #menu .footer - .tabs-icon-right > .tabs .tab-item .icon-help:before, - .tabs-icon-right > .tabs .tab-item .tab-title:before { - font-size: 24px; - line-height: 49px; } - - .tabs-icon-left > .tabs .tab-item .icon, .tabs-icon-left > .tabs .tab-item .icon-help, .tabs-icon-left > .tabs .tab-item .icon-alert, .tabs-icon-left > .tabs .tab-item #menu .footer .icon-help, #menu .footer .tabs-icon-left > .tabs .tab-item .icon-help, - .tabs-icon-left.tabs .tab-item .icon, - .tabs-icon-left.tabs .tab-item .icon-help, - .tabs-icon-left.tabs .tab-item .icon-alert, - .tabs-icon-left.tabs .tab-item #menu .footer .icon-help, #menu .footer - .tabs-icon-left.tabs .tab-item .icon-help { - padding-right: 3px; } - - .tabs-icon-right > .tabs .tab-item .icon, .tabs-icon-right > .tabs .tab-item .icon-help, .tabs-icon-right > .tabs .tab-item .icon-alert, .tabs-icon-right > .tabs .tab-item #menu .footer .icon-help, #menu .footer .tabs-icon-right > .tabs .tab-item .icon-help, - .tabs-icon-right.tabs .tab-item .icon, - .tabs-icon-right.tabs .tab-item .icon-help, - .tabs-icon-right.tabs .tab-item .icon-alert, - .tabs-icon-right.tabs .tab-item #menu .footer .icon-help, #menu .footer - .tabs-icon-right.tabs .tab-item .icon-help { - padding-left: 3px; } - - .tabs-icon-only > .tabs .icon, .tabs-icon-only > .tabs .icon-help, .tabs-icon-only > .tabs .icon-alert, .tabs-icon-only > .tabs #menu .footer .icon-help, #menu .footer .tabs-icon-only > .tabs .icon-help, - .tabs-icon-only.tabs .icon, - .tabs-icon-only.tabs .icon-help, - .tabs-icon-only.tabs .icon-alert, - .tabs-icon-only.tabs #menu .footer .icon-help, #menu .footer - .tabs-icon-only.tabs .icon-help { - line-height: inherit; } - - .tab-item.has-badge { - position: relative; } - - .tab-item .badge { - position: absolute; - top: 4%; - right: 33%; - right: calc(50% - 26px); - padding: 1px 6px; - height: auto; - font-size: 12px; - line-height: 16px; } - - /* Navigational tab */ - /* Active state for tab */ - .tab-item.tab-item-active, - .tab-item.active, - .tab-item.activated { - opacity: 1; } - .tab-item.tab-item-active.tab-item-light, - .tab-item.active.tab-item-light, - .tab-item.activated.tab-item-light { - color: #fff; } - .tab-item.tab-item-active.tab-item-stable, - .tab-item.active.tab-item-stable, - .tab-item.activated.tab-item-stable { - color: #f8f8f8; } - .tab-item.tab-item-active.tab-item-positive, - .tab-item.active.tab-item-positive, - .tab-item.activated.tab-item-positive { - color: #387ef5; } - .tab-item.tab-item-active.tab-item-calm, - .tab-item.active.tab-item-calm, - .tab-item.activated.tab-item-calm { - color: #11c1f3; } - .tab-item.tab-item-active.tab-item-assertive, - .tab-item.active.tab-item-assertive, - .tab-item.activated.tab-item-assertive { - color: #ef473a; } - .tab-item.tab-item-active.tab-item-balanced, - .tab-item.active.tab-item-balanced, - .tab-item.activated.tab-item-balanced { - color: #33cd5f; } - .tab-item.tab-item-active.tab-item-energized, - .tab-item.active.tab-item-energized, - .tab-item.activated.tab-item-energized { - color: #ffc900; } - .tab-item.tab-item-active.tab-item-royal, - .tab-item.active.tab-item-royal, - .tab-item.activated.tab-item-royal { - color: #886aea; } - .tab-item.tab-item-active.tab-item-dark, - .tab-item.active.tab-item-dark, - .tab-item.activated.tab-item-dark { - color: #444; } - - .item.tabs { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - padding: 0; } - .item.tabs .icon:before, .item.tabs .icon-help:before, .item.tabs .icon-alert:before, .item.tabs #menu .footer .icon-help:before, #menu .footer .item.tabs .icon-help:before { - position: relative; } - - .tab-item.disabled, - .tab-item[disabled] { - opacity: .4; - cursor: default; - pointer-events: none; } - - .nav-bar-tabs-top.hide ~ .view-container .tabs-top .tabs { - top: 0; } - - .pane[hide-nav-bar="true"] .has-tabs-top { - top: 49px; } - - /** - * Menus - * -------------------------------------------------- - * Side panel structure - */ - .menu { - position: absolute; - top: 0; - bottom: 0; - z-index: 0; - overflow: hidden; - min-height: 100%; - max-height: 100%; - width: 275px; - background-color: #fff; } - .menu .scroll-content { - z-index: 10; } - .menu .bar-header { - z-index: 11; } - - .menu-content { - -webkit-transform: none; - transform: none; - box-shadow: -1px 0px 2px rgba(0, 0, 0, 0.2), 1px 0px 2px rgba(0, 0, 0, 0.2); } - - .menu-open .menu-content .pane, - .menu-open .menu-content .scroll-content { - pointer-events: none; } - - .menu-open .menu-content .scroll-content .scroll { - pointer-events: none; } - - .menu-open .menu-content .scroll-content:not(.overflow-scroll) { - overflow: hidden; } - - .grade-b .menu-content, - .grade-c .menu-content { - -webkit-box-sizing: content-box; - -moz-box-sizing: content-box; - box-sizing: content-box; - right: -1px; - left: -1px; - border-right: 1px solid #ccc; - border-left: 1px solid #ccc; - box-shadow: none; } - - .menu-left { - left: 0; } - - .menu-right { - right: 0; } - - .aside-open.aside-resizing .menu-right { - display: none; } - - .menu-animated { - -webkit-transition: -webkit-transform 200ms ease; - transition: transform 200ms ease; } - - /** - * Modals - * -------------------------------------------------- - * Modals are independent windows that slide in from off-screen. - */ - .modal-backdrop, - .modal-backdrop-bg { - position: fixed; - top: 0; - left: 0; - z-index: 10; - width: 100%; - height: 100%; } - - .modal-backdrop-bg { - pointer-events: none; } - - .modal { - display: block; - position: absolute; - top: 0; - z-index: 10; - overflow: hidden; - min-height: 100%; - width: 100%; - background-color: #fff; } - - @media (min-width: 680px) { - .modal { - top: 20%; - right: 20%; - bottom: 20%; - left: 20%; - min-height: 240px; - width: 60%; } - .modal.ng-leave-active { - bottom: 0; } - .platform-ios.platform-cordova .modal-wrapper .modal .bar-header:not(.bar-subheader) { - height: 44px; } - .platform-ios.platform-cordova .modal-wrapper .modal .bar-header:not(.bar-subheader) > * { - margin-top: 0; } - .platform-ios.platform-cordova .modal-wrapper .modal .tabs-top > .tabs, - .platform-ios.platform-cordova .modal-wrapper .modal .tabs.tabs-top { - top: 44px; } - .platform-ios.platform-cordova .modal-wrapper .modal .has-header, - .platform-ios.platform-cordova .modal-wrapper .modal .bar-subheader { - top: 44px; } - .platform-ios.platform-cordova .modal-wrapper .modal .has-subheader { - top: 88px; } - .platform-ios.platform-cordova .modal-wrapper .modal .has-header.has-tabs-top { - top: 93px; } - .platform-ios.platform-cordova .modal-wrapper .modal .has-header.has-subheader.has-tabs-top { - top: 137px; } - .modal-backdrop-bg { - -webkit-transition: opacity 300ms ease-in-out; - transition: opacity 300ms ease-in-out; - background-color: #000; - opacity: 0; } - .active .modal-backdrop-bg { - opacity: 0.5; } } - - .modal-open { - pointer-events: none; } - .modal-open .modal, - .modal-open .modal-backdrop { - pointer-events: auto; } - .modal-open.loading-active .modal, - .modal-open.loading-active .modal-backdrop { - pointer-events: none; } - - /** - * Popovers - * -------------------------------------------------- - * Popovers are independent views which float over content - */ - .popover-backdrop { - position: fixed; - top: 0; - left: 0; - z-index: 10; - width: 100%; - height: 100%; - background-color: transparent; } - .popover-backdrop.active { - background-color: rgba(0, 0, 0, 0.1); } - - .popover { - position: absolute; - top: 25%; - left: 50%; - z-index: 10; - display: block; - margin-top: 12px; - margin-left: -110px; - height: 280px; - width: 220px; - background-color: #fff; - box-shadow: 0 1px 3px rgba(0, 0, 0, 0.4); - opacity: 0; } - .popover .item:first-child { - border-top: 0; } - .popover .item:last-child { - border-bottom: 0; } - .popover.popover-bottom { - margin-top: -12px; } - - .popover, - .popover .bar-header { - border-radius: 2px; } - - .popover .scroll-content { - z-index: 1; - margin: 2px 0; } - - .popover .bar-header { - border-bottom-right-radius: 0; - border-bottom-left-radius: 0; } - - .popover .has-header { - border-top-right-radius: 0; - border-top-left-radius: 0; } - - .popover-arrow { - display: none; } - - .platform-ios .popover { - box-shadow: 0 0 40px rgba(0, 0, 0, 0.08); - border-radius: 10px; } - - .platform-ios .popover .bar-header { - -webkit-border-top-right-radius: 10px; - border-top-right-radius: 10px; - -webkit-border-top-left-radius: 10px; - border-top-left-radius: 10px; } - - .platform-ios .popover .scroll-content { - margin: 8px 0; - border-radius: 10px; } - - .platform-ios .popover .scroll-content.has-header { - margin-top: 0; } - - .platform-ios .popover-arrow { - position: absolute; - display: block; - top: -17px; - width: 30px; - height: 19px; - overflow: hidden; } - .platform-ios .popover-arrow:after { - position: absolute; - top: 12px; - left: 5px; - width: 20px; - height: 20px; - background-color: #fff; - border-radius: 3px; - content: ''; - -webkit-transform: rotate(-45deg); - transform: rotate(-45deg); } - - .platform-ios .popover-bottom .popover-arrow { - top: auto; - bottom: -10px; } - .platform-ios .popover-bottom .popover-arrow:after { - top: -6px; } - - .platform-android .popover { - margin-top: -32px; - background-color: #fafafa; - box-shadow: 0 2px 6px rgba(0, 0, 0, 0.35); } - .platform-android .popover .item { - border-color: #fafafa; - background-color: #fafafa; - color: #4d4d4d; } - .platform-android .popover.popover-bottom { - margin-top: 32px; } - - .platform-android .popover-backdrop, - .platform-android .popover-backdrop.active { - background-color: transparent; } - - .popover-open { - pointer-events: none; } - .popover-open .popover, - .popover-open .popover-backdrop { - pointer-events: auto; } - .popover-open.loading-active .popover, - .popover-open.loading-active .popover-backdrop { - pointer-events: none; } - - @media (min-width: 680px) { - .popover { - width: 360px; - margin-left: -180px; } } - - /** - * Popups - * -------------------------------------------------- - */ - .popup-container { - position: absolute; - top: 0; - left: 0; - bottom: 0; - right: 0; - background: transparent; - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-pack: center; - -ms-flex-pack: center; - -webkit-justify-content: center; - -moz-justify-content: center; - justify-content: center; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - z-index: 12; - visibility: hidden; } - .popup-container.popup-showing { - visibility: visible; } - .popup-container.popup-hidden .popup { - -webkit-animation-name: scaleOut; - animation-name: scaleOut; - -webkit-animation-duration: 0.1s; - animation-duration: 0.1s; - -webkit-animation-timing-function: ease-in-out; - animation-timing-function: ease-in-out; - -webkit-animation-fill-mode: both; - animation-fill-mode: both; } - .popup-container.active .popup { - -webkit-animation-name: superScaleIn; - animation-name: superScaleIn; - -webkit-animation-duration: 0.2s; - animation-duration: 0.2s; - -webkit-animation-timing-function: ease-in-out; - animation-timing-function: ease-in-out; - -webkit-animation-fill-mode: both; - animation-fill-mode: both; } - .popup-container .popup { - width: 250px; - max-width: 100%; - max-height: 90%; - border-radius: 0px; - background-color: rgba(255, 255, 255, 0.9); - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-direction: normal; - -webkit-box-orient: vertical; - -webkit-flex-direction: column; - -moz-flex-direction: column; - -ms-flex-direction: column; - flex-direction: column; } - .popup-container input, - .popup-container textarea { - width: 100%; } - - .popup-head { - padding: 15px 10px; - border-bottom: 1px solid #eee; - text-align: center; } - - .popup-title { - margin: 0; - padding: 0; - font-size: 15px; } - - .popup-sub-title { - margin: 5px 0 0 0; - padding: 0; - font-weight: normal; - font-size: 11px; } - - .popup-body { - padding: 10px; - overflow: auto; } - - .popup-buttons { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-direction: normal; - -webkit-box-orient: horizontal; - -webkit-flex-direction: row; - -moz-flex-direction: row; - -ms-flex-direction: row; - flex-direction: row; - padding: 10px; - min-height: 65px; } - .popup-buttons .button { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - display: block; - min-height: 45px; - border-radius: 2px; - line-height: 20px; - margin-right: 5px; } - .popup-buttons .button:last-child { - margin-right: 0px; } - - .popup-open { - pointer-events: none; } - .popup-open.modal-open .modal { - pointer-events: none; } - .popup-open .popup-backdrop, .popup-open .popup { - pointer-events: auto; } - - /** - * Loading - * -------------------------------------------------- - */ - .loading-container { - position: absolute; - left: 0; - top: 0; - right: 0; - bottom: 0; - z-index: 13; - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-pack: center; - -ms-flex-pack: center; - -webkit-justify-content: center; - -moz-justify-content: center; - justify-content: center; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - -webkit-transition: 0.2s opacity linear; - transition: 0.2s opacity linear; - visibility: hidden; - opacity: 0; } - .loading-container:not(.visible) .icon, .loading-container:not(.visible) .icon-help, .loading-container:not(.visible) .icon-alert, .loading-container:not(.visible) #menu .footer .icon-help, #menu .footer .loading-container:not(.visible) .icon-help, - .loading-container:not(.visible) .spinner { - display: none; } - .loading-container.visible { - visibility: visible; } - .loading-container.active { - opacity: 1; } - .loading-container .loading { - padding: 20px; - border-radius: 5px; - background-color: rgba(0, 0, 0, 0.7); - color: #fff; - text-align: center; - text-overflow: ellipsis; - font-size: 15px; } - .loading-container .loading h1, .loading-container .loading h2, .loading-container .loading h3, .loading-container .loading h4, .loading-container .loading h5, .loading-container .loading h6 { - color: #fff; } - - /** - * Items - * -------------------------------------------------- - */ - .item { - border-color: #ddd; - background-color: #fff; - color: #444; - position: relative; - z-index: 2; - display: block; - margin: -1px; - padding: 16px; - border-width: 1px; - border-style: solid; - font-size: 16px; } - .item h2 { - margin: 0 0 2px 0; - font-size: 16px; - font-weight: normal; } - .item h3 { - margin: 0 0 4px 0; - font-size: 14px; } - .item h4 { - margin: 0 0 4px 0; - font-size: 12px; } - .item h5, .item h6 { - margin: 0 0 3px 0; - font-size: 10px; } - .item p { - color: #666; - font-size: 14px; - margin-bottom: 2px; } - .item h1:last-child, - .item h2:last-child, - .item h3:last-child, - .item h4:last-child, - .item h5:last-child, - .item h6:last-child, - .item p:last-child { - margin-bottom: 0; } - .item .badge { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - position: absolute; - top: 16px; - right: 32px; } - .item.item-button-right .badge { - right: 67px; } - .item.item-divider .badge { - top: 8px; } - .item .badge + .badge { - margin-right: 5px; } - .item.item-light { - border-color: #ddd; - background-color: #fff; - color: #444; } - .item.item-stable { - border-color: #b2b2b2; - background-color: #f8f8f8; - color: #444; } - .item.item-positive { - border-color: #0c60ee; - background-color: #387ef5; - color: #fff; } - .item.item-calm { - border-color: #0a9dc7; - background-color: #11c1f3; - color: #fff; } - .item.item-assertive { - border-color: #e42112; - background-color: #ef473a; - color: #fff; } - .item.item-balanced { - border-color: #28a54c; - background-color: #33cd5f; - color: #fff; } - .item.item-energized { - border-color: #e6b500; - background-color: #ffc900; - color: #fff; } - .item.item-royal { - border-color: #6b46e5; - background-color: #886aea; - color: #fff; } - .item.item-dark { - border-color: #111; - background-color: #444; - color: #fff; } - .item[ng-click]:hover { - cursor: pointer; } - - .list-borderless .item, - .item-borderless { - border-width: 0; } - - .item.active, - .item.activated, - .item-complex.active .item-content, - .item-complex.activated .item-content, - .item .item-content.active, - .item .item-content.activated { - border-color: #ccc; - background-color: #D9D9D9; } - .item.active.item-complex > .item-content, - .item.activated.item-complex > .item-content, - .item-complex.active .item-content.item-complex > .item-content, - .item-complex.activated .item-content.item-complex > .item-content, - .item .item-content.active.item-complex > .item-content, - .item .item-content.activated.item-complex > .item-content { - border-color: #ccc; - background-color: #D9D9D9; } - .item.active.item-light, - .item.activated.item-light, - .item-complex.active .item-content.item-light, - .item-complex.activated .item-content.item-light, - .item .item-content.active.item-light, - .item .item-content.activated.item-light { - border-color: #ccc; - background-color: #fafafa; } - .item.active.item-light.item-complex > .item-content, - .item.activated.item-light.item-complex > .item-content, - .item-complex.active .item-content.item-light.item-complex > .item-content, - .item-complex.activated .item-content.item-light.item-complex > .item-content, - .item .item-content.active.item-light.item-complex > .item-content, - .item .item-content.activated.item-light.item-complex > .item-content { - border-color: #ccc; - background-color: #fafafa; } - .item.active.item-stable, - .item.activated.item-stable, - .item-complex.active .item-content.item-stable, - .item-complex.activated .item-content.item-stable, - .item .item-content.active.item-stable, - .item .item-content.activated.item-stable { - border-color: #a2a2a2; - background-color: #e5e5e5; } - .item.active.item-stable.item-complex > .item-content, - .item.activated.item-stable.item-complex > .item-content, - .item-complex.active .item-content.item-stable.item-complex > .item-content, - .item-complex.activated .item-content.item-stable.item-complex > .item-content, - .item .item-content.active.item-stable.item-complex > .item-content, - .item .item-content.activated.item-stable.item-complex > .item-content { - border-color: #a2a2a2; - background-color: #e5e5e5; } - .item.active.item-positive, - .item.activated.item-positive, - .item-complex.active .item-content.item-positive, - .item-complex.activated .item-content.item-positive, - .item .item-content.active.item-positive, - .item .item-content.activated.item-positive { - border-color: #0c60ee; - background-color: #0c60ee; } - .item.active.item-positive.item-complex > .item-content, - .item.activated.item-positive.item-complex > .item-content, - .item-complex.active .item-content.item-positive.item-complex > .item-content, - .item-complex.activated .item-content.item-positive.item-complex > .item-content, - .item .item-content.active.item-positive.item-complex > .item-content, - .item .item-content.activated.item-positive.item-complex > .item-content { - border-color: #0c60ee; - background-color: #0c60ee; } - .item.active.item-calm, - .item.activated.item-calm, - .item-complex.active .item-content.item-calm, - .item-complex.activated .item-content.item-calm, - .item .item-content.active.item-calm, - .item .item-content.activated.item-calm { - border-color: #0a9dc7; - background-color: #0a9dc7; } - .item.active.item-calm.item-complex > .item-content, - .item.activated.item-calm.item-complex > .item-content, - .item-complex.active .item-content.item-calm.item-complex > .item-content, - .item-complex.activated .item-content.item-calm.item-complex > .item-content, - .item .item-content.active.item-calm.item-complex > .item-content, - .item .item-content.activated.item-calm.item-complex > .item-content { - border-color: #0a9dc7; - background-color: #0a9dc7; } - .item.active.item-assertive, - .item.activated.item-assertive, - .item-complex.active .item-content.item-assertive, - .item-complex.activated .item-content.item-assertive, - .item .item-content.active.item-assertive, - .item .item-content.activated.item-assertive { - border-color: #e42112; - background-color: #e42112; } - .item.active.item-assertive.item-complex > .item-content, - .item.activated.item-assertive.item-complex > .item-content, - .item-complex.active .item-content.item-assertive.item-complex > .item-content, - .item-complex.activated .item-content.item-assertive.item-complex > .item-content, - .item .item-content.active.item-assertive.item-complex > .item-content, - .item .item-content.activated.item-assertive.item-complex > .item-content { - border-color: #e42112; - background-color: #e42112; } - .item.active.item-balanced, - .item.activated.item-balanced, - .item-complex.active .item-content.item-balanced, - .item-complex.activated .item-content.item-balanced, - .item .item-content.active.item-balanced, - .item .item-content.activated.item-balanced { - border-color: #28a54c; - background-color: #28a54c; } - .item.active.item-balanced.item-complex > .item-content, - .item.activated.item-balanced.item-complex > .item-content, - .item-complex.active .item-content.item-balanced.item-complex > .item-content, - .item-complex.activated .item-content.item-balanced.item-complex > .item-content, - .item .item-content.active.item-balanced.item-complex > .item-content, - .item .item-content.activated.item-balanced.item-complex > .item-content { - border-color: #28a54c; - background-color: #28a54c; } - .item.active.item-energized, - .item.activated.item-energized, - .item-complex.active .item-content.item-energized, - .item-complex.activated .item-content.item-energized, - .item .item-content.active.item-energized, - .item .item-content.activated.item-energized { - border-color: #e6b500; - background-color: #e6b500; } - .item.active.item-energized.item-complex > .item-content, - .item.activated.item-energized.item-complex > .item-content, - .item-complex.active .item-content.item-energized.item-complex > .item-content, - .item-complex.activated .item-content.item-energized.item-complex > .item-content, - .item .item-content.active.item-energized.item-complex > .item-content, - .item .item-content.activated.item-energized.item-complex > .item-content { - border-color: #e6b500; - background-color: #e6b500; } - .item.active.item-royal, - .item.activated.item-royal, - .item-complex.active .item-content.item-royal, - .item-complex.activated .item-content.item-royal, - .item .item-content.active.item-royal, - .item .item-content.activated.item-royal { - border-color: #6b46e5; - background-color: #6b46e5; } - .item.active.item-royal.item-complex > .item-content, - .item.activated.item-royal.item-complex > .item-content, - .item-complex.active .item-content.item-royal.item-complex > .item-content, - .item-complex.activated .item-content.item-royal.item-complex > .item-content, - .item .item-content.active.item-royal.item-complex > .item-content, - .item .item-content.activated.item-royal.item-complex > .item-content { - border-color: #6b46e5; - background-color: #6b46e5; } - .item.active.item-dark, - .item.activated.item-dark, - .item-complex.active .item-content.item-dark, - .item-complex.activated .item-content.item-dark, - .item .item-content.active.item-dark, - .item .item-content.activated.item-dark { - border-color: #000; - background-color: #262626; } - .item.active.item-dark.item-complex > .item-content, - .item.activated.item-dark.item-complex > .item-content, - .item-complex.active .item-content.item-dark.item-complex > .item-content, - .item-complex.activated .item-content.item-dark.item-complex > .item-content, - .item .item-content.active.item-dark.item-complex > .item-content, - .item .item-content.activated.item-dark.item-complex > .item-content { - border-color: #000; - background-color: #262626; } - - .item, - .item h1, - .item h2, - .item h3, - .item h4, - .item h5, - .item h6, - .item p, - .item-content, - .item-content h1, - .item-content h2, - .item-content h3, - .item-content h4, - .item-content h5, - .item-content h6, - .item-content p { - overflow: hidden; - text-overflow: ellipsis; - white-space: nowrap; } - - a.item { - color: inherit; - text-decoration: none; } - a.item:hover, a.item:focus { - text-decoration: none; } - - /** - * Complex Items - * -------------------------------------------------- - * Adding .item-complex allows the .item to be slidable and - * have options underneath the button, but also requires an - * additional .item-content element inside .item. - * Basically .item-complex removes any default settings which - * .item added, so that .item-content looks them as just .item. - */ - .item-complex, - a.item.item-complex, - button.item.item-complex { - padding: 0; } - - .item-complex .item-content, - .item-radio .item-content { - position: relative; - z-index: 2; - padding: 16px 49px 16px 16px; - border: none; - background-color: #fff; } - - a.item-content { - display: block; - color: inherit; - text-decoration: none; } - - .item-text-wrap .item, - .item-text-wrap .item-content, - .item-text-wrap, - .item-text-wrap h1, - .item-text-wrap h2, - .item-text-wrap h3, - .item-text-wrap h4, - .item-text-wrap h5, - .item-text-wrap h6, - .item-text-wrap p, - .item-complex.item-text-wrap .item-content, - .item-body h1, - .item-body h2, - .item-body h3, - .item-body h4, - .item-body h5, - .item-body h6, - .item-body p { - overflow: visible; - white-space: normal; } - - .item-complex.item-text-wrap, - .item-complex.item-text-wrap h1, - .item-complex.item-text-wrap h2, - .item-complex.item-text-wrap h3, - .item-complex.item-text-wrap h4, - .item-complex.item-text-wrap h5, - .item-complex.item-text-wrap h6, - .item-complex.item-text-wrap p { - overflow: visible; - white-space: normal; } - - .item-complex.item-light > .item-content { - border-color: #ddd; - background-color: #fff; - color: #444; } - .item-complex.item-light > .item-content.active, .item-complex.item-light > .item-content:active { - border-color: #ccc; - background-color: #fafafa; } - .item-complex.item-light > .item-content.active.item-complex > .item-content, .item-complex.item-light > .item-content:active.item-complex > .item-content { - border-color: #ccc; - background-color: #fafafa; } - - .item-complex.item-stable > .item-content { - border-color: #b2b2b2; - background-color: #f8f8f8; - color: #444; } - .item-complex.item-stable > .item-content.active, .item-complex.item-stable > .item-content:active { - border-color: #a2a2a2; - background-color: #e5e5e5; } - .item-complex.item-stable > .item-content.active.item-complex > .item-content, .item-complex.item-stable > .item-content:active.item-complex > .item-content { - border-color: #a2a2a2; - background-color: #e5e5e5; } - - .item-complex.item-positive > .item-content { - border-color: #0c60ee; - background-color: #387ef5; - color: #fff; } - .item-complex.item-positive > .item-content.active, .item-complex.item-positive > .item-content:active { - border-color: #0c60ee; - background-color: #0c60ee; } - .item-complex.item-positive > .item-content.active.item-complex > .item-content, .item-complex.item-positive > .item-content:active.item-complex > .item-content { - border-color: #0c60ee; - background-color: #0c60ee; } - - .item-complex.item-calm > .item-content { - border-color: #0a9dc7; - background-color: #11c1f3; - color: #fff; } - .item-complex.item-calm > .item-content.active, .item-complex.item-calm > .item-content:active { - border-color: #0a9dc7; - background-color: #0a9dc7; } - .item-complex.item-calm > .item-content.active.item-complex > .item-content, .item-complex.item-calm > .item-content:active.item-complex > .item-content { - border-color: #0a9dc7; - background-color: #0a9dc7; } - - .item-complex.item-assertive > .item-content { - border-color: #e42112; - background-color: #ef473a; - color: #fff; } - .item-complex.item-assertive > .item-content.active, .item-complex.item-assertive > .item-content:active { - border-color: #e42112; - background-color: #e42112; } - .item-complex.item-assertive > .item-content.active.item-complex > .item-content, .item-complex.item-assertive > .item-content:active.item-complex > .item-content { - border-color: #e42112; - background-color: #e42112; } - - .item-complex.item-balanced > .item-content { - border-color: #28a54c; - background-color: #33cd5f; - color: #fff; } - .item-complex.item-balanced > .item-content.active, .item-complex.item-balanced > .item-content:active { - border-color: #28a54c; - background-color: #28a54c; } - .item-complex.item-balanced > .item-content.active.item-complex > .item-content, .item-complex.item-balanced > .item-content:active.item-complex > .item-content { - border-color: #28a54c; - background-color: #28a54c; } - - .item-complex.item-energized > .item-content { - border-color: #e6b500; - background-color: #ffc900; - color: #fff; } - .item-complex.item-energized > .item-content.active, .item-complex.item-energized > .item-content:active { - border-color: #e6b500; - background-color: #e6b500; } - .item-complex.item-energized > .item-content.active.item-complex > .item-content, .item-complex.item-energized > .item-content:active.item-complex > .item-content { - border-color: #e6b500; - background-color: #e6b500; } - - .item-complex.item-royal > .item-content { - border-color: #6b46e5; - background-color: #886aea; - color: #fff; } - .item-complex.item-royal > .item-content.active, .item-complex.item-royal > .item-content:active { - border-color: #6b46e5; - background-color: #6b46e5; } - .item-complex.item-royal > .item-content.active.item-complex > .item-content, .item-complex.item-royal > .item-content:active.item-complex > .item-content { - border-color: #6b46e5; - background-color: #6b46e5; } - - .item-complex.item-dark > .item-content { - border-color: #111; - background-color: #444; - color: #fff; } - .item-complex.item-dark > .item-content.active, .item-complex.item-dark > .item-content:active { - border-color: #000; - background-color: #262626; } - .item-complex.item-dark > .item-content.active.item-complex > .item-content, .item-complex.item-dark > .item-content:active.item-complex > .item-content { - border-color: #000; - background-color: #262626; } - - /** - * Item Icons - * -------------------------------------------------- - */ - .item-icon-left .icon, .item-icon-left .icon-help, .item-icon-left .icon-alert, .item-icon-left #menu .footer .icon-help, #menu .footer .item-icon-left .icon-help, - .item-icon-right .icon, - .item-icon-right .icon-help, - .item-icon-right .icon-alert, - .item-icon-right #menu .footer .icon-help, #menu .footer - .item-icon-right .icon-help { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: absolute; - top: 0; - height: 100%; - font-size: 32px; } - .item-icon-left .icon:before, .item-icon-left .icon-help:before, .item-icon-left .icon-alert:before, .item-icon-left #menu .footer .icon-help:before, #menu .footer .item-icon-left .icon-help:before, - .item-icon-right .icon:before, - .item-icon-right .icon-help:before, - .item-icon-right .icon-alert:before, - .item-icon-right #menu .footer .icon-help:before, #menu .footer - .item-icon-right .icon-help:before { - display: block; - width: 32px; - text-align: center; } - - .item .fill-icon { - min-width: 30px; - min-height: 30px; - font-size: 28px; } - - .item-icon-left { - padding-left: 54px; } - .item-icon-left .icon, .item-icon-left .icon-help, .item-icon-left .icon-alert, .item-icon-left #menu .footer .icon-help, #menu .footer .item-icon-left .icon-help { - left: 11px; } - - .item-complex.item-icon-left { - padding-left: 0; } - .item-complex.item-icon-left .item-content { - padding-left: 54px; } - - .item-icon-right { - padding-right: 54px; } - .item-icon-right .icon, .item-icon-right .icon-help, .item-icon-right .icon-alert, .item-icon-right #menu .footer .icon-help, #menu .footer .item-icon-right .icon-help { - right: 11px; } - - .item-complex.item-icon-right { - padding-right: 0; } - .item-complex.item-icon-right .item-content { - padding-right: 54px; } - - .item-icon-left.item-icon-right .icon:first-child, .item-icon-left.item-icon-right .icon-help:first-child, .item-icon-left.item-icon-right .icon-alert:first-child, .item-icon-left.item-icon-right #menu .footer .icon-help:first-child, #menu .footer .item-icon-left.item-icon-right .icon-help:first-child { - right: auto; } - - .item-icon-left.item-icon-right .icon:last-child, .item-icon-left.item-icon-right .icon-help:last-child, .item-icon-left.item-icon-right .icon-alert:last-child, .item-icon-left.item-icon-right #menu .footer .icon-help:last-child, #menu .footer .item-icon-left.item-icon-right .icon-help:last-child, - .item-icon-left .item-delete .icon, - .item-icon-left .item-delete .icon-help, - .item-icon-left .item-delete .icon-alert, - .item-icon-left .item-delete #menu .footer .icon-help, #menu .footer - .item-icon-left .item-delete .icon-help { - left: auto; } - - .item-icon-left .icon-accessory, - .item-icon-right .icon-accessory { - color: #ccc; - font-size: 16px; } - - .item-icon-left .icon-accessory { - left: 3px; } - - .item-icon-right .icon-accessory { - right: 3px; } - - /** - * Item Button - * -------------------------------------------------- - * An item button is a child button inside an .item (not the entire .item) - */ - .item-button-left { - padding-left: 72px; } - - .item-button-left > .button, - .item-button-left .item-content > .button { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: absolute; - top: 8px; - left: 11px; - min-width: 34px; - min-height: 34px; - font-size: 18px; - line-height: 32px; } - .item-button-left > .button .icon:before, .item-button-left > .button .icon-help:before, .item-button-left > .button .icon-alert:before, .item-button-left > .button #menu .footer .icon-help:before, #menu .footer .item-button-left > .button .icon-help:before, - .item-button-left .item-content > .button .icon:before, - .item-button-left .item-content > .button .icon-help:before, - .item-button-left .item-content > .button .icon-alert:before, - .item-button-left .item-content > .button #menu .footer .icon-help:before, #menu .footer - .item-button-left .item-content > .button .icon-help:before { - position: relative; - left: auto; - width: auto; - line-height: 31px; } - .item-button-left > .button > .button, - .item-button-left .item-content > .button > .button { - margin: 0px 2px; - min-height: 34px; - font-size: 18px; - line-height: 32px; } - - .item-button-right, - a.item.item-button-right, - button.item.item-button-right { - padding-right: 80px; } - - .item-button-right > .button, - .item-button-right .item-content > .button, - .item-button-right > .buttons, - .item-button-right .item-content > .buttons { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: absolute; - top: 8px; - right: 16px; - min-width: 34px; - min-height: 34px; - font-size: 18px; - line-height: 32px; } - .item-button-right > .button .icon:before, .item-button-right > .button .icon-help:before, .item-button-right > .button .icon-alert:before, .item-button-right > .button #menu .footer .icon-help:before, #menu .footer .item-button-right > .button .icon-help:before, - .item-button-right .item-content > .button .icon:before, - .item-button-right .item-content > .button .icon-help:before, - .item-button-right .item-content > .button .icon-alert:before, - .item-button-right .item-content > .button #menu .footer .icon-help:before, #menu .footer - .item-button-right .item-content > .button .icon-help:before, - .item-button-right > .buttons .icon:before, - .item-button-right > .buttons .icon-help:before, - .item-button-right > .buttons .icon-alert:before, - .item-button-right > .buttons #menu .footer .icon-help:before, #menu .footer - .item-button-right > .buttons .icon-help:before, - .item-button-right .item-content > .buttons .icon:before, - .item-button-right .item-content > .buttons .icon-help:before, - .item-button-right .item-content > .buttons .icon-alert:before, - .item-button-right .item-content > .buttons #menu .footer .icon-help:before, #menu .footer - .item-button-right .item-content > .buttons .icon-help:before { - position: relative; - left: auto; - width: auto; - line-height: 31px; } - .item-button-right > .button > .button, - .item-button-right .item-content > .button > .button, - .item-button-right > .buttons > .button, - .item-button-right .item-content > .buttons > .button { - margin: 0px 2px; - min-width: 34px; - min-height: 34px; - font-size: 18px; - line-height: 32px; } - - .item-button-left.item-button-right .button:first-child { - right: auto; } - - .item-button-left.item-button-right .button:last-child { - left: auto; } - - .item-avatar, - .item-avatar .item-content, - .item-avatar-left, - .item-avatar-left .item-content { - padding-left: 72px; - min-height: 72px; } - .item-avatar > img:first-child, - .item-avatar .item-image, - .item-avatar .item-content > img:first-child, - .item-avatar .item-content .item-image, - .item-avatar-left > img:first-child, - .item-avatar-left .item-image, - .item-avatar-left .item-content > img:first-child, - .item-avatar-left .item-content .item-image { - position: absolute; - top: 16px; - left: 16px; - max-width: 40px; - max-height: 40px; - width: 100%; - height: 100%; - border-radius: 50%; } - - .item-avatar-right, - .item-avatar-right .item-content { - padding-right: 72px; - min-height: 72px; } - .item-avatar-right > img:first-child, - .item-avatar-right .item-image, - .item-avatar-right .item-content > img:first-child, - .item-avatar-right .item-content .item-image { - position: absolute; - top: 16px; - right: 16px; - max-width: 40px; - max-height: 40px; - width: 100%; - height: 100%; - border-radius: 50%; } - - .item-thumbnail-left, - .item-thumbnail-left .item-content { - padding-top: 8px; - padding-left: 106px; - min-height: 100px; } - .item-thumbnail-left > img:first-child, - .item-thumbnail-left .item-image, - .item-thumbnail-left .item-content > img:first-child, - .item-thumbnail-left .item-content .item-image { - position: absolute; - top: 10px; - left: 10px; - max-width: 80px; - max-height: 80px; - width: 100%; - height: 100%; } - - .item-avatar.item-complex, - .item-avatar-left.item-complex, - .item-thumbnail-left.item-complex { - padding-top: 0; - padding-left: 0; } - - .item-thumbnail-right, - .item-thumbnail-right .item-content { - padding-top: 8px; - padding-right: 106px; - min-height: 100px; } - .item-thumbnail-right > img:first-child, - .item-thumbnail-right .item-image, - .item-thumbnail-right .item-content > img:first-child, - .item-thumbnail-right .item-content .item-image { - position: absolute; - top: 10px; - right: 10px; - max-width: 80px; - max-height: 80px; - width: 100%; - height: 100%; } - - .item-avatar-right.item-complex, - .item-thumbnail-right.item-complex { - padding-top: 0; - padding-right: 0; } - - .item-image { - padding: 0; - text-align: center; } - .item-image img:first-child, .item-image .list-img { - width: 100%; - vertical-align: middle; } - - .item-body { - overflow: auto; - padding: 16px; - text-overflow: inherit; - white-space: normal; } - .item-body h1, .item-body h2, .item-body h3, .item-body h4, .item-body h5, .item-body h6, .item-body p { - margin-top: 16px; - margin-bottom: 16px; } - - .item-divider { - padding-top: 8px; - padding-bottom: 8px; - min-height: 30px; - background-color: #f5f5f5; - color: #222; - font-weight: 500; } - - .platform-ios .item-divider-platform, - .item-divider-ios { - padding-top: 26px; - text-transform: uppercase; - font-weight: 300; - font-size: 13px; - background-color: #efeff4; - color: #555; } - - .platform-android .item-divider-platform, - .item-divider-android { - font-weight: 300; - font-size: 13px; } - - .item-note { - float: right; - color: #aaa; - font-size: 14px; } - - .item-left-editable .item-content, - .item-right-editable .item-content { - -webkit-transition-duration: 250ms; - transition-duration: 250ms; - -webkit-transition-timing-function: ease-in-out; - transition-timing-function: ease-in-out; - -webkit-transition-property: -webkit-transform; - -moz-transition-property: -moz-transform; - transition-property: transform; } - - .list-left-editing .item-left-editable .item-content, - .item-left-editing.item-left-editable .item-content { - -webkit-transform: translate3d(50px, 0, 0); - transform: translate3d(50px, 0, 0); } - - .item-remove-animate.ng-leave { - -webkit-transition-duration: 300ms; - transition-duration: 300ms; } - - .item-remove-animate.ng-leave .item-content, .item-remove-animate.ng-leave:last-of-type { - -webkit-transition-duration: 300ms; - transition-duration: 300ms; - -webkit-transition-timing-function: ease-in; - transition-timing-function: ease-in; - -webkit-transition-property: all; - transition-property: all; } - - .item-remove-animate.ng-leave.ng-leave-active .item-content { - opacity: 0; - -webkit-transform: translate3d(-100%, 0, 0) !important; - transform: translate3d(-100%, 0, 0) !important; } - - .item-remove-animate.ng-leave.ng-leave-active:last-of-type { - opacity: 0; } - - .item-remove-animate.ng-leave.ng-leave-active ~ ion-item:not(.ng-leave) { - -webkit-transform: translate3d(0, -webkit-calc(-100% + 1px), 0); - transform: translate3d(0, calc(-100% + 1px), 0); - -webkit-transition-duration: 300ms; - transition-duration: 300ms; - -webkit-transition-timing-function: cubic-bezier(0.25, 0.81, 0.24, 1); - transition-timing-function: cubic-bezier(0.25, 0.81, 0.24, 1); - -webkit-transition-property: all; - transition-property: all; } - - .item-left-edit { - -webkit-transition: all ease-in-out 125ms; - transition: all ease-in-out 125ms; - position: absolute; - top: 0; - left: 0; - z-index: 0; - width: 50px; - height: 100%; - line-height: 100%; - display: none; - opacity: 0; - -webkit-transform: translate3d(-21px, 0, 0); - transform: translate3d(-21px, 0, 0); } - .item-left-edit .button { - height: 100%; } - .item-left-edit .button.icon, .item-left-edit .button.icon-help, .item-left-edit .button.icon-alert, .item-left-edit #menu .footer .button.icon-help, #menu .footer .item-left-edit .button.icon-help { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: absolute; - top: 0; - height: 100%; } - .item-left-edit.visible { - display: block; } - .item-left-edit.visible.active { - opacity: 1; - -webkit-transform: translate3d(8px, 0, 0); - transform: translate3d(8px, 0, 0); } - - .list-left-editing .item-left-edit { - -webkit-transition-delay: 125ms; - transition-delay: 125ms; } - - .item-delete .button.icon, .item-delete .button.icon-help, .item-delete .button.icon-alert, .item-delete #menu .footer .button.icon-help, #menu .footer .item-delete .button.icon-help { - color: #ef473a; - font-size: 24px; } - .item-delete .button.icon:hover, .item-delete .button.icon-help:hover, .item-delete .button.icon-alert:hover, .item-delete #menu .footer .button.icon-help:hover, #menu .footer .item-delete .button.icon-help:hover { - opacity: .7; } - - .item-right-edit { - -webkit-transition: all ease-in-out 250ms; - transition: all ease-in-out 250ms; - position: absolute; - top: 0; - right: 0; - z-index: 3; - width: 75px; - height: 100%; - background: inherit; - padding-left: 20px; - display: block; - opacity: 0; - -webkit-transform: translate3d(75px, 0, 0); - transform: translate3d(75px, 0, 0); } - .item-right-edit .button { - min-width: 50px; - height: 100%; } - .item-right-edit .button.icon, .item-right-edit .button.icon-help, .item-right-edit .button.icon-alert, .item-right-edit #menu .footer .button.icon-help, #menu .footer .item-right-edit .button.icon-help { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: absolute; - top: 0; - height: 100%; - font-size: 32px; } - .item-right-edit.visible { - display: block; } - .item-right-edit.visible.active { - opacity: 1; - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .item-reorder .button.icon, .item-reorder .button.icon-help, .item-reorder .button.icon-alert, .item-reorder #menu .footer .button.icon-help, #menu .footer .item-reorder .button.icon-help { - color: #444; - font-size: 32px; } - - .item-reordering { - position: absolute; - left: 0; - top: 0; - z-index: 9; - width: 100%; - box-shadow: 0px 0px 10px 0px #aaa; } - .item-reordering .item-reorder { - z-index: 9; } - - .item-placeholder { - opacity: 0.7; } - - /** - * The hidden right-side buttons that can be exposed under a list item - * with dragging. - */ - .item-options { - position: absolute; - top: 0; - right: 0; - z-index: 1; - height: 100%; } - .item-options .button { - height: 100%; - border: none; - border-radius: 0; - display: -webkit-inline-box; - display: -webkit-inline-flex; - display: -moz-inline-flex; - display: -ms-inline-flexbox; - display: inline-flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; } - .item-options .button:before { - margin: 0 auto; } - - /** - * Lists - * -------------------------------------------------- - */ - .list { - position: relative; - padding-top: 1px; - padding-bottom: 1px; - padding-left: 0; - margin-bottom: 20px; } - - .list:last-child { - margin-bottom: 0px; } - .list:last-child.card { - margin-bottom: 40px; } - - /** - * List Header - * -------------------------------------------------- - */ - .list-header { - margin-top: 20px; - padding: 5px 15px; - background-color: transparent; - color: #222; - font-weight: bold; } - - .card.list .list-item { - padding-right: 1px; - padding-left: 1px; } - - /** - * Cards and Inset Lists - * -------------------------------------------------- - * A card and list-inset are close to the same thing, except a card as a box shadow. - */ - .card, - .list-inset { - overflow: hidden; - margin: 20px 10px; - border-radius: 2px; - background-color: #fff; } - - .card { - padding-top: 1px; - padding-bottom: 1px; - box-shadow: 0 1px 3px rgba(0, 0, 0, 0.3); } - .card .item { - border-left: 0; - border-right: 0; } - .card .item:first-child { - border-top: 0; } - .card .item:last-child { - border-bottom: 0; } - - .padding .card, .item.large-button-bar .card, .padding .list-inset, .item.large-button-bar .list-inset { - margin-left: 0; - margin-right: 0; } - - .card .item:first-child, - .list-inset .item:first-child, - .padding > .list .item:first-child, .item.large-button-bar > .list .item:first-child { - border-top-left-radius: 2px; - border-top-right-radius: 2px; } - .card .item:first-child .item-content, - .list-inset .item:first-child .item-content, - .padding > .list .item:first-child .item-content, .item.large-button-bar > .list .item:first-child .item-content { - border-top-left-radius: 2px; - border-top-right-radius: 2px; } - - .card .item:last-child, - .list-inset .item:last-child, - .padding > .list .item:last-child, .item.large-button-bar > .list .item:last-child { - border-bottom-right-radius: 2px; - border-bottom-left-radius: 2px; } - .card .item:last-child .item-content, - .list-inset .item:last-child .item-content, - .padding > .list .item:last-child .item-content, .item.large-button-bar > .list .item:last-child .item-content { - border-bottom-right-radius: 2px; - border-bottom-left-radius: 2px; } - - .card .item:last-child, - .list-inset .item:last-child { - margin-bottom: -1px; } - - .card .item, - .list-inset .item, - .padding > .list .item, .item.large-button-bar > .list .item, - .padding-horizontal > .list .item { - margin-right: 0; - margin-left: 0; } - .card .item.item-input input, - .list-inset .item.item-input input, - .padding > .list .item.item-input input, .item.large-button-bar > .list .item.item-input input, - .padding-horizontal > .list .item.item-input input { - padding-right: 44px; } - - .padding-left > .list .item { - margin-left: 0; } - - .padding-right > .list .item, .popover-share .bar-footer .button-close > .list .item { - margin-right: 0; } - - /** - * Badges - * -------------------------------------------------- - */ - .badge { - background-color: transparent; - color: #AAAAAA; - z-index: 1; - display: inline-block; - padding: 3px 8px; - min-width: 10px; - border-radius: 10px; - vertical-align: baseline; - text-align: center; - white-space: nowrap; - font-weight: bold; - font-size: 14px; - line-height: 16px; } - .badge:empty { - display: none; } - - .tabs .tab-item .badge.badge-light, - .badge.badge-light { - background-color: #fff; - color: #444; } - - .tabs .tab-item .badge.badge-stable, - .badge.badge-stable { - background-color: #f8f8f8; - color: #444; } - - .tabs .tab-item .badge.badge-positive, - .badge.badge-positive { - background-color: #387ef5; - color: #fff; } - - .tabs .tab-item .badge.badge-calm, - .badge.badge-calm { - background-color: #11c1f3; - color: #fff; } - - .tabs .tab-item .badge.badge-assertive, .tabs .tab-item .badge.badge-editable:hover, - .badge.badge-assertive, - .badge.badge-editable:hover { - background-color: #ef473a; - color: #fff; } - - .tabs .tab-item .badge.badge-balanced, - .badge.badge-balanced { - background-color: #33cd5f; - color: #fff; } - - .tabs .tab-item .badge.badge-energized, - .badge.badge-energized { - background-color: #ffc900; - color: #fff; } - - .tabs .tab-item .badge.badge-royal, - .badge.badge-royal { - background-color: #886aea; - color: #fff; } - - .tabs .tab-item .badge.badge-dark, - .badge.badge-dark { - background-color: #444; - color: #fff; } - - .button .badge { - position: relative; - top: -1px; } - - /** - * Slide Box - * -------------------------------------------------- - */ - .slider { - position: relative; - visibility: hidden; - overflow: hidden; } - - .slider-slides { - position: relative; - height: 100%; } - - .slider-slide { - position: relative; - display: block; - float: left; - width: 100%; - height: 100%; - vertical-align: top; } - - .slider-slide-image > img { - width: 100%; } - - .slider-pager { - position: absolute; - bottom: 20px; - z-index: 1; - width: 100%; - height: 15px; - text-align: center; } - .slider-pager .slider-pager-page { - display: inline-block; - margin: 0px 3px; - width: 15px; - color: #000; - text-decoration: none; - opacity: 0.3; } - .slider-pager .slider-pager-page.active { - -webkit-transition: opacity 0.4s ease-in; - transition: opacity 0.4s ease-in; - opacity: 1; } - - .slider-slide.ng-enter, .slider-slide.ng-leave, .slider-slide.ng-animate, - .slider-pager-page.ng-enter, - .slider-pager-page.ng-leave, - .slider-pager-page.ng-animate { - -webkit-transition: none !important; - transition: none !important; } - - .slider-slide.ng-animate, - .slider-pager-page.ng-animate { - -webkit-animation: none 0s; - animation: none 0s; } - - /** - * Swiper 3.2.7 - * Most modern mobile touch slider and framework with hardware accelerated transitions - * - * http://www.idangero.us/swiper/ - * - * Copyright 2015, Vladimir Kharlampidi - * The iDangero.us - * http://www.idangero.us/ - * - * Licensed under MIT - * - * Released on: December 7, 2015 - */ - .swiper-container { - margin: 0 auto; - position: relative; - overflow: hidden; - /* Fix of Webkit flickering */ - z-index: 1; } - - .swiper-container-no-flexbox .swiper-slide { - float: left; } - - .swiper-container-vertical > .swiper-wrapper { - -webkit-box-orient: vertical; - -moz-box-orient: vertical; - -ms-flex-direction: column; - -webkit-flex-direction: column; - flex-direction: column; } - - .swiper-wrapper { - position: relative; - width: 100%; - height: 100%; - z-index: 1; - display: -webkit-box; - display: -moz-box; - display: -ms-flexbox; - display: -webkit-flex; - display: flex; - -webkit-transition-property: -webkit-transform; - -moz-transition-property: -moz-transform; - -o-transition-property: -o-transform; - -ms-transition-property: -ms-transform; - transition-property: transform; - -webkit-box-sizing: content-box; - -moz-box-sizing: content-box; - box-sizing: content-box; } - - .swiper-container-android .swiper-slide, - .swiper-wrapper { - -webkit-transform: translate3d(0px, 0, 0); - -moz-transform: translate3d(0px, 0, 0); - -o-transform: translate(0px, 0px); - -ms-transform: translate3d(0px, 0, 0); - transform: translate3d(0px, 0, 0); } - - .swiper-container-multirow > .swiper-wrapper { - -webkit-box-lines: multiple; - -moz-box-lines: multiple; - -ms-flex-wrap: wrap; - -webkit-flex-wrap: wrap; - flex-wrap: wrap; } - - .swiper-container-free-mode > .swiper-wrapper { - -webkit-transition-timing-function: ease-out; - -moz-transition-timing-function: ease-out; - -ms-transition-timing-function: ease-out; - -o-transition-timing-function: ease-out; - transition-timing-function: ease-out; - margin: 0 auto; } - - .swiper-slide { - display: block; - -webkit-flex-shrink: 0; - -ms-flex: 0 0 auto; - flex-shrink: 0; - width: 100%; - height: 100%; - position: relative; } - - /* Auto Height */ - .swiper-container-autoheight, - .swiper-container-autoheight .swiper-slide { - height: auto; } - - .swiper-container-autoheight .swiper-wrapper { - -webkit-box-align: start; - -ms-flex-align: start; - -webkit-align-items: flex-start; - align-items: flex-start; - -webkit-transition-property: -webkit-transform, height; - -moz-transition-property: -moz-transform; - -o-transition-property: -o-transform; - -ms-transition-property: -ms-transform; - transition-property: transform, height; } - - /* a11y */ - .swiper-container .swiper-notification { - position: absolute; - left: 0; - top: 0; - pointer-events: none; - opacity: 0; - z-index: -1000; } - - /* IE10 Windows Phone 8 Fixes */ - .swiper-wp8-horizontal { - -ms-touch-action: pan-y; - touch-action: pan-y; } - - .swiper-wp8-vertical { - -ms-touch-action: pan-x; - touch-action: pan-x; } - - /* Arrows */ - .swiper-button-prev, - .swiper-button-next { - position: absolute; - top: 50%; - width: 27px; - height: 44px; - margin-top: -22px; - z-index: 10; - cursor: pointer; - -moz-background-size: 27px 44px; - -webkit-background-size: 27px 44px; - background-size: 27px 44px; - background-position: center; - background-repeat: no-repeat; } - - .swiper-button-prev.swiper-button-disabled, - .swiper-button-next.swiper-button-disabled { - opacity: 0.35; - cursor: auto; - pointer-events: none; } - - .swiper-button-prev, - .swiper-container-rtl .swiper-button-next { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M0%2C22L22%2C0l2.1%2C2.1L4.2%2C22l19.9%2C19.9L22%2C44L0%2C22L0%2C22L0%2C22z'%20fill%3D'%23007aff'%2F%3E%3C%2Fsvg%3E"); - left: 10px; - right: auto; } - - .swiper-button-prev.swiper-button-black, - .swiper-container-rtl .swiper-button-next.swiper-button-black { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M0%2C22L22%2C0l2.1%2C2.1L4.2%2C22l19.9%2C19.9L22%2C44L0%2C22L0%2C22L0%2C22z'%20fill%3D'%23000000'%2F%3E%3C%2Fsvg%3E"); } - - .swiper-button-prev.swiper-button-white, - .swiper-container-rtl .swiper-button-next.swiper-button-white { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M0%2C22L22%2C0l2.1%2C2.1L4.2%2C22l19.9%2C19.9L22%2C44L0%2C22L0%2C22L0%2C22z'%20fill%3D'%23ffffff'%2F%3E%3C%2Fsvg%3E"); } - - .swiper-button-next, - .swiper-container-rtl .swiper-button-prev { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M27%2C22L27%2C22L5%2C44l-2.1-2.1L22.8%2C22L2.9%2C2.1L5%2C0L27%2C22L27%2C22z'%20fill%3D'%23007aff'%2F%3E%3C%2Fsvg%3E"); - right: 10px; - left: auto; } - - .swiper-button-next.swiper-button-black, - .swiper-container-rtl .swiper-button-prev.swiper-button-black { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M27%2C22L27%2C22L5%2C44l-2.1-2.1L22.8%2C22L2.9%2C2.1L5%2C0L27%2C22L27%2C22z'%20fill%3D'%23000000'%2F%3E%3C%2Fsvg%3E"); } - - .swiper-button-next.swiper-button-white, - .swiper-container-rtl .swiper-button-prev.swiper-button-white { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M27%2C22L27%2C22L5%2C44l-2.1-2.1L22.8%2C22L2.9%2C2.1L5%2C0L27%2C22L27%2C22z'%20fill%3D'%23ffffff'%2F%3E%3C%2Fsvg%3E"); } - - /* Pagination Styles */ - .swiper-pagination { - position: absolute; - text-align: center; - -webkit-transition: 300ms; - -moz-transition: 300ms; - -o-transition: 300ms; - transition: 300ms; - -webkit-transform: translate3d(0, 0, 0); - -ms-transform: translate3d(0, 0, 0); - -o-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - z-index: 10; } - - .swiper-pagination.swiper-pagination-hidden { - opacity: 0; } - - .swiper-pagination-bullet { - width: 8px; - height: 8px; - display: inline-block; - border-radius: 100%; - background: #000; - opacity: 0.2; } - - button.swiper-pagination-bullet { - border: none; - margin: 0; - padding: 0; - box-shadow: none; - -moz-appearance: none; - -ms-appearance: none; - -webkit-appearance: none; - appearance: none; } - - .swiper-pagination-clickable .swiper-pagination-bullet { - cursor: pointer; } - - .swiper-pagination-white .swiper-pagination-bullet { - background: #fff; } - - .swiper-pagination-bullet-active { - opacity: 1; } - - .swiper-pagination-white .swiper-pagination-bullet-active { - background: #fff; } - - .swiper-pagination-black .swiper-pagination-bullet-active { - background: #000; } - - .swiper-container-vertical > .swiper-pagination { - right: 10px; - top: 50%; - -webkit-transform: translate3d(0px, -50%, 0); - -moz-transform: translate3d(0px, -50%, 0); - -o-transform: translate(0px, -50%); - -ms-transform: translate3d(0px, -50%, 0); - transform: translate3d(0px, -50%, 0); } - - .swiper-container-vertical > .swiper-pagination .swiper-pagination-bullet { - margin: 5px 0; - display: block; } - - .swiper-container-horizontal > .swiper-pagination { - bottom: 10px; - left: 0; - width: 100%; } - - .swiper-container-horizontal > .swiper-pagination .swiper-pagination-bullet { - margin: 0 5px; } - - /* 3D Container */ - .swiper-container-3d { - -webkit-perspective: 1200px; - -moz-perspective: 1200px; - -o-perspective: 1200px; - perspective: 1200px; } - - .swiper-container-3d .swiper-wrapper, - .swiper-container-3d .swiper-slide, - .swiper-container-3d .swiper-slide-shadow-left, - .swiper-container-3d .swiper-slide-shadow-right, - .swiper-container-3d .swiper-slide-shadow-top, - .swiper-container-3d .swiper-slide-shadow-bottom, - .swiper-container-3d .swiper-cube-shadow { - -webkit-transform-style: preserve-3d; - -moz-transform-style: preserve-3d; - -ms-transform-style: preserve-3d; - transform-style: preserve-3d; } - - .swiper-container-3d .swiper-slide-shadow-left, - .swiper-container-3d .swiper-slide-shadow-right, - .swiper-container-3d .swiper-slide-shadow-top, - .swiper-container-3d .swiper-slide-shadow-bottom { - position: absolute; - left: 0; - top: 0; - width: 100%; - height: 100%; - pointer-events: none; - z-index: 10; } - - .swiper-container-3d .swiper-slide-shadow-left { - background-image: -webkit-gradient(linear, left top, right top, from(rgba(0, 0, 0, 0.5)), to(transparent)); - /* Safari 4+, Chrome */ - background-image: -webkit-linear-gradient(right, rgba(0, 0, 0, 0.5), transparent); - /* Chrome 10+, Safari 5.1+, iOS 5+ */ - background-image: -moz-linear-gradient(right, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 3.6-15 */ - background-image: -o-linear-gradient(right, rgba(0, 0, 0, 0.5), transparent); - /* Opera 11.10-12.00 */ - background-image: linear-gradient(to left, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 16+, IE10, Opera 12.50+ */ } - - .swiper-container-3d .swiper-slide-shadow-right { - background-image: -webkit-gradient(linear, right top, left top, from(rgba(0, 0, 0, 0.5)), to(transparent)); - /* Safari 4+, Chrome */ - background-image: -webkit-linear-gradient(left, rgba(0, 0, 0, 0.5), transparent); - /* Chrome 10+, Safari 5.1+, iOS 5+ */ - background-image: -moz-linear-gradient(left, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 3.6-15 */ - background-image: -o-linear-gradient(left, rgba(0, 0, 0, 0.5), transparent); - /* Opera 11.10-12.00 */ - background-image: linear-gradient(to right, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 16+, IE10, Opera 12.50+ */ } - - .swiper-container-3d .swiper-slide-shadow-top { - background-image: -webkit-gradient(linear, left top, left bottom, from(rgba(0, 0, 0, 0.5)), to(transparent)); - /* Safari 4+, Chrome */ - background-image: -webkit-linear-gradient(bottom, rgba(0, 0, 0, 0.5), transparent); - /* Chrome 10+, Safari 5.1+, iOS 5+ */ - background-image: -moz-linear-gradient(bottom, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 3.6-15 */ - background-image: -o-linear-gradient(bottom, rgba(0, 0, 0, 0.5), transparent); - /* Opera 11.10-12.00 */ - background-image: linear-gradient(to top, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 16+, IE10, Opera 12.50+ */ } - - .swiper-container-3d .swiper-slide-shadow-bottom { - background-image: -webkit-gradient(linear, left bottom, left top, from(rgba(0, 0, 0, 0.5)), to(transparent)); - /* Safari 4+, Chrome */ - background-image: -webkit-linear-gradient(top, rgba(0, 0, 0, 0.5), transparent); - /* Chrome 10+, Safari 5.1+, iOS 5+ */ - background-image: -moz-linear-gradient(top, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 3.6-15 */ - background-image: -o-linear-gradient(top, rgba(0, 0, 0, 0.5), transparent); - /* Opera 11.10-12.00 */ - background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.5), transparent); - /* Firefox 16+, IE10, Opera 12.50+ */ } - - /* Coverflow */ - .swiper-container-coverflow .swiper-wrapper { - /* Windows 8 IE 10 fix */ - -ms-perspective: 1200px; } - - /* Fade */ - .swiper-container-fade.swiper-container-free-mode .swiper-slide { - -webkit-transition-timing-function: ease-out; - -moz-transition-timing-function: ease-out; - -ms-transition-timing-function: ease-out; - -o-transition-timing-function: ease-out; - transition-timing-function: ease-out; } - - .swiper-container-fade .swiper-slide { - pointer-events: none; } - - .swiper-container-fade .swiper-slide .swiper-slide { - pointer-events: none; } - - .swiper-container-fade .swiper-slide-active, - .swiper-container-fade .swiper-slide-active .swiper-slide-active { - pointer-events: auto; } - - /* Cube */ - .swiper-container-cube { - overflow: visible; } - - .swiper-container-cube .swiper-slide { - pointer-events: none; - visibility: hidden; - -webkit-transform-origin: 0 0; - -moz-transform-origin: 0 0; - -ms-transform-origin: 0 0; - transform-origin: 0 0; - -webkit-backface-visibility: hidden; - -moz-backface-visibility: hidden; - -ms-backface-visibility: hidden; - backface-visibility: hidden; - width: 100%; - height: 100%; - z-index: 1; } - - .swiper-container-cube.swiper-container-rtl .swiper-slide { - -webkit-transform-origin: 100% 0; - -moz-transform-origin: 100% 0; - -ms-transform-origin: 100% 0; - transform-origin: 100% 0; } - - .swiper-container-cube .swiper-slide-active, - .swiper-container-cube .swiper-slide-next, - .swiper-container-cube .swiper-slide-prev, - .swiper-container-cube .swiper-slide-next + .swiper-slide { - pointer-events: auto; - visibility: visible; } - - .swiper-container-cube .swiper-slide-shadow-top, - .swiper-container-cube .swiper-slide-shadow-bottom, - .swiper-container-cube .swiper-slide-shadow-left, - .swiper-container-cube .swiper-slide-shadow-right { - z-index: 0; - -webkit-backface-visibility: hidden; - -moz-backface-visibility: hidden; - -ms-backface-visibility: hidden; - backface-visibility: hidden; } - - .swiper-container-cube .swiper-cube-shadow { - position: absolute; - left: 0; - bottom: 0px; - width: 100%; - height: 100%; - background: #000; - opacity: 0.6; - -webkit-filter: blur(50px); - filter: blur(50px); - z-index: 0; } - - /* Scrollbar */ - .swiper-scrollbar { - border-radius: 10px; - position: relative; - -ms-touch-action: none; - background: rgba(0, 0, 0, 0.1); } - - .swiper-container-horizontal > .swiper-scrollbar { - position: absolute; - left: 1%; - bottom: 3px; - z-index: 50; - height: 5px; - width: 98%; } - - .swiper-container-vertical > .swiper-scrollbar { - position: absolute; - right: 3px; - top: 1%; - z-index: 50; - width: 5px; - height: 98%; } - - .swiper-scrollbar-drag { - height: 100%; - width: 100%; - position: relative; - background: rgba(0, 0, 0, 0.5); - border-radius: 10px; - left: 0; - top: 0; } - - .swiper-scrollbar-cursor-drag { - cursor: move; } - - /* Preloader */ - .swiper-lazy-preloader { - width: 42px; - height: 42px; - position: absolute; - left: 50%; - top: 50%; - margin-left: -21px; - margin-top: -21px; - z-index: 10; - -webkit-transform-origin: 50%; - -moz-transform-origin: 50%; - transform-origin: 50%; - -webkit-animation: swiper-preloader-spin 1s steps(12, end) infinite; - -moz-animation: swiper-preloader-spin 1s steps(12, end) infinite; - animation: swiper-preloader-spin 1s steps(12, end) infinite; } - - .swiper-lazy-preloader:after { - display: block; - content: ""; - width: 100%; - height: 100%; - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20viewBox%3D'0%200%20120%20120'%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20xmlns%3Axlink%3D'http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink'%3E%3Cdefs%3E%3Cline%20id%3D'l'%20x1%3D'60'%20x2%3D'60'%20y1%3D'7'%20y2%3D'27'%20stroke%3D'%236c6c6c'%20stroke-width%3D'11'%20stroke-linecap%3D'round'%2F%3E%3C%2Fdefs%3E%3Cg%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(30%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(60%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(90%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(120%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(150%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.37'%20transform%3D'rotate(180%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.46'%20transform%3D'rotate(210%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.56'%20transform%3D'rotate(240%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.66'%20transform%3D'rotate(270%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.75'%20transform%3D'rotate(300%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.85'%20transform%3D'rotate(330%2060%2C60)'%2F%3E%3C%2Fg%3E%3C%2Fsvg%3E"); - background-position: 50%; - -webkit-background-size: 100%; - background-size: 100%; - background-repeat: no-repeat; } - - .swiper-lazy-preloader-white:after { - background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20viewBox%3D'0%200%20120%20120'%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20xmlns%3Axlink%3D'http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink'%3E%3Cdefs%3E%3Cline%20id%3D'l'%20x1%3D'60'%20x2%3D'60'%20y1%3D'7'%20y2%3D'27'%20stroke%3D'%23fff'%20stroke-width%3D'11'%20stroke-linecap%3D'round'%2F%3E%3C%2Fdefs%3E%3Cg%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(30%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(60%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(90%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(120%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(150%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.37'%20transform%3D'rotate(180%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.46'%20transform%3D'rotate(210%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.56'%20transform%3D'rotate(240%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.66'%20transform%3D'rotate(270%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.75'%20transform%3D'rotate(300%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.85'%20transform%3D'rotate(330%2060%2C60)'%2F%3E%3C%2Fg%3E%3C%2Fsvg%3E"); } - - @-webkit-keyframes swiper-preloader-spin { - 100% { - -webkit-transform: rotate(360deg); } } - - @keyframes swiper-preloader-spin { - 100% { - transform: rotate(360deg); } } - - ion-slides { - width: 100%; - height: 100%; - display: block; } - - .slide-zoom { - display: block; - width: 100%; - text-align: center; } - - .swiper-container { - width: 100%; - height: 100%; - padding: 0; - overflow: hidden; } - - .swiper-wrapper { - position: absolute; - left: 0; - top: 0; - width: 100%; - height: 100%; - padding: 0; } - - .swiper-slide { - width: 100%; - height: 100%; - box-sizing: border-box; - /* Center slide text vertically */ } - .swiper-slide img { - width: auto; - height: auto; - max-width: 100%; - max-height: 100%; } - - .scroll-refresher { - position: absolute; - top: -60px; - right: 0; - left: 0; - overflow: hidden; - margin: auto; - height: 60px; } - .scroll-refresher .ionic-refresher-content { - position: absolute; - bottom: 15px; - left: 0; - width: 100%; - color: #666666; - text-align: center; - font-size: 30px; } - .scroll-refresher .ionic-refresher-content .text-refreshing, - .scroll-refresher .ionic-refresher-content .text-pulling { - font-size: 16px; - line-height: 16px; } - .scroll-refresher .ionic-refresher-content.ionic-refresher-with-text { - bottom: 10px; } - .scroll-refresher .icon-refreshing, - .scroll-refresher .icon-pulling { - width: 100%; - -webkit-backface-visibility: hidden; - backface-visibility: hidden; - -webkit-transform-style: preserve-3d; - transform-style: preserve-3d; } - .scroll-refresher .icon-pulling { - -webkit-animation-name: refresh-spin-back; - animation-name: refresh-spin-back; - -webkit-animation-duration: 200ms; - animation-duration: 200ms; - -webkit-animation-timing-function: linear; - animation-timing-function: linear; - -webkit-animation-fill-mode: none; - animation-fill-mode: none; - -webkit-transform: translate3d(0, 0, 0) rotate(0deg); - transform: translate3d(0, 0, 0) rotate(0deg); } - .scroll-refresher .icon-refreshing, - .scroll-refresher .text-refreshing { - display: none; } - .scroll-refresher .icon-refreshing { - -webkit-animation-duration: 1.5s; - animation-duration: 1.5s; } - .scroll-refresher.active .icon-pulling:not(.pulling-rotation-disabled) { - -webkit-animation-name: refresh-spin; - animation-name: refresh-spin; - -webkit-transform: translate3d(0, 0, 0) rotate(-180deg); - transform: translate3d(0, 0, 0) rotate(-180deg); } - .scroll-refresher.active.refreshing { - -webkit-transition: -webkit-transform 0.2s; - transition: -webkit-transform 0.2s; - -webkit-transition: transform 0.2s; - transition: transform 0.2s; - -webkit-transform: scale(1, 1); - transform: scale(1, 1); } - .scroll-refresher.active.refreshing .icon-pulling, - .scroll-refresher.active.refreshing .text-pulling { - display: none; } - .scroll-refresher.active.refreshing .icon-refreshing, - .scroll-refresher.active.refreshing .text-refreshing { - display: block; } - .scroll-refresher.active.refreshing.refreshing-tail { - -webkit-transform: scale(0, 0); - transform: scale(0, 0); } - - .overflow-scroll > .scroll { - -webkit-overflow-scrolling: touch; - width: 100%; } - .overflow-scroll > .scroll.overscroll { - position: fixed; - right: 0; - left: 0; } - - .overflow-scroll.padding > .scroll.overscroll, .overflow-scroll.item.large-button-bar > .scroll.overscroll { - padding: 10px; } - - @-webkit-keyframes refresh-spin { - 0% { - -webkit-transform: translate3d(0, 0, 0) rotate(0); } - 100% { - -webkit-transform: translate3d(0, 0, 0) rotate(180deg); } } - - @keyframes refresh-spin { - 0% { - transform: translate3d(0, 0, 0) rotate(0); } - 100% { - transform: translate3d(0, 0, 0) rotate(180deg); } } - - @-webkit-keyframes refresh-spin-back { - 0% { - -webkit-transform: translate3d(0, 0, 0) rotate(180deg); } - 100% { - -webkit-transform: translate3d(0, 0, 0) rotate(0); } } - - @keyframes refresh-spin-back { - 0% { - transform: translate3d(0, 0, 0) rotate(180deg); } - 100% { - transform: translate3d(0, 0, 0) rotate(0); } } - - /** - * Spinners - * -------------------------------------------------- - */ - .spinner { - stroke: #444; - fill: #444; } - .spinner svg { - width: 28px; - height: 28px; } - .spinner.spinner-light { - stroke: #fff; - fill: #fff; } - .spinner.spinner-stable { - stroke: #f8f8f8; - fill: #f8f8f8; } - .spinner.spinner-positive { - stroke: #387ef5; - fill: #387ef5; } - .spinner.spinner-calm { - stroke: #11c1f3; - fill: #11c1f3; } - .spinner.spinner-balanced { - stroke: #33cd5f; - fill: #33cd5f; } - .spinner.spinner-assertive { - stroke: #ef473a; - fill: #ef473a; } - .spinner.spinner-energized { - stroke: #ffc900; - fill: #ffc900; } - .spinner.spinner-royal { - stroke: #886aea; - fill: #886aea; } - .spinner.spinner-dark { - stroke: #444; - fill: #444; } - - .spinner-android { - stroke: #4b8bf4; } - - .spinner-ios, - .spinner-ios-small { - stroke: #69717d; } - - .spinner-spiral .stop1 { - stop-color: #fff; - stop-opacity: 0; } - - .spinner-spiral.spinner-light .stop1 { - stop-color: #444; } - - .spinner-spiral.spinner-light .stop2 { - stop-color: #fff; } - - .spinner-spiral.spinner-stable .stop2 { - stop-color: #f8f8f8; } - - .spinner-spiral.spinner-positive .stop2 { - stop-color: #387ef5; } - - .spinner-spiral.spinner-calm .stop2 { - stop-color: #11c1f3; } - - .spinner-spiral.spinner-balanced .stop2 { - stop-color: #33cd5f; } - - .spinner-spiral.spinner-assertive .stop2 { - stop-color: #ef473a; } - - .spinner-spiral.spinner-energized .stop2 { - stop-color: #ffc900; } - - .spinner-spiral.spinner-royal .stop2 { - stop-color: #886aea; } - - .spinner-spiral.spinner-dark .stop2 { - stop-color: #444; } - - /** - * Forms - * -------------------------------------------------- - */ - form { - margin: 0 0 1.42857; } - - legend { - display: block; - margin-bottom: 1.42857; - padding: 0; - width: 100%; - border: 1px solid #ddd; - color: #444; - font-size: 21px; - line-height: 2.85714; } - legend small { - color: #f8f8f8; - font-size: 1.07143; } - - label, - input, - button, - select, - textarea { - font-weight: normal; - font-size: 14px; - line-height: 1.42857; } - - input, - button, - select, - textarea { - font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; } - - .item-input { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: relative; - overflow: hidden; - padding: 6px 0 5px 16px; } - .item-input input { - -webkit-border-radius: 0; - border-radius: 0; - -webkit-box-flex: 1; - -webkit-flex: 1 220px; - -moz-box-flex: 1; - -moz-flex: 1 220px; - -ms-flex: 1 220px; - flex: 1 220px; - -webkit-appearance: none; - -moz-appearance: none; - appearance: none; - margin: 0; - padding-right: 24px; - background-color: transparent; } - .item-input .button .icon, .item-input .button .icon-help, .item-input .button .icon-alert, .item-input .button #menu .footer .icon-help, #menu .footer .item-input .button .icon-help { - -webkit-box-flex: 0; - -webkit-flex: 0 0 24px; - -moz-box-flex: 0; - -moz-flex: 0 0 24px; - -ms-flex: 0 0 24px; - flex: 0 0 24px; - position: static; - display: inline-block; - height: auto; - text-align: center; - font-size: 16px; } - .item-input .button-bar { - -webkit-border-radius: 0; - border-radius: 0; - -webkit-box-flex: 1; - -webkit-flex: 1 0 220px; - -moz-box-flex: 1; - -moz-flex: 1 0 220px; - -ms-flex: 1 0 220px; - flex: 1 0 220px; - -webkit-appearance: none; - -moz-appearance: none; - appearance: none; } - .item-input .icon, .item-input .icon-help, .item-input .icon-alert, .item-input #menu .footer .icon-help, #menu .footer .item-input .icon-help { - min-width: 14px; } - - .platform-windowsphone .item-input input { - flex-shrink: 1; } - - .item-input-inset { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - position: relative; - overflow: hidden; - padding: 10.66667px; } - - .item-input-wrapper { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-flex: 1; - -webkit-flex: 1 0; - -moz-box-flex: 1; - -moz-flex: 1 0; - -ms-flex: 1 0; - flex: 1 0; - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; - -webkit-border-radius: 4px; - border-radius: 4px; - padding-right: 8px; - padding-left: 8px; - background: #eee; } - - .item-input-inset .item-input-wrapper input { - padding-left: 4px; - height: 29px; - background: transparent; - line-height: 18px; } - - .item-input-wrapper ~ .button { - margin-left: 10.66667px; } - - .input-label { - display: table; - padding: 7px 10px 7px 0px; - max-width: 200px; - width: 35%; - color: #444; - font-size: 16px; } - - .placeholder-icon { - color: #aaa; } - .placeholder-icon:first-child { - padding-right: 6px; } - .placeholder-icon:last-child { - padding-left: 6px; } - - .item-stacked-label { - display: block; - background-color: transparent; - box-shadow: none; } - .item-stacked-label .input-label, .item-stacked-label .icon, .item-stacked-label .icon-help, .item-stacked-label .icon-alert, .item-stacked-label #menu .footer .icon-help, #menu .footer .item-stacked-label .icon-help { - display: inline-block; - padding: 4px 0 0 0px; - vertical-align: middle; } - - .item-stacked-label input, - .item-stacked-label textarea { - -webkit-border-radius: 2px; - border-radius: 2px; - padding: 4px 8px 3px 0; - border: none; - background-color: #fff; } - - .item-stacked-label input { - overflow: hidden; - height: 46px; } - - .item-select.item-stacked-label select { - position: relative; - padding: 0px; - max-width: 90%; - direction: ltr; - white-space: pre-wrap; - margin: -3px; } - - .item-floating-label { - display: block; - background-color: transparent; - box-shadow: none; } - .item-floating-label .input-label { - position: relative; - padding: 5px 0 0 0; - opacity: 0; - top: 10px; - -webkit-transition: opacity 0.15s ease-in, top 0.2s linear; - transition: opacity 0.15s ease-in, top 0.2s linear; } - .item-floating-label .input-label.has-input { - opacity: 1; - top: 0; - -webkit-transition: opacity 0.15s ease-in, top 0.2s linear; - transition: opacity 0.15s ease-in, top 0.2s linear; } - - textarea, - input[type="text"], - input[type="password"], - input[type="datetime"], - input[type="datetime-local"], - input[type="date"], - input[type="month"], - input[type="time"], - input[type="week"], - input[type="number"], - input[type="email"], - input[type="url"], - input[type="search"], - input[type="tel"], - input[type="color"] { - display: block; - padding-top: 2px; - padding-left: 0; - height: 34px; - color: #111; - vertical-align: middle; - font-size: 14px; - line-height: 16px; } - - .platform-ios input[type="datetime-local"], - .platform-ios input[type="date"], - .platform-ios input[type="month"], - .platform-ios input[type="time"], - .platform-ios input[type="week"], - .platform-android input[type="datetime-local"], - .platform-android input[type="date"], - .platform-android input[type="month"], - .platform-android input[type="time"], - .platform-android input[type="week"] { - padding-top: 8px; } - - .item-input input, - .item-input textarea { - width: 100%; } - - textarea { - padding-left: 0; } - textarea::-moz-placeholder { - color: #aaaaaa; } - textarea:-ms-input-placeholder { - color: #aaaaaa; } - textarea::-webkit-input-placeholder { - color: #aaaaaa; - text-indent: -3px; } - - textarea { - height: auto; } - - textarea, - input[type="text"], - input[type="password"], - input[type="datetime"], - input[type="datetime-local"], - input[type="date"], - input[type="month"], - input[type="time"], - input[type="week"], - input[type="number"], - input[type="email"], - input[type="url"], - input[type="search"], - input[type="tel"], - input[type="color"] { - border: 0; } - - input[type="radio"], - input[type="checkbox"] { - margin: 0; - line-height: normal; } - - .item-input input[type="file"], - .item-input input[type="image"], - .item-input input[type="submit"], - .item-input input[type="reset"], - .item-input input[type="button"], - .item-input input[type="radio"], - .item-input input[type="checkbox"] { - width: auto; } - - input[type="file"] { - line-height: 34px; } - - .previous-input-focus, - .cloned-text-input + input, - .cloned-text-input + textarea { - position: absolute !important; - left: -9999px; - width: 200px; } - - input::-moz-placeholder, - textarea::-moz-placeholder { - color: #aaaaaa; } - - input:-ms-input-placeholder, - textarea:-ms-input-placeholder { - color: #aaaaaa; } - - input::-webkit-input-placeholder, - textarea::-webkit-input-placeholder { - color: #aaaaaa; - text-indent: 0; } - - input[disabled], - select[disabled], - textarea[disabled], - input[readonly]:not(.cloned-text-input), - textarea[readonly]:not(.cloned-text-input), - select[readonly] { - background-color: #f8f8f8; - cursor: not-allowed; } - - input[type="radio"][disabled], - input[type="checkbox"][disabled], - input[type="radio"][readonly], - input[type="checkbox"][readonly] { - background-color: transparent; } - - /** - * Checkbox - * -------------------------------------------------- - */ - .checkbox { - position: relative; - display: inline-block; - padding: 7px 7px; - cursor: pointer; } - .checkbox input:before, - .checkbox .checkbox-icon:before { - border-color: #ddd; } - .checkbox input:checked:before, - .checkbox input:checked + .checkbox-icon:before { - background: #387ef5; - border-color: #387ef5; } - - .checkbox-light input:before, - .checkbox-light .checkbox-icon:before { - border-color: #ddd; } - - .checkbox-light input:checked:before, - .checkbox-light input:checked + .checkbox-icon:before { - background: #ddd; - border-color: #ddd; } - - .checkbox-stable input:before, - .checkbox-stable .checkbox-icon:before { - border-color: #b2b2b2; } - - .checkbox-stable input:checked:before, - .checkbox-stable input:checked + .checkbox-icon:before { - background: #b2b2b2; - border-color: #b2b2b2; } - - .checkbox-positive input:before, - .checkbox-positive .checkbox-icon:before { - border-color: #387ef5; } - - .checkbox-positive input:checked:before, - .checkbox-positive input:checked + .checkbox-icon:before { - background: #387ef5; - border-color: #387ef5; } - - .checkbox-calm input:before, - .checkbox-calm .checkbox-icon:before { - border-color: #11c1f3; } - - .checkbox-calm input:checked:before, - .checkbox-calm input:checked + .checkbox-icon:before { - background: #11c1f3; - border-color: #11c1f3; } - - .checkbox-assertive input:before, - .checkbox-assertive .checkbox-icon:before { - border-color: #ef473a; } - - .checkbox-assertive input:checked:before, - .checkbox-assertive input:checked + .checkbox-icon:before { - background: #ef473a; - border-color: #ef473a; } - - .checkbox-balanced input:before, - .checkbox-balanced .checkbox-icon:before { - border-color: #33cd5f; } - - .checkbox-balanced input:checked:before, - .checkbox-balanced input:checked + .checkbox-icon:before { - background: #33cd5f; - border-color: #33cd5f; } - - .checkbox-energized input:before, - .checkbox-energized .checkbox-icon:before { - border-color: #ffc900; } - - .checkbox-energized input:checked:before, - .checkbox-energized input:checked + .checkbox-icon:before { - background: #ffc900; - border-color: #ffc900; } - - .checkbox-royal input:before, - .checkbox-royal .checkbox-icon:before { - border-color: #886aea; } - - .checkbox-royal input:checked:before, - .checkbox-royal input:checked + .checkbox-icon:before { - background: #886aea; - border-color: #886aea; } - - .checkbox-dark input:before, - .checkbox-dark .checkbox-icon:before { - border-color: #444; } - - .checkbox-dark input:checked:before, - .checkbox-dark input:checked + .checkbox-icon:before { - background: #444; - border-color: #444; } - - .checkbox input:disabled:before, - .checkbox input:disabled + .checkbox-icon:before { - border-color: #ddd; } - - .checkbox input:disabled:checked:before, - .checkbox input:disabled:checked + .checkbox-icon:before { - background: #ddd; } - - .checkbox.checkbox-input-hidden input { - display: none !important; } - - .checkbox input, - .checkbox-icon { - position: relative; - width: 28px; - height: 28px; - display: block; - border: 0; - background: transparent; - cursor: pointer; - -webkit-appearance: none; } - .checkbox input:before, - .checkbox-icon:before { - display: table; - width: 100%; - height: 100%; - border-width: 1px; - border-style: solid; - border-radius: 28px; - background: #fff; - content: ' '; - -webkit-transition: background-color 20ms ease-in-out; - transition: background-color 20ms ease-in-out; } - - .checkbox input:checked:before, - input:checked + .checkbox-icon:before { - border-width: 2px; } - - .checkbox input:after, - .checkbox-icon:after { - -webkit-transition: opacity 0.05s ease-in-out; - transition: opacity 0.05s ease-in-out; - -webkit-transform: rotate(-45deg); - transform: rotate(-45deg); - position: absolute; - top: 33%; - left: 25%; - display: table; - width: 14px; - height: 6px; - border: 1px solid #fff; - border-top: 0; - border-right: 0; - content: ' '; - opacity: 0; } - - .platform-android .checkbox-platform input:before, - .platform-android .checkbox-platform .checkbox-icon:before, - .checkbox-square input:before, - .checkbox-square .checkbox-icon:before { - border-radius: 2px; - width: 72%; - height: 72%; - margin-top: 14%; - margin-left: 14%; - border-width: 2px; } - - .platform-android .checkbox-platform input:after, - .platform-android .checkbox-platform .checkbox-icon:after, - .checkbox-square input:after, - .checkbox-square .checkbox-icon:after { - border-width: 2px; - top: 19%; - left: 25%; - width: 13px; - height: 7px; } - - .platform-android .item-checkbox-right .checkbox-square .checkbox-icon::after { - top: 31%; } - - .grade-c .checkbox input:after, - .grade-c .checkbox-icon:after { - -webkit-transform: rotate(0); - transform: rotate(0); - top: 3px; - left: 4px; - border: none; - color: #fff; - content: '\2713'; - font-weight: bold; - font-size: 20px; } - - .checkbox input:checked:after, - input:checked + .checkbox-icon:after { - opacity: 1; } - - .item-checkbox { - padding-left: 60px; } - .item-checkbox.active { - box-shadow: none; } - - .item-checkbox .checkbox { - position: absolute; - top: 50%; - right: 8px; - left: 8px; - z-index: 3; - margin-top: -21px; } - - .item-checkbox.item-checkbox-right { - padding-right: 60px; - padding-left: 16px; } - - .item-checkbox-right .checkbox input, - .item-checkbox-right .checkbox-icon { - float: right; } - - /** - * Toggle - * -------------------------------------------------- - */ - .item-toggle { - pointer-events: none; } - - .toggle { - position: relative; - display: inline-block; - pointer-events: auto; - margin: -5px; - padding: 5px; } - .toggle input:checked + .track { - border-color: #4cd964; - background-color: #4cd964; } - .toggle.dragging .handle { - background-color: #f2f2f2 !important; } - - .toggle.toggle-light input:checked + .track { - border-color: #ddd; - background-color: #ddd; } - - .toggle.toggle-stable input:checked + .track { - border-color: #b2b2b2; - background-color: #b2b2b2; } - - .toggle.toggle-positive input:checked + .track { - border-color: #387ef5; - background-color: #387ef5; } - - .toggle.toggle-calm input:checked + .track { - border-color: #11c1f3; - background-color: #11c1f3; } - - .toggle.toggle-assertive input:checked + .track { - border-color: #ef473a; - background-color: #ef473a; } - - .toggle.toggle-balanced input:checked + .track { - border-color: #33cd5f; - background-color: #33cd5f; } - - .toggle.toggle-energized input:checked + .track { - border-color: #ffc900; - background-color: #ffc900; } - - .toggle.toggle-royal input:checked + .track { - border-color: #886aea; - background-color: #886aea; } - - .toggle.toggle-dark input:checked + .track { - border-color: #444; - background-color: #444; } - - .toggle input { - display: none; } - - /* the track appearance when the toggle is "off" */ - .toggle .track { - -webkit-transition-timing-function: ease-in-out; - transition-timing-function: ease-in-out; - -webkit-transition-duration: 0.3s; - transition-duration: 0.3s; - -webkit-transition-property: background-color, border; - transition-property: background-color, border; - display: inline-block; - box-sizing: border-box; - width: 51px; - height: 31px; - border: solid 2px #e6e6e6; - border-radius: 20px; - background-color: #fff; - content: ' '; - cursor: pointer; - pointer-events: none; } - - /* Fix to avoid background color bleeding */ - /* (occurred on (at least) Android 4.2, Asus MeMO Pad HD7 ME173X) */ - .platform-android4_2 .toggle .track { - -webkit-background-clip: padding-box; } - - /* the handle (circle) thats inside the toggle's track area */ - /* also the handle's appearance when it is "off" */ - .toggle .handle { - -webkit-transition: 0.3s cubic-bezier(0, 1.1, 1, 1.1); - transition: 0.3s cubic-bezier(0, 1.1, 1, 1.1); - -webkit-transition-property: background-color, transform; - transition-property: background-color, transform; - position: absolute; - display: block; - width: 27px; - height: 27px; - border-radius: 27px; - background-color: #fff; - top: 7px; - left: 7px; - box-shadow: 0 2px 7px rgba(0, 0, 0, 0.35), 0 1px 1px rgba(0, 0, 0, 0.15); } - .toggle .handle:before { - position: absolute; - top: -4px; - left: -21.5px; - padding: 18.5px 34px; - content: " "; } - - .toggle input:checked + .track .handle { - -webkit-transform: translate3d(20px, 0, 0); - transform: translate3d(20px, 0, 0); - background-color: #fff; } - - .item-toggle.active { - box-shadow: none; } - - .item-toggle, - .item-toggle.item-complex .item-content { - padding-right: 99px; } - - .item-toggle.item-complex { - padding-right: 0; } - - .item-toggle .toggle { - position: absolute; - top: 10px; - right: 16px; - z-index: 3; } - - .toggle input:disabled + .track { - opacity: .6; } - - .toggle-small .track { - border: 0; - width: 34px; - height: 15px; - background: #9e9e9e; } - - .toggle-small input:checked + .track { - background: rgba(0, 150, 137, 0.5); } - - .toggle-small .handle { - top: 2px; - left: 4px; - width: 21px; - height: 21px; - box-shadow: 0 2px 5px rgba(0, 0, 0, 0.25); } - - .toggle-small input:checked + .track .handle { - -webkit-transform: translate3d(16px, 0, 0); - transform: translate3d(16px, 0, 0); - background: #009689; } - - .toggle-small.item-toggle .toggle { - top: 19px; } - - .toggle-small .toggle-light input:checked + .track { - background-color: rgba(221, 221, 221, 0.5); } - - .toggle-small .toggle-light input:checked + .track .handle { - background-color: #ddd; } - - .toggle-small .toggle-stable input:checked + .track { - background-color: rgba(178, 178, 178, 0.5); } - - .toggle-small .toggle-stable input:checked + .track .handle { - background-color: #b2b2b2; } - - .toggle-small .toggle-positive input:checked + .track { - background-color: rgba(56, 126, 245, 0.5); } - - .toggle-small .toggle-positive input:checked + .track .handle { - background-color: #387ef5; } - - .toggle-small .toggle-calm input:checked + .track { - background-color: rgba(17, 193, 243, 0.5); } - - .toggle-small .toggle-calm input:checked + .track .handle { - background-color: #11c1f3; } - - .toggle-small .toggle-assertive input:checked + .track { - background-color: rgba(239, 71, 58, 0.5); } - - .toggle-small .toggle-assertive input:checked + .track .handle { - background-color: #ef473a; } - - .toggle-small .toggle-balanced input:checked + .track { - background-color: rgba(51, 205, 95, 0.5); } - - .toggle-small .toggle-balanced input:checked + .track .handle { - background-color: #33cd5f; } - - .toggle-small .toggle-energized input:checked + .track { - background-color: rgba(255, 201, 0, 0.5); } - - .toggle-small .toggle-energized input:checked + .track .handle { - background-color: #ffc900; } - - .toggle-small .toggle-royal input:checked + .track { - background-color: rgba(136, 106, 234, 0.5); } - - .toggle-small .toggle-royal input:checked + .track .handle { - background-color: #886aea; } - - .toggle-small .toggle-dark input:checked + .track { - background-color: rgba(68, 68, 68, 0.5); } - - .toggle-small .toggle-dark input:checked + .track .handle { - background-color: #444; } - - /** - * Radio Button Inputs - * -------------------------------------------------- - */ - .item-radio { - padding: 0; } - .item-radio:hover { - cursor: pointer; } - - .item-radio .item-content { - /* give some room to the right for the checkmark icon */ - padding-right: 64px; } - - .item-radio .radio-icon { - /* checkmark icon will be hidden by default */ - position: absolute; - top: 0; - right: 0; - z-index: 3; - visibility: hidden; - padding: 14px; - height: 100%; - font-size: 24px; } - - .item-radio input { - /* hide any radio button inputs elements (the ugly circles) */ - position: absolute; - left: -9999px; } - .item-radio input:checked + .radio-content .item-content { - /* style the item content when its checked */ - background: #f7f7f7; } - .item-radio input:checked + .radio-content .radio-icon { - /* show the checkmark icon when its checked */ - visibility: visible; } - - /** - * Buttons - * -------------------------------------------------- - */ - .button { - border-color: transparent; - background-color: #f8f8f8; - color: #444; - position: relative; - display: inline-block; - margin: 0; - padding: 0 12px; - min-width: 52px; - min-height: 47px; - border-width: 1px; - border-style: solid; - border-radius: 4px; - vertical-align: top; - text-align: center; - text-overflow: ellipsis; - font-size: 16px; - line-height: 42px; - cursor: pointer; } - .button:hover { - color: #444; - text-decoration: none; } - .button.active, .button.activated { - border-color: #a2a2a2; - background-color: #e5e5e5; } - .button:after { - position: absolute; - top: -6px; - right: -6px; - bottom: -6px; - left: -6px; - content: ' '; } - .button .icon, .button .icon-help, .button .icon-alert, .button #menu .footer .icon-help, #menu .footer .button .icon-help { - vertical-align: top; - pointer-events: none; } - .button .icon:before, .button .icon-help:before, .button .icon-alert:before, .button #menu .footer .icon-help:before, #menu .footer .button .icon-help:before, .button.icon:before, .button.icon-help:before, .button.icon-alert:before, #menu .footer .button.icon-help:before, .button.icon-left:before, .button.icon-right:before { - display: inline-block; - padding: 0 0 1px 0; - vertical-align: inherit; - font-size: 24px; - line-height: 41px; - pointer-events: none; } - .button.icon-left:before { - float: left; - padding-right: .2em; - padding-left: 0; } - .button.icon-right:before { - float: right; - padding-right: 0; - padding-left: .2em; } - .button.button-block, .button.button-full { - margin-top: 10px; - margin-bottom: 10px; } - .button.button-light { - border-color: transparent; - background-color: #fff; - color: #444; } - .button.button-light:hover { - color: #444; - text-decoration: none; } - .button.button-light.active, .button.button-light.activated { - border-color: #a2a2a2; - background-color: #fafafa; } - .button.button-light.button-clear, .button.button-light.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #ddd; } - .button.button-light.button-icon { - border-color: transparent; - background: none; } - .button.button-light.button-outline { - border-color: #ddd; - background: transparent; - color: #ddd; } - .button.button-light.button-outline.active, .button.button-light.button-outline.activated { - background-color: #ddd; - box-shadow: none; - color: #fff; } - .button.button-stable { - border-color: transparent; - background-color: #f8f8f8; - color: #444; } - .button.button-stable:hover { - color: #444; - text-decoration: none; } - .button.button-stable.active, .button.button-stable.activated { - border-color: #a2a2a2; - background-color: #e5e5e5; } - .button.button-stable.button-clear, .button.button-stable.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #b2b2b2; } - .button.button-stable.button-icon { - border-color: transparent; - background: none; } - .button.button-stable.button-outline { - border-color: #b2b2b2; - background: transparent; - color: #b2b2b2; } - .button.button-stable.button-outline.active, .button.button-stable.button-outline.activated { - background-color: #b2b2b2; - box-shadow: none; - color: #fff; } - .button.button-positive, .button.button-text { - border-color: transparent; - background-color: #387ef5; - color: #fff; } - .button.button-positive:hover, .button.button-text:hover { - color: #fff; - text-decoration: none; } - .button.button-positive.active, .button.active.button-text, .button.button-positive.activated, .button.activated.button-text { - border-color: #a2a2a2; - background-color: #0c60ee; } - .button.button-positive.button-clear, .button.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #387ef5; } - .button.button-positive.button-icon, .button.button-icon.button-text { - border-color: transparent; - background: none; } - .button.button-positive.button-outline, .button.button-outline.button-text { - border-color: #387ef5; - background: transparent; - color: #387ef5; } - .button.button-positive.button-outline.active, .button.button-outline.active.button-text, .button.button-positive.button-outline.activated, .button.button-outline.activated.button-text { - background-color: #387ef5; - box-shadow: none; - color: #fff; } - .button.button-calm { - border-color: transparent; - background-color: #11c1f3; - color: #fff; } - .button.button-calm:hover { - color: #fff; - text-decoration: none; } - .button.button-calm.active, .button.button-calm.activated { - border-color: #a2a2a2; - background-color: #0a9dc7; } - .button.button-calm.button-clear, .button.button-calm.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #11c1f3; } - .button.button-calm.button-icon { - border-color: transparent; - background: none; } - .button.button-calm.button-outline { - border-color: #11c1f3; - background: transparent; - color: #11c1f3; } - .button.button-calm.button-outline.active, .button.button-calm.button-outline.activated { - background-color: #11c1f3; - box-shadow: none; - color: #fff; } - .button.button-assertive { - border-color: transparent; - background-color: #ef473a; - color: #fff; } - .button.button-assertive:hover { - color: #fff; - text-decoration: none; } - .button.button-assertive.active, .button.button-assertive.activated { - border-color: #a2a2a2; - background-color: #e42112; } - .button.button-assertive.button-clear, .button.button-assertive.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #ef473a; } - .button.button-assertive.button-icon { - border-color: transparent; - background: none; } - .button.button-assertive.button-outline { - border-color: #ef473a; - background: transparent; - color: #ef473a; } - .button.button-assertive.button-outline.active, .button.button-assertive.button-outline.activated { - background-color: #ef473a; - box-shadow: none; - color: #fff; } - .button.button-balanced { - border-color: transparent; - background-color: #33cd5f; - color: #fff; } - .button.button-balanced:hover { - color: #fff; - text-decoration: none; } - .button.button-balanced.active, .button.button-balanced.activated { - border-color: #a2a2a2; - background-color: #28a54c; } - .button.button-balanced.button-clear, .button.button-balanced.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #33cd5f; } - .button.button-balanced.button-icon { - border-color: transparent; - background: none; } - .button.button-balanced.button-outline { - border-color: #33cd5f; - background: transparent; - color: #33cd5f; } - .button.button-balanced.button-outline.active, .button.button-balanced.button-outline.activated { - background-color: #33cd5f; - box-shadow: none; - color: #fff; } - .button.button-energized { - border-color: transparent; - background-color: #ffc900; - color: #fff; } - .button.button-energized:hover { - color: #fff; - text-decoration: none; } - .button.button-energized.active, .button.button-energized.activated { - border-color: #a2a2a2; - background-color: #e6b500; } - .button.button-energized.button-clear, .button.button-energized.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #ffc900; } - .button.button-energized.button-icon { - border-color: transparent; - background: none; } - .button.button-energized.button-outline { - border-color: #ffc900; - background: transparent; - color: #ffc900; } - .button.button-energized.button-outline.active, .button.button-energized.button-outline.activated { - background-color: #ffc900; - box-shadow: none; - color: #fff; } - .button.button-royal { - border-color: transparent; - background-color: #886aea; - color: #fff; } - .button.button-royal:hover { - color: #fff; - text-decoration: none; } - .button.button-royal.active, .button.button-royal.activated { - border-color: #a2a2a2; - background-color: #6b46e5; } - .button.button-royal.button-clear, .button.button-royal.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #886aea; } - .button.button-royal.button-icon { - border-color: transparent; - background: none; } - .button.button-royal.button-outline { - border-color: #886aea; - background: transparent; - color: #886aea; } - .button.button-royal.button-outline.active, .button.button-royal.button-outline.activated { - background-color: #886aea; - box-shadow: none; - color: #fff; } - .button.button-dark { - border-color: transparent; - background-color: #444; - color: #fff; } - .button.button-dark:hover { - color: #fff; - text-decoration: none; } - .button.button-dark.active, .button.button-dark.activated { - border-color: #a2a2a2; - background-color: #262626; } - .button.button-dark.button-clear, .button.button-dark.button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: #444; } - .button.button-dark.button-icon { - border-color: transparent; - background: none; } - .button.button-dark.button-outline { - border-color: #444; - background: transparent; - color: #444; } - .button.button-dark.button-outline.active, .button.button-dark.button-outline.activated { - background-color: #444; - box-shadow: none; - color: #fff; } - - .button-small, .button-text.button-small { - padding: 2px 4px 1px; - min-width: 28px; - min-height: 30px; - font-size: 12px; - line-height: 26px; } - .button-small .icon:before, .button-small .icon-help:before, .button-small .icon-alert:before, .button-small #menu .footer .icon-help:before, #menu .footer .button-small .icon-help:before, .button-small.icon:before, .button-small.icon-help:before, .button-small.icon-alert:before, #menu .footer .button-small.icon-help:before, .button-small.icon-left:before, .button-small.icon-right:before { - font-size: 16px; - line-height: 19px; - margin-top: 3px; } - - .button-large { - padding: 0 16px; - min-width: 68px; - min-height: 59px; - font-size: 20px; - line-height: 53px; } - .button-large .icon:before, .button-large .icon-help:before, .button-large .icon-alert:before, .button-large #menu .footer .icon-help:before, #menu .footer .button-large .icon-help:before, .button-large.icon:before, .button-large.icon-help:before, .button-large.icon-alert:before, #menu .footer .button-large.icon-help:before, .button-large.icon-left:before, .button-large.icon-right:before { - padding-bottom: 2px; - font-size: 32px; - line-height: 51px; } - - .button-icon { - -webkit-transition: opacity 0.1s; - transition: opacity 0.1s; - padding: 0 6px; - min-width: initial; - border-color: transparent; - background: none; } - .button-icon.button.active, .button-icon.button.activated { - border-color: transparent; - background: none; - box-shadow: none; - opacity: 0.3; } - .button-icon .icon:before, .button-icon .icon-help:before, .button-icon .icon-alert:before, .button-icon #menu .footer .icon-help:before, #menu .footer .button-icon .icon-help:before, .button-icon.icon:before, .button-icon.icon-help:before, .button-icon.icon-alert:before, #menu .footer .button-icon.icon-help:before { - font-size: 32px; } - - .button-clear, .button-text { - -webkit-transition: opacity 0.1s; - transition: opacity 0.1s; - padding: 0 6px; - max-height: 42px; - border-color: transparent; - background: none; - box-shadow: none; } - .button-clear.button-clear, .button-text { - border-color: transparent; - background: none; - box-shadow: none; - color: transparent; } - .button-clear.button-icon, .button-icon.button-text { - border-color: transparent; - background: none; } - .button-clear.active, .active.button-text, .button-clear.activated, .activated.button-text { - opacity: 0.3; } - - .button-outline { - -webkit-transition: opacity 0.1s; - transition: opacity 0.1s; - background: none; - box-shadow: none; } - .button-outline.button-outline { - border-color: transparent; - background: transparent; - color: transparent; } - .button-outline.button-outline.active, .button-outline.button-outline.activated { - background-color: transparent; - box-shadow: none; - color: #fff; } - - .padding > .button.button-block:first-child, .item.large-button-bar > .button.button-block:first-child { - margin-top: 0; } - - .button-block { - display: block; - clear: both; } - .button-block:after { - clear: both; } - - .button-full, - .button-full > .button { - display: block; - margin-right: 0; - margin-left: 0; - border-right-width: 0; - border-left-width: 0; - border-radius: 0; } - - button.button-block, - button.button-full, - .button-full > button.button, - input.button.button-block { - width: 100%; } - - a.button { - text-decoration: none; } - a.button .icon:before, a.button .icon-help:before, a.button .icon-alert:before, a.button #menu .footer .icon-help:before, #menu .footer a.button .icon-help:before, a.button.icon:before, a.button.icon-help:before, a.button.icon-alert:before, #menu .footer a.button.icon-help:before, a.button.icon-left:before, a.button.icon-right:before { - margin-top: 2px; } - - .button.disabled, - .button[disabled] { - opacity: .4; - cursor: default !important; - pointer-events: none; } - - /** - * Button Bar - * -------------------------------------------------- - */ - .button-bar { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - width: 100%; } - .button-bar.button-bar-inline { - display: block; - width: auto; - *zoom: 1; } - .button-bar.button-bar-inline:before, .button-bar.button-bar-inline:after { - display: table; - content: ""; - line-height: 0; } - .button-bar.button-bar-inline:after { - clear: both; } - .button-bar.button-bar-inline > .button { - width: auto; - display: inline-block; - float: left; } - .button-bar.bar-light > .button { - border-color: #ddd; } - .button-bar.bar-stable > .button { - border-color: #b2b2b2; } - .button-bar.bar-positive > .button { - border-color: #0c60ee; } - .button-bar.bar-calm > .button { - border-color: #0a9dc7; } - .button-bar.bar-assertive > .button { - border-color: #e42112; } - .button-bar.bar-balanced > .button { - border-color: #28a54c; } - .button-bar.bar-energized > .button { - border-color: #e6b500; } - .button-bar.bar-royal > .button { - border-color: #6b46e5; } - .button-bar.bar-dark > .button { - border-color: #111; } - - .button-bar > .button { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - display: block; - overflow: hidden; - padding: 0 16px; - width: 0; - border-width: 1px 0px 1px 1px; - border-radius: 0; - text-align: center; - text-overflow: ellipsis; - white-space: nowrap; } - .button-bar > .button:before, - .button-bar > .button .icon:before, - .button-bar > .button .icon-help:before, - .button-bar > .button .icon-alert:before, - .button-bar > .button #menu .footer .icon-help:before, #menu .footer - .button-bar > .button .icon-help:before { - line-height: 44px; } - .button-bar > .button:first-child { - border-radius: 4px 0px 0px 4px; } - .button-bar > .button:last-child { - border-right-width: 1px; - border-radius: 0px 4px 4px 0px; } - .button-bar > .button:only-child { - border-radius: 4px; } - - .button-bar > .button-small:before, - .button-bar > .button-small .icon:before, - .button-bar > .button-small .icon-help:before, - .button-bar > .button-small .icon-alert:before, - .button-bar > .button-small #menu .footer .icon-help:before, #menu .footer - .button-bar > .button-small .icon-help:before { - line-height: 28px; } - - /** - * Grid - * -------------------------------------------------- - * Using flexbox for the grid, inspired by Philip Walton: - * http://philipwalton.github.io/solved-by-flexbox/demos/grids/ - * By default each .col within a .row will evenly take up - * available width, and the height of each .col with take - * up the height of the tallest .col in the same .row. - */ - .row { - display: -webkit-box; - display: -webkit-flex; - display: -moz-box; - display: -moz-flex; - display: -ms-flexbox; - display: flex; - padding: 5px; - width: 100%; } - - .row-wrap { - -webkit-flex-wrap: wrap; - -moz-flex-wrap: wrap; - -ms-flex-wrap: wrap; - flex-wrap: wrap; } - - .row-no-padding { - padding: 0; } - .row-no-padding > .col { - padding: 0; } - - .row + .row { - margin-top: -5px; - padding-top: 0; } - - .col { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - display: block; - padding: 5px; - width: 100%; } - - /* Vertically Align Columns */ - /* .row-* vertically aligns every .col in the .row */ - .row-top { - -webkit-box-align: start; - -ms-flex-align: start; - -webkit-align-items: flex-start; - -moz-align-items: flex-start; - align-items: flex-start; } - - .row-bottom { - -webkit-box-align: end; - -ms-flex-align: end; - -webkit-align-items: flex-end; - -moz-align-items: flex-end; - align-items: flex-end; } - - .row-center { - -webkit-box-align: center; - -ms-flex-align: center; - -webkit-align-items: center; - -moz-align-items: center; - align-items: center; } - - .row-stretch { - -webkit-box-align: stretch; - -ms-flex-align: stretch; - -webkit-align-items: stretch; - -moz-align-items: stretch; - align-items: stretch; } - - .row-baseline { - -webkit-box-align: baseline; - -ms-flex-align: baseline; - -webkit-align-items: baseline; - -moz-align-items: baseline; - align-items: baseline; } - - /* .col-* vertically aligns an individual .col */ - .col-top { - -webkit-align-self: flex-start; - -moz-align-self: flex-start; - -ms-flex-item-align: start; - align-self: flex-start; } - - .col-bottom { - -webkit-align-self: flex-end; - -moz-align-self: flex-end; - -ms-flex-item-align: end; - align-self: flex-end; } - - .col-center { - -webkit-align-self: center; - -moz-align-self: center; - -ms-flex-item-align: center; - align-self: center; } - - /* Column Offsets */ - .col-offset-10 { - margin-left: 10%; } - - .col-offset-20 { - margin-left: 20%; } - - .col-offset-25 { - margin-left: 25%; } - - .col-offset-33, .col-offset-34 { - margin-left: 33.3333%; } - - .col-offset-50 { - margin-left: 50%; } - - .col-offset-66, .col-offset-67 { - margin-left: 66.6666%; } - - .col-offset-75 { - margin-left: 75%; } - - .col-offset-80 { - margin-left: 80%; } - - .col-offset-90 { - margin-left: 90%; } - - /* Explicit Column Percent Sizes */ - /* By default each grid column will evenly distribute */ - /* across the grid. However, you can specify individual */ - /* columns to take up a certain size of the available area */ - .col-10 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 10%; - -moz-box-flex: 0; - -moz-flex: 0 0 10%; - -ms-flex: 0 0 10%; - flex: 0 0 10%; - max-width: 10%; } - - .col-20 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 20%; - -moz-box-flex: 0; - -moz-flex: 0 0 20%; - -ms-flex: 0 0 20%; - flex: 0 0 20%; - max-width: 20%; } - - .col-25 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 25%; - -moz-box-flex: 0; - -moz-flex: 0 0 25%; - -ms-flex: 0 0 25%; - flex: 0 0 25%; - max-width: 25%; } - - .col-33, .col-34 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 33.3333%; - -moz-box-flex: 0; - -moz-flex: 0 0 33.3333%; - -ms-flex: 0 0 33.3333%; - flex: 0 0 33.3333%; - max-width: 33.3333%; } - - .col-40 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 40%; - -moz-box-flex: 0; - -moz-flex: 0 0 40%; - -ms-flex: 0 0 40%; - flex: 0 0 40%; - max-width: 40%; } - - .col-50 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 50%; - -moz-box-flex: 0; - -moz-flex: 0 0 50%; - -ms-flex: 0 0 50%; - flex: 0 0 50%; - max-width: 50%; } - - .col-60 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 60%; - -moz-box-flex: 0; - -moz-flex: 0 0 60%; - -ms-flex: 0 0 60%; - flex: 0 0 60%; - max-width: 60%; } - - .col-66, .col-67 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 66.6666%; - -moz-box-flex: 0; - -moz-flex: 0 0 66.6666%; - -ms-flex: 0 0 66.6666%; - flex: 0 0 66.6666%; - max-width: 66.6666%; } - - .col-75 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 75%; - -moz-box-flex: 0; - -moz-flex: 0 0 75%; - -ms-flex: 0 0 75%; - flex: 0 0 75%; - max-width: 75%; } - - .col-80 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 80%; - -moz-box-flex: 0; - -moz-flex: 0 0 80%; - -ms-flex: 0 0 80%; - flex: 0 0 80%; - max-width: 80%; } - - .col-90 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 90%; - -moz-box-flex: 0; - -moz-flex: 0 0 90%; - -ms-flex: 0 0 90%; - flex: 0 0 90%; - max-width: 90%; } - - /* Responsive Grid Classes */ - /* Adding a class of responsive-X to a row */ - /* will trigger the flex-direction to */ - /* change to column and add some margin */ - /* to any columns in the row for clearity */ - @media (max-width: 567px) { - .responsive-sm { - -webkit-box-direction: normal; - -moz-box-direction: normal; - -webkit-box-orient: vertical; - -moz-box-orient: vertical; - -webkit-flex-direction: column; - -ms-flex-direction: column; - flex-direction: column; } - .responsive-sm .col, .responsive-sm .col-10, .responsive-sm .col-20, .responsive-sm .col-25, .responsive-sm .col-33, .responsive-sm .col-34, .responsive-sm .col-50, .responsive-sm .col-66, .responsive-sm .col-67, .responsive-sm .col-75, .responsive-sm .col-80, .responsive-sm .col-90 { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - margin-bottom: 15px; - margin-left: 0; - max-width: 100%; - width: 100%; } } - - @media (max-width: 767px) { - .responsive-md { - -webkit-box-direction: normal; - -moz-box-direction: normal; - -webkit-box-orient: vertical; - -moz-box-orient: vertical; - -webkit-flex-direction: column; - -ms-flex-direction: column; - flex-direction: column; } - .responsive-md .col, .responsive-md .col-10, .responsive-md .col-20, .responsive-md .col-25, .responsive-md .col-33, .responsive-md .col-34, .responsive-md .col-50, .responsive-md .col-66, .responsive-md .col-67, .responsive-md .col-75, .responsive-md .col-80, .responsive-md .col-90 { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - margin-bottom: 15px; - margin-left: 0; - max-width: 100%; - width: 100%; } } - - @media (max-width: 1023px) { - .responsive-lg { - -webkit-box-direction: normal; - -moz-box-direction: normal; - -webkit-box-orient: vertical; - -moz-box-orient: vertical; - -webkit-flex-direction: column; - -ms-flex-direction: column; - flex-direction: column; } - .responsive-lg .col, .responsive-lg .col-10, .responsive-lg .col-20, .responsive-lg .col-25, .responsive-lg .col-33, .responsive-lg .col-34, .responsive-lg .col-50, .responsive-lg .col-66, .responsive-lg .col-67, .responsive-lg .col-75, .responsive-lg .col-80, .responsive-lg .col-90 { - -webkit-box-flex: 1; - -webkit-flex: 1; - -moz-box-flex: 1; - -moz-flex: 1; - -ms-flex: 1; - flex: 1; - margin-bottom: 15px; - margin-left: 0; - max-width: 100%; - width: 100%; } } - - /** - * Utility Classes - * -------------------------------------------------- - */ - .hide { - display: none; } - - .opacity-hide { - opacity: 0; } - - .grade-b .opacity-hide, - .grade-c .opacity-hide { - opacity: 1; - display: none; } - - .show { - display: block; } - - .opacity-show { - opacity: 1; } - - .invisible { - visibility: hidden; } - - .keyboard-open .hide-on-keyboard-open { - display: none; } - - .keyboard-open .tabs.hide-on-keyboard-open + .pane .has-tabs, - .keyboard-open .bar-footer.hide-on-keyboard-open + .pane .has-footer { - bottom: 0; } - - .inline { - display: inline-block; } - - .disable-pointer-events { - pointer-events: none; } - - .enable-pointer-events { - pointer-events: auto; } - - .disable-user-behavior { - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - -webkit-touch-callout: none; - -webkit-tap-highlight-color: transparent; - -webkit-tap-highlight-color: transparent; - -webkit-user-drag: none; - -ms-touch-action: none; - -ms-content-zooming: none; } - - .click-block { - position: absolute; - top: 0; - right: 0; - bottom: 0; - left: 0; - opacity: 0; - z-index: 99999; - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - overflow: hidden; } - - .click-block-hide { - -webkit-transform: translate3d(-9999px, 0, 0); - transform: translate3d(-9999px, 0, 0); } - - .no-resize { - resize: none; } - - .block { - display: block; - clear: both; } - .block:after { - display: block; - visibility: hidden; - clear: both; - height: 0; - content: "."; } - - .full-image { - width: 100%; } - - .clearfix { - *zoom: 1; } - .clearfix:before, .clearfix:after { - display: table; - content: ""; - line-height: 0; } - .clearfix:after { - clear: both; } - - /** - * Content Padding - * -------------------------------------------------- - */ - .padding, .item.large-button-bar { - padding: 10px; } - - .padding-top, - .padding-vertical { - padding-top: 10px; } - - .padding-right, .popover-share .bar-footer .button-close, - .padding-horizontal { - padding-right: 10px; } - - .padding-bottom, .popover-share .bar-footer .button-close, - .padding-vertical { - padding-bottom: 10px; } - - .padding-left, - .padding-horizontal { - padding-left: 10px; } - - /** - * Scrollable iFrames - * -------------------------------------------------- - */ - .iframe-wrapper { - position: fixed; - -webkit-overflow-scrolling: touch; - overflow: scroll; } - .iframe-wrapper iframe { - height: 100%; - width: 100%; } - - /** - * Rounded - * -------------------------------------------------- - */ - .rounded { - border-radius: 4px; } - - /** - * Utility Colors - * -------------------------------------------------- - * Utility colors are added to help set a naming convention. You'll - * notice we purposely do not use words like "red" or "blue", but - * instead have colors which represent an emotion or generic theme. - */ - .light, a.light { - color: #fff; } - - .light-bg { - background-color: #fff; } - - .light-border { - border-color: #ddd; } - - .stable, a.stable { - color: #f8f8f8; } - - .stable-bg { - background-color: #f8f8f8; } - - .stable-border { - border-color: #b2b2b2; } - - .positive, .icon-help, a.positive, a.icon-help { - color: #387ef5; } - - .positive-bg { - background-color: #387ef5; } - - .positive-border { - border-color: #0c60ee; } - - .calm, #menu .footer .icon-help, a.calm, #menu .footer a.icon-help { - color: #11c1f3; } - - .calm-bg { - background-color: #11c1f3; } - - .calm-border { - border-color: #0a9dc7; } - - .assertive, .icon-alert, a.assertive, a.icon-alert { - color: #ef473a; } - - .assertive-bg { - background-color: #ef473a; } - - .assertive-border { - border-color: #e42112; } - - .balanced, a.balanced { - color: #33cd5f; } - - .balanced-bg { - background-color: #33cd5f; } - - .balanced-border { - border-color: #28a54c; } - - .energized, a.energized { - color: #ffc900; } - - .energized-bg { - background-color: #ffc900; } - - .energized-border { - border-color: #e6b500; } - - .royal, a.royal { - color: #886aea; } - - .royal-bg { - background-color: #886aea; } - - .royal-border { - border-color: #6b46e5; } - - .dark, a.dark { - color: #444; } - - .dark-bg { - background-color: #444; } - - .dark-border { - border-color: #111; } - - [collection-repeat] { - /* Position is set by transforms */ - left: 0 !important; - top: 0 !important; - position: absolute !important; - z-index: 1; } - - .collection-repeat-container { - position: relative; - z-index: 1; } - - .collection-repeat-after-container { - z-index: 0; - display: block; - /* when scrolling horizontally, make sure the after container doesn't take up 100% width */ } - .collection-repeat-after-container.horizontal { - display: inline-block; } - - [ng\:cloak], [ng-cloak], [data-ng-cloak], [x-ng-cloak], .ng-cloak, - .x-ng-cloak, .ng-hide:not(.ng-hide-animate) { - display: none !important; } - - /** - * Platform - * -------------------------------------------------- - * Platform specific tweaks - */ - .platform-ios.platform-cordova:not(.fullscreen) .bar-header:not(.bar-subheader) { - height: 64px; } - .platform-ios.platform-cordova:not(.fullscreen) .bar-header:not(.bar-subheader).item-input-inset .item-input-wrapper { - margin-top: 19px !important; } - .platform-ios.platform-cordova:not(.fullscreen) .bar-header:not(.bar-subheader) > * { - margin-top: 20px; } - - .platform-ios.platform-cordova:not(.fullscreen) .tabs-top > .tabs, - .platform-ios.platform-cordova:not(.fullscreen) .tabs.tabs-top { - top: 64px; } - - .platform-ios.platform-cordova:not(.fullscreen) .has-header, - .platform-ios.platform-cordova:not(.fullscreen) .bar-subheader { - top: 64px; } - - .platform-ios.platform-cordova:not(.fullscreen) .has-subheader { - top: 108px; } - - .platform-ios.platform-cordova:not(.fullscreen) .has-header.has-tabs-top { - top: 113px; } - - .platform-ios.platform-cordova:not(.fullscreen) .has-header.has-subheader.has-tabs-top { - top: 157px; } - - .platform-ios.platform-cordova .popover .bar-header:not(.bar-subheader) { - height: 44px; } - .platform-ios.platform-cordova .popover .bar-header:not(.bar-subheader).item-input-inset .item-input-wrapper { - margin-top: -1px; } - .platform-ios.platform-cordova .popover .bar-header:not(.bar-subheader) > * { - margin-top: 0; } - - .platform-ios.platform-cordova .popover .has-header, - .platform-ios.platform-cordova .popover .bar-subheader { - top: 44px; } - - .platform-ios.platform-cordova .popover .has-subheader { - top: 88px; } - - .platform-ios.platform-cordova.status-bar-hide { - margin-bottom: 20px; } - - @media (orientation: landscape) { - .platform-ios.platform-browser.platform-ipad { - position: fixed; } } - - .platform-c:not(.enable-transitions) * { - -webkit-transition: none !important; - transition: none !important; } - - .slide-in-up { - -webkit-transform: translate3d(0, 100%, 0); - transform: translate3d(0, 100%, 0); } - - .slide-in-up.ng-enter, - .slide-in-up > .ng-enter { - -webkit-transition: all cubic-bezier(0.1, 0.7, 0.1, 1) 400ms; - transition: all cubic-bezier(0.1, 0.7, 0.1, 1) 400ms; } - - .slide-in-up.ng-enter-active, - .slide-in-up > .ng-enter-active { - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .slide-in-up.ng-leave, - .slide-in-up > .ng-leave { - -webkit-transition: all ease-in-out 250ms; - transition: all ease-in-out 250ms; } - - @-webkit-keyframes scaleOut { - from { - -webkit-transform: scale(1); - opacity: 1; } - to { - -webkit-transform: scale(0.8); - opacity: 0; } } - - @keyframes scaleOut { - from { - transform: scale(1); - opacity: 1; } - to { - transform: scale(0.8); - opacity: 0; } } - - @-webkit-keyframes superScaleIn { - from { - -webkit-transform: scale(1.2); - opacity: 0; } - to { - -webkit-transform: scale(1); - opacity: 1; } } - - @keyframes superScaleIn { - from { - transform: scale(1.2); - opacity: 0; } - to { - transform: scale(1); - opacity: 1; } } - - [nav-view-transition="ios"] [nav-view="entering"], - [nav-view-transition="ios"] [nav-view="leaving"] { - -webkit-transition-duration: 500ms; - transition-duration: 500ms; - -webkit-transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); - transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); - -webkit-transition-property: opacity, -webkit-transform, box-shadow; - transition-property: opacity, transform, box-shadow; } - - [nav-view-transition="ios"][nav-view-direction="forward"], [nav-view-transition="ios"][nav-view-direction="back"] { - background-color: #000; } - - [nav-view-transition="ios"] [nav-view="active"], - [nav-view-transition="ios"][nav-view-direction="forward"] [nav-view="entering"], - [nav-view-transition="ios"][nav-view-direction="back"] [nav-view="leaving"] { - z-index: 3; } - - [nav-view-transition="ios"][nav-view-direction="back"] [nav-view="entering"], - [nav-view-transition="ios"][nav-view-direction="forward"] [nav-view="leaving"] { - z-index: 2; } - - [nav-bar-transition="ios"] .title, - [nav-bar-transition="ios"] .buttons, - [nav-bar-transition="ios"] .back-text { - -webkit-transition-duration: 500ms; - transition-duration: 500ms; - -webkit-transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); - transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); - -webkit-transition-property: opacity, -webkit-transform; - transition-property: opacity, transform; } - - [nav-bar-transition="ios"] [nav-bar="active"], - [nav-bar-transition="ios"] [nav-bar="entering"] { - z-index: 10; } - [nav-bar-transition="ios"] [nav-bar="active"] .bar, - [nav-bar-transition="ios"] [nav-bar="entering"] .bar { - background: transparent; } - - [nav-bar-transition="ios"] [nav-bar="cached"] { - display: block; } - [nav-bar-transition="ios"] [nav-bar="cached"] .header-item { - display: none; } - - [nav-view-transition="android"] [nav-view="entering"], - [nav-view-transition="android"] [nav-view="leaving"] { - -webkit-transition-duration: 200ms; - transition-duration: 200ms; - -webkit-transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); - transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); - -webkit-transition-property: -webkit-transform; - transition-property: transform; } - - [nav-view-transition="android"] [nav-view="active"], - [nav-view-transition="android"][nav-view-direction="forward"] [nav-view="entering"], - [nav-view-transition="android"][nav-view-direction="back"] [nav-view="leaving"] { - z-index: 3; } - - [nav-view-transition="android"][nav-view-direction="back"] [nav-view="entering"], - [nav-view-transition="android"][nav-view-direction="forward"] [nav-view="leaving"] { - z-index: 2; } - - [nav-bar-transition="android"] .title, - [nav-bar-transition="android"] .buttons { - -webkit-transition-duration: 200ms; - transition-duration: 200ms; - -webkit-transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); - transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); - -webkit-transition-property: opacity; - transition-property: opacity; } - - [nav-bar-transition="android"] [nav-bar="active"], - [nav-bar-transition="android"] [nav-bar="entering"] { - z-index: 10; } - [nav-bar-transition="android"] [nav-bar="active"] .bar, - [nav-bar-transition="android"] [nav-bar="entering"] .bar { - background: transparent; } - - [nav-bar-transition="android"] [nav-bar="cached"] { - display: block; } - [nav-bar-transition="android"] [nav-bar="cached"] .header-item { - display: none; } - - [nav-swipe="fast"] [nav-view], - [nav-swipe="fast"] .title, - [nav-swipe="fast"] .buttons, - [nav-swipe="fast"] .back-text { - -webkit-transition-duration: 50ms; - transition-duration: 50ms; - -webkit-transition-timing-function: linear; - transition-timing-function: linear; } - - [nav-swipe="slow"] [nav-view], - [nav-swipe="slow"] .title, - [nav-swipe="slow"] .buttons, - [nav-swipe="slow"] .back-text { - -webkit-transition-duration: 160ms; - transition-duration: 160ms; - -webkit-transition-timing-function: linear; - transition-timing-function: linear; } - - [nav-view="cached"], - [nav-bar="cached"] { - display: none; } - - [nav-view="stage"] { - opacity: 0; - -webkit-transition-duration: 0; - transition-duration: 0; } - - [nav-bar="stage"] .title, - [nav-bar="stage"] .buttons, - [nav-bar="stage"] .back-text { - position: absolute; - opacity: 0; - -webkit-transition-duration: 0s; - transition-duration: 0s; } - - /* BEGIN Thin */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Thin/RobotoDraft-Thin.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Thin/RobotoDraft-Thin.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Thin/RobotoDraft-Thin.ttf?v=1.1.0") format("truetype"); - font-weight: 100; - font-style: normal; } - - /* END Thin */ - /* BEGIN Light */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Light/RobotoDraft-Light.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Light/RobotoDraft-Light.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Light/RobotoDraft-Light.ttf?v=1.1.0") format("truetype"); - font-weight: 300; - font-style: normal; } - - /* END Light */ - /* BEGIN Regular */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.ttf?v=1.1.0") format("truetype"); - font-weight: 400; - font-style: normal; } - - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.ttf?v=1.1.0") format("truetype"); - font-weight: normal; - font-style: normal; } - - /* END Regular */ - /* BEGIN Italic */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.ttf?v=1.1.0") format("truetype"); - font-weight: 400; - font-style: italic; } - - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.ttf?v=1.1.0") format("truetype"); - font-weight: normal; - font-style: italic; } - - /* END Italic */ - /* BEGIN Medium */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Medium/RobotoDraft-Medium.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Medium/RobotoDraft-Medium.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Medium/RobotoDraft-Medium.ttf?v=1.1.0") format("truetype"); - font-weight: 500; - font-style: normal; } - - /* END Medium */ - /* BEGIN Bold */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.ttf?v=1.1.0") format("truetype"); - font-weight: 700; - font-style: normal; } - - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.ttf?v=1.1.0") format("truetype"); - font-weight: bold; - font-style: normal; } - - /* END Bold */ - /* BEGIN Bold Italic */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.ttf?v=1.1.0") format("truetype"); - font-weight: 700; - font-style: italic; } - - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.ttf?v=1.1.0") format("truetype"); - font-weight: bold; - font-style: italic; } - - /* END Bold Italic */ - /* BEGIN Black */ - @font-face { - font-family: RobotoDraft; - src: url("../lib/ionic/fonts/robotdraft/Black/RobotoDraft-Black.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Black/RobotoDraft-Black.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Black/RobotoDraft-Black.ttf?v=1.1.0") format("truetype"); - font-weight: 900; - font-style: normal; } - - /* END Black */ - /* Directives : MD Label - ==================================*/ - .item-md-label { - display: block; - background: transparent; - box-shadow: none; - margin-left: 12px; - margin-right: 12px; - padding: 30px 0 0; } - - .item-md-label .input-label { - position: absolute; - padding: 5px 0 0; - z-index: 2; - -webkit-transform: translate3d(0, -30px, 0) scale(1); - transform: translate3d(0, -30px, 0) scale(1); - -webkit-transition: all 0.2s ease; - transition: all 0.2s ease; - color: #fff; - opacity: 0.5; - filter: alpha(opacity=50); - -webkit-transform-origin: 0; - -ms-transform-origin: 0; - transform-origin: 0; } - - .item-md-label input { - background-color: rgba(0, 0, 0, 0.6); - bottom: 0; - color: #fff; - letter-spacing: 0.25rem; - padding: 20px 10px; - position: relative; - z-index: 1; } - - .item-md-label .highlight { - position: absolute; - bottom: 0; - height: 2px; - left: 0; - width: 100%; - -webkit-transform: translate3d(-100%, 0, 0); - transform: translate3d(-100%, 0, 0); - -webkit-transition: all 0.15s ease; - transition: all 0.15s ease; - z-index: 1; } - - .item-md-label .highlight-light { - background: #fff; } - - .item-md-label .highlight-stable { - background: #f8f8f8; } - - .item-md-label .highlight-positive { - background: #387ef5; } - - .item-md-label .highlight-calm { - background: #11c1f3; } - - .item-md-label .highlight-balanced { - background: #33cd5f; } - - .item-md-label .highlight-energized { - background: #ffc900; } - - .item-md-label .highlight-assertive { - background: #ef473a; } - - .item-md-label .highlight-royal { - background: #886aea; } - - .item-md-label .highlight-dark { - background: #444; } - - .item-md-label .input-label { - letter-spacing: 0.25rem; - padding: 0 10px; } - - .item-md-label input:focus ~ .input-label, .item-md-label input.used ~ .input-label { - font-weight: bold; - opacity: 0.7; - filter: alpha(opacity=70); - padding: 0; - text-transform: uppercase; - -webkit-transform: translate3d(0, -60px, 0) scale(0.9); - transform: translate3d(0, -60px, 0) scale(0.9); } - - .item-md-label input:focus ~ .highlight { - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - /* Bar - Header - Expanded - ==================================*/ - .expanded .bar.bar-header, - .bar.bar-header.expanded { - height: 75px; } - - .expanded.bar.bar-header .title, - .bar.bar-header.expanded .title { - bottom: 0; - top: initial; - padding-left: 16px; } - - .expanded .bar.bar-header .title.fab-left, - .bar.bar-header.expanded .title.fab-left { - bottom: 0; - left: 90px; - position: absolute; - right: initial; - top: initial; } - - .expanded .bar.bar-header .title.fab-right, - .bar.bar-header.expanded .title.fab-right { - bottom: 0; - left: 4px; - position: absolute; - top: initial; - right: initial; } - - .expanded .bar.bar-header + .button-fab, - .bar.bar-header.expanded + .button-fab { - top: 50px; } - - .expanded .bar.bar-header.push-down, - .bar.bar-header.expanded.push-down { - height: 44px; - overflow: hidden; } - - .expanded .bar.bar-header, - .bar.bar-header.expanded { - -webkit-transition: height 1s cubic-bezier(0.55, 0, 0.1, 1); - transition: height 1s cubic-bezier(0.55, 0, 0.1, 1); - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .expanded .bar.bar-header + .button-fab, - .bar.bar-header.expanded + .button-fab { - -webkit-transition: all 1.1s cubic-bezier(0.55, 0, 0.1, 1); - transition: all 1.1s cubic-bezier(0.55, 0, 0.1, 1); - -webkit-transform: translate3d(0, 0, 0) scale(1); - transform: translate3d(0, 0, 0) scale(1); } - - .expanded .bar.bar-header.push-down + .button-fab, - .bar.bar-header.expanded.push-down + .button-fab { - top: 0; - -webkit-transform: translate3d(-100px, -100px, 0) scale(2.5); - transform: translate3d(-100px, -100px, 0) scale(2.5); } - - .expanded .bar.bar-header.push-down .title, - .bar.bar-header.expanded.push-down .title { - opacity: 0; - filter: alpha(opacity=0); - left: initial; - right: initial; } - - .expanded .bar.bar-header .title, - .bar.bar-header.expanded .title { - opacity: 1; - filter: alpha(opacity=100); - -webkit-transition: all 2s cubic-bezier(0.55, 0, 0.1, 1); - transition: all 2s cubic-bezier(0.55, 0, 0.1, 1); } - - .expanded .bar.bar-header .title, .bar.bar-header.expanded .title { - bottom: 0; - left: 42px !important; - top: initial; } - - .expanded.has-header-fab-left .bar.bar-header .title, .bar.bar-header.expanded.has-header-fab-left .title { - left: 76px !important; } - - /* Bar - ==================================*/ - .bar { - z-index: 2; - font-size: 1.3em; - width: 100%; - box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); } - - .bar .button { - min-width: 38px; - z-index: 3; } - - .bar .no-text span.back-text { - display: none; } - - .bar .title sup { - opacity: 0.7; } - - .bar.bar-header .button + .title { - text-align: left; - left: 35px; - line-height: 46px; } - - /* Button Bar - ==================================*/ - .button-bar { - box-shadow: 0px 1px 3px rgba(0, 0, 0, 0.15); } - - .button-bar > .button { - box-shadow: none; - /* line-height: initial; */ } - - .button-bar > .button .icon:before, .button-bar > .button .icon-help:before, .button-bar > .button .icon-alert:before, .button-bar > .button #menu .footer .icon-help:before, #menu .footer .button-bar > .button .icon-help:before, - .button-bar > .button:before { - line-height: initial; } - - .bar-footer .button-fab { - position: absolute; - top: -26px; - bottom: initial; } - - .bar-footer .buttons-left .button-fab { - left: 8px; } - - .bar-footer .buttons-right .button-fab { - right: 8px; } - - .bar .button.button-clear, .bar .button.button-text { - box-shadow: none; } - - .left-buttons .button-fab { - left: 8px; - top: 16px; } - - .right-buttons .button-fab { - right: 8px; - top: 16px; } - - .fab-left.title-left, - .fab-left.title.title-left { - left: 68px; } - - /* Button : FAB - ==================================*/ - .button.button-fab, - .bar .button.button-fab { - box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.16), 0 2px 10px 0 rgba(0, 0, 0, 0.12); - z-index: 9999; - width: 56px; - height: 56px; - max-height: initial; - max-width: initial; - border-radius: 50%; - border-radius: 50%; - overflow: hidden; - padding: 0; - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - -webkit-transition: 0.3s fade-in-out; - transition: 0.3s fade-in-out; - -webkit-transition-property: -webkit-transform, box-shadow; - transition-property: transform, box-shadow; } - - .button.button-fab.button-fab-bottom-right, - .bar .button.button-fab.button-fab-bottom-right { - top: auto; - right: 16px; - bottom: 16px; - left: auto; - position: absolute; } - - .button.button-fab.button-fab-bottom-left, - .bar .button.button-fab.button-fab-bottom-left { - top: auto; - right: auto; - bottom: 16px; - left: 16px; - position: absolute; } - - .button.button-fab.button-fab-top-right, - .bar .button.button-fab.button-fab-top-right { - top: 32px; - right: 16px; - bottom: auto; - left: auto; - position: absolute; } - - .button.button-fab.button-fab-top-left, - .bar .button.button-fab.button-fab-top-left { - top: 32px; - right: auto; - bottom: auto; - left: 16px; - position: absolute; } - - .button.button-fab.button-fab-top-left.expanded, - .button.button-fab.button-fab-top-right.expanded, - .bar .button.button-fab.button-fab-top-left.expanded, - .bar .button.button-fab.button-fab-top-right.expanded { - top: 48px; } - - .button.button-fab i, - .bar .button.button-fab i { - font-size: 2.5rem; - margin-top: 0; } - - .button.button-fab.mini, - .bar .button.button-fab.mini { - width: 40px; - height: 40px; } - - .button.button-fab.mini i, - .bar .button.button-fab.mini i { - font-size: 2rem; } - - /* Motion */ - .motion { - -webkit-transition: all 0.5s ease-out; - transition: all 0.5s ease-out; } - - .fade { - opacity: 0; - filter: alpha(opacity=0); - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .spin-back { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(360deg) scale(0) !important; - transform: translateZ(0) rotate(360deg) scale(0) !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .spiral { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; - transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .spiral-back { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; - transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .menu-open .avatar { - opacity: 1; - filter: alpha(opacity=100); - -webkit-transform: translateZ(0) rotate(0) scale(1) !important; - transform: translateZ(0) rotate(0) scale(1) !important; - -webkit-transition: all 0.3s ease-out !important; - transition: all 0.3s ease-out !important; } - - .button.button-fab.button-fab-top-left.motion { - -webkit-backface-visibility: hidden; - backface-visibility: hidden; - -webkit-transform: translate3d(-120px, 60px, 0); - transform: translate3d(-120px, 60px, 0); - -webkit-transition: all 0.1s ease-out; - transition: all 0.1s ease-out; } - - .button.button-fab.button-fab-top-right.motion { - -webkit-backface-visibility: hidden; - backface-visibility: hidden; - -webkit-transform: translate3d(120px, 60px, 0); - transform: translate3d(120px, 60px, 0); - -webkit-transition: all 0.1s ease-out; - transition: all 0.1s ease-out; } - - .button.button-fab.button-fab-bottom-left.motion { - -webkit-backface-visibility: hidden; - backface-visibility: hidden; - -webkit-transform: translate3d(-120px, 60px, 0); - transform: translate3d(-120px, 60px, 0); - -webkit-transition: all 0.1s ease-out; - transition: all 0.1s ease-out; } - - .button.button-fab.button-fab-bottom-right.motion { - -webkit-backface-visibility: hidden; - backface-visibility: hidden; - -webkit-transform: translate3d(120px, 60px, 0); - transform: translate3d(120px, 60px, 0); - -webkit-transition: all 0.1s ease-out; - transition: all 0.1s ease-out; } - - .spin { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(0) scale(0) !important; - transform: translateZ(0) rotate(0) scale(0) !important; - -webkit-transition: all 0.3s ease-out !important; - transition: all 0.3s ease-out !important; } - - .spin.on { - -webkit-transform: translateZ(0) rotate(-360deg) scale(1) !important; - transform: translateZ(0) rotate(-360deg) scale(1) !important; } - - .flap { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; - transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .flap.on { - -webkit-transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; - transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .drop { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) scale(3) !important; - transform: translateZ(0) scale(3) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .drop.on { - -webkit-transform: translateZ(0) scale(1) !important; - transform: translateZ(0) scale(1) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .flip { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotateY(0) scale(0) !important; - transform: translateZ(0) rotateY(0) scale(0) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .flip.on { - -webkit-transform: translateZ(0) rotateY(-720deg) scale(1) !important; - transform: translateZ(0) rotateY(-720deg) scale(1) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - /* Button : Floating - ==================================*/ - .button.button-floating, .bar .button.button-floating { - display: inline-block; - color: #FFF; - position: relative; - z-index: 1; - width: 37px; - height: 37px; - line-height: 37px; - padding: 0; - border-radius: 50%; - background-clip: padding-box; - -webkit-transition: 0.3s; - transition: 0.3s; - cursor: pointer; } - - .button.button-floating i, .bar .button.button-floating i { - width: inherit; - display: inline-block; - text-align: center; - color: #FFF; - font-size: 1.6rem; - line-height: 37px; } - - .button.button-floating.button-large, .bar .button.button-floating.button-large { - width: 55.5px; - height: 55.5px; } - - .button.button-floating.button-large i, .bar .button.button-floating.button-large i { - line-height: 55.5px; } - - /* Button - ==================================*/ - .button, - .button.button-large, - .button.button-flat, - .bar .button, - .bar .button.button-large, - .bar .button.button-flat { - box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.16), 0 2px 10px 0 rgba(0, 0, 0, 0.12); - display: inline-block; - height: 36px; - padding: 0 2rem; - border-radius: 2px; - background-clip: padding-box; - text-transform: uppercase; - border: none; - outline: 0; - -webkit-tap-highlight-color: transparent; } - - .button.disabled, - .button.disabled.button-large, - .button.button-floating.disabled, - .button.button-large.disabled, - .button.button:disabled, - .button.button-large:disabled, - .button.button-large:disabled, - .button.button-floating:disabled, - .bar .button.disabled, - .bar .button.disabled.button-large, - .bar .button.button-floating.disabled, - .bar .button.button-large.disabled, - .bar .button.button:disabled, - .bar .button.button-large:disabled, - .bar .button.button-large:disabled, - .bar .button.button-floating:disabled { - background-color: #DFDFDF; - box-shadow: none; - color: #9F9F9F; } - - .button.disabled:hover, - .button.disabled.button-large:hover, - .button.button-floating.disabled:hover, - .button.button-large.disabled:hover, - .button.button:disabled:hover, - .button.button-large:disabled:hover, - .button.button-large:disabled:hover, - .button.button-floating:disabled:hover, - .bar .button.disabled:hover, - .bar .button.disabled.button-large:hover, - .bar .button.button-floating.disabled:hover, - .bar .button.button-large.disabled:hover, - .bar .button.button:disabled:hover, - .bar .button.button-large:disabled:hover, - .bar .button.button-large:disabled:hover, - .bar .button.button-floating:disabled:hover { - background-color: #DFDFDF; - color: #9F9F9F; } - - .button i, - .button.button-large i, - .button.button-floating i, - .button.button-large i, - .button.button-flat i, - .bar .button i, - .bar .button.button-large i, - .bar .button.button-floating i, - .bar .button.button-large i, - .bar .button.button-flat i { - font-size: 1.3rem; } - - .button-bar .button { - border-radius: 0; } - - .button, - .button-large, - .bar .button, - .bar .button-large { - text-decoration: none; - text-align: center; - letter-spacing: 0.5px; - -webkit-transition: 0.2s ease-out; - transition: 0.2s ease-out; - cursor: pointer; } - - .button { - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - position: relative; - outline: none; - margin: 0; - /* background: transparent; */ - white-space: nowrap; - text-align: center; - text-transform: uppercase; - font-weight: 500; - font-style: inherit; - font-variant: inherit; - font-size: inherit; - text-decoration: none; - cursor: pointer; - overflow: hidden; - -webkit-transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), -webkit-transform 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); - transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), transform 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); } - - .button:focus { - outline: none; } - - .button.ng-hide { - -webkit-transition: none; - transition: none; } - - .button.cornered { - border-radius: 0; } - - .button.raised { - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .button-outline, - .button-outline:hover, - .button-outline:active { - border-style: solid; - border-width: 1px; } - - .button.button-outline.button-assertive, - .button.button-outline.button-balanced, - .button.button-outline.button-calm, - .button.button-outline.button-dark, - .button.button-outline.button-energized, - .button.button-outline.button-light, - .button.button-outline.button-positive, - .button.button-outline.button-text, - .button.button-outline.button-royal, - .button.button-outline.button-stable, - .button.button-outline { - border-color: rgba(0, 0, 0, 0.1); } - - .button-flat, - .bar .button-flat { - box-shadow: none; - background-color: transparent; - color: #343434; - cursor: pointer; } - - .button.button-flat.disabled, - .bar .button.button-flat.disabled { - color: #b3b3b3; } - - .button.button-large i, - .bar .button.button-large i { - font-size: 1.6rem; } - - .button-pin-header.button-floating { - position: absolute; - z-index: 1000; } - - .button-pin-header.button-pin-left { - left: 24px; - top: -24px; } - - .button-pin-header.button-pin-right { - right: 24px; - top: -24px; } - - .button:not([disabled]).raised:focus, - .button:not([disabled]).raised:hover, - .button:not([disabled]).floating:focus, - .button:not([disabled]).floating:hover { - -webkit-transform: translate3d(0, -1px, 0); - transform: translate3d(0, -1px, 0); } - - .button.button-flat { - box-shadow: none; - /* background: transparent; */ - color: inherit; } - - .button.button-flat:hover { - color: inherit; } - - .button.button-flat, - .button.button-flat:hover, - .button.button-flat:active { - color: #fff; } - - .button.button-clear, .button.button-text, - .button.button-clear:hover, - .button.button-text:hover, - .button.button-clear:active, - .button.button-text:active { - background: transparent; } - - .button-full.ink, - .button-block.ink { - display: block; } - - /* Card - ==================================*/ - .card-item.item { - border: none; - padding-bottom: 4px; - padding-top: 4px; } - - .card-item.item:first-child { - padding-top: 16px; } - - .card { - box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); - display: block; - margin: 8px; - padding: 0; - position: relative; } - - .card .image { - display: block; - margin-top: 10px; - margin-bottom: 5px; } - - .card img { - box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); - display: block; - max-width: 100%; - max-height: initial; - position: static; } - - .card.card-gallery img { - border: none; - box-shadow: none; - display: block; } - - .card .card-footer { - font-size: 90%; - opacity: 0.8; - filter: alpha(opacity=80); - padding-top: 10px; } - - .card > .item { - border: none; } - - .card.card-gallery > .item { - background: inherit; } - - .card .icon + .icon, .card .icon-help + .icon, .card .icon-alert + .icon, .card #menu .footer .icon-help + .icon, #menu .footer .card .icon-help + .icon, .card .icon + .icon-help, .card .icon-help + .icon-help, .card .icon-alert + .icon-help, .card .icon + .icon-alert, .card .icon-help + .icon-alert, .card .icon-alert + .icon-alert, .card #menu .footer .icon-help + .icon-alert, #menu .footer .card .icon-help + .icon-alert, .card #menu .footer .icon + .icon-help, #menu .footer .card .icon + .icon-help, .card #menu .footer .icon-alert + .icon-help, #menu .footer .card .icon-alert + .icon-help, .card #menu .footer .icon-help + .icon-help, #menu .footer .card .icon-help + .icon-help { - padding-left: 1rem; } - - .card.animate-fade-in { - opacity: 0; - filter: alpha(opacity=0); - -webkit-transform: translate3d(-30px, 1px, 0); - -webkit-transition: all 1s ease-in-out; } - - .card.animate-fade-in.done, .animate-fade-slide-in .expanded .card.animate-fade-in.item, - .animate-fade-slide-in .card.animate-fade-in.expanded.item, - .animate-fade-slide-in-right .expanded .card.animate-fade-in.item, - .animate-fade-slide-in-right .card.animate-fade-in.expanded.item, - .animate-ripple .expanded .card.animate-fade-in.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.animate-fade-in.card-comment { - opacity: 1; - filter: alpha(opacity=100); - -webkit-transform: translate3d(0, 0, 0); } - - .card .item.item-avatar { - min-height: 88px; - padding-left: 88px; } - - /* Hero - ==================================*/ - .hero { - background-size: cover; - box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); - color: #fff; - height: 200px; - position: relative; - text-align: center; - -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - width: 100%; } - - .hero > * { - -webkit-transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); - transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); - opacity: 1; - filter: alpha(opacity=100); } - - .hero + .mid-bar { - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - height: initial; - opacity: 1; - filter: alpha(opacity=100); } - - .hero .hero-icon { - box-shadow: 0px 0 2px 0 rgba(0, 0, 0, 0.26); - border-radius: 50%; - display: inline-block; - font-size: 65px; - height: 150px; - padding: 10px 30px; - line-height: 136px; - width: 150px; } - - .hero.no-header { - height: 244px; } - - .hero > .content { - bottom: 0; - position: absolute; - text-align: center; - width: 100%; - z-index: 1; } - - .hero > .content > .avatar { - background-position: center; - background-size: cover; - border: solid 1px rgba(255, 255, 255, 0.8); - border-radius: 50%; - display: inline-block; - height: 88px; - left: auto; - margin-bottom: 10px; - position: relative; - width: 88px; } - - .hero h1 .hero h2, .hero h3, .hero h4, .hero h5, .hero h6 { - color: #fff; - margin: 0; } - - .hero h4 { - color: rgba(255, 255, 255, 0.7); - margin: 3px 0 16px; } - - .hero h1 > a, .hero h2 > a, .hero h3 > a, .hero h4 > a, .hero h5 > a, .hero h6 > a { - text-decoration: none; } - - .hero + .button-bar { - border-radius: 0; - margin-top: 0; } - - .hero + .button-bar > .button:first-child, .hero + .button-bar > .button:last-child { - border-radius: 0; } - - .hero .hero-icon { - color: #fff; - font-size: 96px; } - - .hero .hero-icon + h1 { - color: white; - letter-spacing: 0.15rem; } - - .hero .button, .hero .button.button-large, .hero .button.button-flat { - margin: 0; } - - .hero h1.title { - color: #fff; - font-size: 23px; - margin: 0; - text-align: left; - padding-left: 80px; - line-height: 59px; } - - .hero + .mid-bar { - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); - -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - height: initial; - opacity: 1; - filter: alpha(opacity=100); } - - .hero > * { - -webkit-transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); - transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); - opacity: 1; - filter: alpha(opacity=100); } - - /* Item - ==================================*/ - .item { - font-size: 14px; - width: 100%; } - - .item-icon-left .icon, .item-icon-left .icon-help, .item-icon-left .icon-alert, .item-icon-left #menu .footer .icon-help, #menu .footer .item-icon-left .icon-help { - left: 16px; } - - .item-icon-right .icon, .item-icon-right .icon-help, .item-icon-right .icon-alert, .item-icon-right #menu .footer .icon-help, #menu .footer .item-icon-right .icon-help { - right: 16px; } - - /* - .list .item.item-icon-right { - padding-right: 60px; - } - */ - .item-thumbnail-left > img:first-child, .item-thumbnail-left .item-image, .item-thumbnail-left .item-content > img:first-child, .item-thumbnail-left .item-content .item-image { - border-radius: 50%; } - - .tab-item.activated { - height: calc(100% + 3px); - /* Stretch */ } - - /* List - ==================================*/ - .content + .list { - padding-top: 0; } - - .list .item { - border: none; - /* - padding-left: 16px; - padding-right: 16px; - */ - min-height: 48px; - text-align: left; } - - .list .item.tabs { - padding: initial; } - - .list .item.item-bg-image { - max-height: 150px; - min-height: 150px; } - - .list .item.item-bg-image > img { - height: 100%; - left: 0; - max-width: initial; - opacity: 0.65; - filter: alpha(opacity=65); - position: absolute; - top: 0; - width: 100%; - z-index: 0; } - - .list a.item { - opacity: 1; - filter: alpha(opacity=100); } - - .list .item.item-bg-image h1, .list .item.item-bg-image h2, .list .item.item-bg-image h3, .list .item.item-bg-image h4, .list .item.item-bg-image h5, .list .item.item-bg-image h6 { - color: #fff; - font-weight: bold; - position: relative; - text-shadow: 0 0 3px rgba(0, 0, 0, 0.95); - z-index: 1; } - - .list .item.item-bg-image h2 { - font-size: 24px; } - - .list .item.item-bg-image h2 { - font-size: 24px; } - - .list .item.item-bg-image p { - color: white; - font-size: 17px; - position: relative; - text-shadow: 0 0 4px rgba(0, 0, 0, 0.95); - z-index: 1; } - - .item-avatar, .item-avatar .item-content, .item-avatar-left, .item-avatar-left .item-content { - min-height: 80px; } - - /* List: Thumbnails - ==================================*/ - .item-thumbnail-left, .card > .item.item-thumbnail-left, .item-thumbnail-left .item-content { - padding-left: 106px; } - - .item-thumbnail-right, .card > .item.item-thumbnail-right, .item-thumbnail-right .item-content { - padding-right: 106px; } - - /* List: Avatar - ==================================*/ - .item-avatar > img:first-child, .item-avatar .item-image, .item-avatar .item-content > img:first-child, .item-avatar .item-content .item-image, .item-avatar-left > img:first-child, .item-avatar-left .item-image, .item-avatar-left .item-content > img:first-child, .item-avatar-left .item-content .item-image { - border-radius: 50%; - left: 16px; - max-height: 40px; - max-width: 40px; } - - /* - .item-avatar, .list .item-avatar { - padding-left: 100px; - } - */ - .avatar, .item-avatar .avatar { - background-position: center; - background-size: cover; - border-radius: 50%; - display: inline-block; - height: 56px; - left: 16px; - position: absolute; - width: 56px; } - - /* List: Gallery - ==================================*/ - .list.half { - display: inline-block; - float: left; - margin: 0; - padding: 0; - width: 50%; } - - .list.half:first-child { - padding: 16px 8px 16px 16px; } - - .list.half:last-child { - padding: 16px 16px 16px 8px; } - - .list.half:first-child .card.card-gallery { - margin-left: 0; - margin-right: 0; } - - .list.half:last-child .card.card-gallery { - margin-left: 0; - margin-right: 0; } - - .list.condensed-space > .card, .list.condensed-space > .item { - margin: 0px 0px 2px; } - - .list .card.card-gallery { - display: block; - float: left; - margin: 0 0 0 13px; - padding: 0; - width: auto; } - - .list.half .item { - width: 100%; } - - .list.half .item.card { - margin-bottom: 16px; } - - .list .card.card-gallery.item h2 { - padding: 12px; } - - .list .item.item-gallery img { - width: 100%; } - - .item.item-divider { - border-top: solid 1px rgba(0, 0, 0, 0.12); - font-size: 14px; - font-weight: bold; - height: 48px; - line-height: 48px; - color: rgba(0, 0, 0, 0.54); } - .item.item-divider:first-child { - border: none; } - - .item-avatar, .item-avatar .item-content, .item-avatar-left, .item-avatar-left .item-content, .card > .item-avatar { - padding-left: 72px; } - - .item.active, .item.activated, .item-complex.active .item-content, .item-complex.activated .item-content, .item .item-content.active, .item .item-content.activated { - background-color: transparent; } - - .list-inset { - margin: 20px 30px; - border-left: solid 1px #ccc; - border-radius: 0; - background-color: #fff; } - - .list .item.item-floating-label, - .item-floating-label { - border-bottom: solid 1px #ccc; } - - .loader { - position: relative; - margin: 0px auto; - width: 100px; - height: 100px; - zoom: 1.7; } - - .circular { - -webkit-animation: rotate 2s linear infinite; - animation: rotate 2s linear infinite; - height: 100px; - position: relative; - width: 100px; } - - .path { - stroke-dasharray: 1,200; - stroke-dashoffset: 0; - -webkit-animation: dash 1.5s ease-in-out infinite, color 6s ease-in-out infinite; - animation: dash 1.5s ease-in-out infinite, color 6s ease-in-out infinite; - stroke-linecap: round; } - - @-webkit-keyframes rotate { - 100% { - -webkit-transform: rotate(360deg); - transform: rotate(360deg); } } - - @keyframes rotate { - 100% { - -webkit-transform: rotate(360deg); - transform: rotate(360deg); } } - - @-webkit-keyframes dash { - 0% { - stroke-dasharray: 1,200; - stroke-dashoffset: 0; } - 50% { - stroke-dasharray: 89,200; - stroke-dashoffset: -35; } - 100% { - stroke-dasharray: 89,200; - stroke-dashoffset: -124; } } - - @keyframes dash { - 0% { - stroke-dasharray: 1,200; - stroke-dashoffset: 0; } - 50% { - stroke-dasharray: 89,200; - stroke-dashoffset: -35; } - 100% { - stroke-dasharray: 89,200; - stroke-dashoffset: -124; } } - - @-webkit-keyframes color { - 100%, 0% { - stroke: #d62d20; } - 40% { - stroke: #0057e7; } - 66% { - stroke: #008744; } - 80%, 90% { - stroke: #ffa700; } } - - @keyframes color { - 100%, 0% { - stroke: #d62d20; } - 40% { - stroke: #0057e7; } - 66% { - stroke: #008744; } - 80%, 90% { - stroke: #ffa700; } } - - /* Layouts: Login - ==================================*/ - .login { - background-position: 25% 25%; - background-size: 180% 180%; - height: 100%; - -webkit-transition: all 1.5s ease-in-out; - transition: all 1.5s ease-in-out; } - - .login .item { - margin: 0 12px; - padding-left: 0; - padding-right: 0; - width: initial; } - - .login .button-bar { - bottom: 0; - margin: 28px 12px 0; - width: initial; } - - .login .light-bg { - background-color: #fff; } - - .icon.hero-icon:before, .hero-icon.icon-help:before, .hero-icon.icon-alert:before, #menu .footer .hero-icon.icon-help:before { - line-height: 130px; } - - /* Mask - ==================================*/ - .hero.has-mask:after, .item.has-mask:after, .card.has-mask:after { - content: ''; - background: -webkit-linear-gradient(top, transparent 0%, rgba(0, 0, 0, 0.6) 100%); - height: 100%; - left: 0; - position: absolute; - top: 0; - z-index: 0; - width: 100%; } - - .hero.has-mask-reverse:after, .item.has-mask-reverse:after, .card.has-mask-reverse:after { - content: ''; - background: -webkit-linear-gradient(top, rgba(0, 0, 0, 0.6) 0%, transparent 100%); - height: 100%; - left: 0; - position: absolute; - top: 0; - z-index: 0; - width: 100%; } - - /* Menu */ - .menu-bottom { - bottom: 16px; - left: 16px; - right: 16px; - position: absolute; } - - .menu-top { - top: 16px; - left: 16px; - right: 16px; - position: absolute; } - - .menu .avatar { - top: 16px; - left: 16px; - height: 65px; - width: 65px; } - - .menu .bar.bar-header.expanded { - box-shadow: none; - min-height: 150px; - color: #fff; } - - .menu-open .bar.bar-header.expanded { - background-position: 0; - background-size: 100%; } - - .has-expanded-header { - top: 150px !important; } - - .motion { - -webkit-transition: all 0.5s ease-out; - transition: all 0.5s ease-out; } - - .fade { - opacity: 0; - filter: alpha(opacity=0); - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .spin-back { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(360deg) scale(0) !important; - transform: translateZ(0) rotate(360deg) scale(0) !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .spiral { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; - transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .spiral-back { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; - transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; - -webkit-transition: all 0.1s ease-out !important; - transition: all 0.1s ease-out !important; } - - .menu-open .avatar { - opacity: 1; - filter: alpha(opacity=100); - -webkit-transform: translateZ(0) rotate(0) scale(1) !important; - transform: translateZ(0) rotate(0) scale(1) !important; - -webkit-transition: all 0.3s ease-out !important; - transition: all 0.3s ease-out !important; } - - .spin { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotate(0) scale(0) !important; - transform: translateZ(0) rotate(0) scale(0) !important; - -webkit-transition: all 0.3s ease-out !important; - transition: all 0.3s ease-out !important; } - - .spin.on { - -webkit-transform: translateZ(0) rotate(-360deg) scale(1) !important; - transform: translateZ(0) rotate(-360deg) scale(1) !important; } - - .flap { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; - transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .flap.on { - -webkit-transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; - transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .drop { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) scale(3) !important; - transform: translateZ(0) scale(3) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .drop.on { - -webkit-transform: translateZ(0) scale(1) !important; - transform: translateZ(0) scale(1) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .flip { - -webkit-backface-visibility: hidden !important; - backface-visibility: hidden !important; - -webkit-transform: translateZ(0) rotateY(0) scale(0) !important; - transform: translateZ(0) rotateY(0) scale(0) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - .flip.on { - -webkit-transform: translateZ(0) rotateY(-720deg) scale(1) !important; - transform: translateZ(0) rotateY(-720deg) scale(1) !important; - -webkit-transition: all 0.5s ease-out !important; - transition: all 0.5s ease-out !important; } - - /* Utilities - ==================================*/ - .bold { - font-weight: bold; } - - .static { - position: static; } - - .pull-left, .popover-helptip .icon.icon-left, .popover-helptip .icon-left.icon-help, .popover-helptip .icon-left.icon-alert, .popover-helptip #menu .footer .icon-left.icon-help, #menu .footer .popover-helptip .icon-left.icon-help, .popover-helptip .icon.icon-bottom-left, .popover-helptip .icon-bottom-left.icon-help, .popover-helptip .icon-bottom-left.icon-alert, .popover-helptip #menu .footer .icon-bottom-left.icon-help, #menu .footer .popover-helptip .icon-bottom-left.icon-help { - float: left; } - - .pull-right, .popover-helptip .icon.icon-right, .popover-helptip .icon-right.icon-help, .popover-helptip .icon-right.icon-alert, .popover-helptip #menu .footer .icon-right.icon-help, #menu .footer .popover-helptip .icon-right.icon-help, .popover-helptip .icon.icon-center, .popover-helptip .icon-center.icon-help, .popover-helptip .icon-center.icon-alert, .popover-helptip #menu .footer .icon-center.icon-help, #menu .footer .popover-helptip .icon-center.icon-help, .popover-helptip .icon.icon-bottom-right, .popover-helptip .icon-bottom-right.icon-help, .popover-helptip .icon-bottom-right.icon-alert, .popover-helptip #menu .footer .icon-bottom-right.icon-help, #menu .footer .popover-helptip .icon-bottom-right.icon-help, .popover-helptip .icon.icon-bottom-center, .popover-helptip .icon-bottom-center.icon-help, .popover-helptip .icon-bottom-center.icon-alert, .popover-helptip #menu .footer .icon-bottom-center.icon-help, #menu .footer .popover-helptip .icon-bottom-center.icon-help { - float: right; } - - .double-padding, .ionic-content.double-padding { - padding: 16px; } - - .double-padding-x { - padding-left: 16px; - padding-right: 16px; } - - .double-padding-y { - padding-top: 16px; - padding-bottom: 16px; } - - .outline { - border-style: solid; - border-width: 1px; } - - .border-top { - border-top: solid 1px #ccc; - padding-top: 30px; } - - .no-border { - border: none; } - - .circle { - border-radius: 50%; } - - .no-padding, .list.no-padding, .bar.no-padding, .button-bar.no-padding, .card.no-padding, .button.no-padding, .item.no-padding { - padding: 0; } - - .flat, .flat.tabs, .flat.button, .flat.button.icon, .flat.button.icon-help, .flat.button.icon-alert, #menu .footer .flat.button.icon-help, .flat.hero { - box-shadow: none; - -webkit-box-shadow: none; } - - /* Utilities : Padding - ==================================*/ - .im-wrapper, .padding, .item.large-button-bar { - padding: 16px !important; } - - .padding-bottom, .popover-share .bar-footer .button-close { - padding-bottom: 16px !important; } - - .padding-top { - padding-top: 16px !important; } - - .padding-left { - padding-left: 16px !important; } - - .padding-right, .popover-share .bar-footer .button-close { - padding-right: 16px !important; } - - .no-padding-bottom { - padding-bottom: 0 !important; } - - .no-padding-top { - padding-top: 0 !important; } - - .no-padding-left { - padding-left: 0 !important; } - - .no-padding-right { - padding-right: 0 !important; } - - /* Utilities : Depth - ==================================*/ - .z1 { - box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); } - - /* Utilities : Color - ==================================*/ - .bar.bar-positive.darker { - background-color: #164FAB; } - - /* TODO: Expand to other colors */ - .bar.bar-positive.dark-positive-bg { - background-color: #2C5CAD; } - - /* TODO: Expand to other colors */ - .muted { - color: #C3C3C3; } - - .clear-bg { - background: transparent; } - - /* Motion: Blinds - ==================================*/ - .animate-blinds .item, - .animate-blinds .item { - visibility: hidden; } - - .animate-blinds .item, - .animate-blinds .item { - -ms-transform: scale3d(0.8, 0, 1); - -webkit-transform: scale3d(0.8, 0, 1); - transform: scale3d(0.8, 0, 1); - -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } - - .animate-blinds .item-bg-image > img.background, - .animate-blinds .item-bg-image > img.background { - box-shadow: none; - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); } - - .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded .animate-blinds .item, .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, - .animate-blinds.done > *, .animate-fade-slide-in .expanded .animate-blinds.item > *, - .animate-fade-slide-in .animate-blinds.expanded.item > *, - .animate-fade-slide-in-right .expanded .animate-blinds.item > *, - .animate-fade-slide-in-right .animate-blinds.expanded.item > *, - .animate-ripple .expanded .animate-blinds.item > *, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment > *, - .animate-blinds .in, - .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-blinds .expanded.item, - .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-blinds .expanded.item, - .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded - .animate-blinds .item, - .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-blinds .card.card-comment, - .animate-blinds.done > *, .animate-fade-slide-in .expanded .animate-blinds.item > *, - .animate-fade-slide-in .animate-blinds.expanded.item > *, - .animate-fade-slide-in-right .expanded .animate-blinds.item > *, - .animate-fade-slide-in-right .animate-blinds.expanded.item > *, - .animate-ripple .expanded .animate-blinds.item > *, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment > * { - -ms-transform: translate3d(0, 0, 0); - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded .animate-blinds .item, .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, - .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, - .animate-fade-slide-in .animate-blinds.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-blinds.item .item, - .animate-fade-slide-in-right .animate-blinds.expanded.item .item, - .animate-ripple .expanded .animate-blinds.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item, - .animate-blinds .in, - .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-blinds .expanded.item, - .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-blinds .expanded.item, - .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded - .animate-blinds .item, - .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-blinds .card.card-comment, - .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, - .animate-fade-slide-in .animate-blinds.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-blinds.item .item, - .animate-fade-slide-in-right .animate-blinds.expanded.item .item, - .animate-ripple .expanded .animate-blinds.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item { - visibility: visible; } - - .animate-blinds .item, - .animate-blinds .item { - visibility: hidden; } - - .animate-blinds .item, - .animate-blinds .item { - opacity: 0; - filter: alpha(opacity=0); } - - .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded .animate-blinds .item, .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, - .animate-blinds.done, - .animate-fade-slide-in .expanded .animate-blinds.item, - .animate-fade-slide-in .animate-blinds.expanded.item, - .animate-fade-slide-in-right .expanded .animate-blinds.item, - .animate-fade-slide-in-right .animate-blinds.expanded.item, - .animate-ripple .expanded .animate-blinds.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment, - .animate-blinds .in, - .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-blinds .expanded.item, - .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-blinds .expanded.item, - .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded - .animate-blinds .item, - .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-blinds .card.card-comment, - .animate-blinds.done, - .animate-fade-slide-in .expanded .animate-blinds.item, - .animate-fade-slide-in .animate-blinds.expanded.item, - .animate-fade-slide-in-right .expanded .animate-blinds.item, - .animate-fade-slide-in-right .animate-blinds.expanded.item, - .animate-ripple .expanded .animate-blinds.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment { - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; - opacity: 1; - filter: alpha(opacity=100); } - - .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded .animate-blinds .item, .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, - .animate-blinds.done, - .animate-fade-slide-in .expanded .animate-blinds.item, - .animate-fade-slide-in .animate-blinds.expanded.item, - .animate-fade-slide-in-right .expanded .animate-blinds.item, - .animate-fade-slide-in-right .animate-blinds.expanded.item, - .animate-ripple .expanded .animate-blinds.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment, - .animate-blinds .in, - .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-blinds .expanded.item, - .animate-blinds - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded - .animate-blinds .item, - .animate-blinds - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-blinds .expanded.item, - .animate-blinds - .animate-ripple .expanded .item, - .animate-ripple .expanded - .animate-blinds .item, - .animate-blinds - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-blinds .card.card-comment, - .animate-blinds.done, - .animate-fade-slide-in .expanded .animate-blinds.item, - .animate-fade-slide-in .animate-blinds.expanded.item, - .animate-fade-slide-in-right .expanded .animate-blinds.item, - .animate-fade-slide-in-right .animate-blinds.expanded.item, - .animate-ripple .expanded .animate-blinds.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment { - visibility: visible; } - - .animate-blinds.done .in, .animate-fade-slide-in .expanded .animate-blinds.item .in, - .animate-fade-slide-in .animate-blinds.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-blinds.item .in, - .animate-fade-slide-in-right .animate-blinds.expanded.item .in, - .animate-ripple .expanded .animate-blinds.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .in, .animate-blinds.done .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, .animate-fade-slide-in .animate-blinds.expanded.item .item, .animate-blinds.done - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-blinds.done .expanded.item, .animate-blinds.done - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded .animate-blinds.done .item, .animate-fade-slide-in-right .expanded .animate-blinds.item .item, .animate-fade-slide-in-right .animate-blinds.expanded.item .item, .animate-blinds.done - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-blinds.done .expanded.item, .animate-blinds.done - .animate-ripple .expanded .item, - .animate-ripple .expanded .animate-blinds.done .item, .animate-ripple .expanded .animate-blinds.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item, .animate-blinds.done - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.done .card.card-comment, - .animate-fade-slide-in - .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-fade-slide-in-right - .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in-right .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-blinds.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .card.card-comment, - .animate-blinds.done .in, .animate-fade-slide-in .expanded .animate-blinds.item .in, - .animate-fade-slide-in .animate-blinds.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-blinds.item .in, - .animate-fade-slide-in-right .animate-blinds.expanded.item .in, - .animate-ripple .expanded .animate-blinds.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .in, - .animate-blinds.done .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded - .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, .animate-fade-slide-in .animate-blinds.expanded.item .item, - .animate-blinds.done - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-blinds.done .expanded.item, - .animate-blinds.done - .animate-fade-slide-in-right .expanded .item, - .animate-fade-slide-in-right .expanded - .animate-blinds.done .item, .animate-fade-slide-in-right .expanded .animate-blinds.item .item, .animate-fade-slide-in-right .animate-blinds.expanded.item .item, - .animate-blinds.done - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-blinds.done .expanded.item, - .animate-blinds.done - .animate-ripple .expanded .item, - .animate-ripple .expanded - .animate-blinds.done .item, .animate-ripple .expanded .animate-blinds.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item, - .animate-blinds.done - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-blinds.done .card.card-comment, - .animate-fade-slide-in - .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-fade-slide-in-right - .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in-right .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-blinds.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .card.card-comment { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-blinds .has-mask-reverse:after, - .animate-blinds .has-mask-reverse:after { - opacity: 0; - filter: alpha(opacity=0); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; } - - .animate-blinds.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-blinds.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-blinds.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-blinds.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-blinds.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-blinds.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .has-mask-reverse:after, - .animate-blinds.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-blinds.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-blinds.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-blinds.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-blinds.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-blinds.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .has-mask-reverse:after { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-blinds .out, - .animate-blinds .out { - -ms-transform: scale3d(0, 0, 1); - -webkit-transform: scale3d(0, 0, 1); - transform: scale3d(0, 0, 1); } - - /* Motion: Pan In Left - ==================================*/ - .animate-pan-in-left, - .animate-pan-in-left { - background-position: 0% 0%; } - - /* Motion: Ripple - ==================================*/ - .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .done, .animate-fade-slide-in .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { - visibility: hidden; } - - .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .done, .animate-fade-slide-in .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { - -ms-transform: scale3d(0.8, 0, 1); - -webkit-transform: scale3d(0.8, 0, 1); - transform: scale3d(0.8, 0, 1); - -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } - - .animate-ripple .item-bg-image img.background, - .animate-ripple .item-bg-image img.background { - box-shadow: none; - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); } - - .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, - .animate-ripple .in, .animate-fade-slide-in .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); } - - .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, - .animate-ripple .in, .animate-fade-slide-in .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { - visibility: visible; } - - .animate-ripple .item { - -ms-transform: scale3d(0, 0, 1); - -webkit-transform: scale3d(0, 0, 1); - transform: scale3d(0, 0, 1); - opacity: 0; - filter: alpha(opacity=0); } - - .animate-ripple .item.in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .item.expanded, - .animate-fade-slide-in .animate-ripple .item.expanded, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .item.expanded, - .animate-fade-slide-in-right .animate-ripple .item.expanded, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .item.card.card-comment { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { - visibility: hidden; } - - .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .done, .animate-fade-slide-in .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in - .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-ripple .item, - .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { - -ms-transform: scale3d(0.8, 0, 1); - -webkit-transform: scale3d(0.8, 0, 1); - transform: scale3d(0.8, 0, 1); - -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } - - /* Uncomment if you want images to fade in after the card - - .animate-ripple .in .item-bg-image img:last-child, - .animate-ripple .in .item-bg-image img:last-child { - opacity: 0; - } - - .animate-ripple.done .item-bg-image img:last-child, - .animate-ripple.done .item-bg-image img:last-child { - opacity: 1; - -moz-transition: all 1s ease-in-out; - -o-transition: all 1s ease-in-out; - -webkit-transition: all 1s ease-in-out; - transition: all 1s ease-in-out; - } - - .animate-ripple .item-bg-image img:last-child, - .animate-ripple .item-bg-image img:last-child { - box-shadow: none; - -moz-transform: scale3d(1, 1, 1); - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); - } - .animate-ripple .in .item-bg-image img:last-child, - .animate-ripple .in .item-bg-image img:last-child { - opacity: 0; - } - - .animate-ripple.done .item-bg-image img:last-child, - .animate-ripple.done .item-bg-image img:last-child { - opacity: 1; - -moz-transition: all 0.3s ease-in-out; - -o-transition: all 0.3s ease-in-out; - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; - } - - .animate-ripple .in, - .animate-ripple .in { - opacity: 0.6; - } - */ - .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; } - - .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, - .animate-fade-slide-in .animate-ripple.expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.item, - .animate-fade-slide-in-right .animate-ripple.expanded.item, - .animate-ripple .expanded .animate-ripple.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { - visibility: visible; } - - .animate-ripple.done .in, .animate-fade-slide-in .expanded .animate-ripple.item .in, - .animate-fade-slide-in .animate-ripple.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-ripple.item .in, - .animate-fade-slide-in-right .animate-ripple.expanded.item .in, - .animate-ripple .expanded .animate-ripple.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .in, .animate-fade-slide-in .expanded .animate-ripple.done .item, .animate-fade-slide-in .expanded .animate-ripple.item .item, .animate-fade-slide-in .animate-ripple.expanded.item .item, .animate-ripple.done - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple.done .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.done .item, .animate-fade-slide-in-right .expanded .animate-ripple.item .item, .animate-fade-slide-in-right .animate-ripple.expanded.item .item, .animate-ripple.done - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple.done .expanded.item, .animate-ripple.done .expanded .item, .animate-ripple .expanded .animate-ripple.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .item, .animate-ripple.done .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-fade-slide-in .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-fade-slide-in-right .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-ripple.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .card.card-comment, .animate-ripple.done .in, .animate-fade-slide-in .expanded .animate-ripple.item .in, - .animate-fade-slide-in .animate-ripple.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-ripple.item .in, - .animate-fade-slide-in-right .animate-ripple.expanded.item .in, - .animate-ripple .expanded .animate-ripple.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .in, .animate-fade-slide-in .expanded .animate-ripple.done .item, .animate-fade-slide-in .expanded .animate-ripple.item .item, .animate-fade-slide-in .animate-ripple.expanded.item .item, .animate-ripple.done - .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in .animate-ripple.done .expanded.item, - .animate-fade-slide-in-right .expanded .animate-ripple.done .item, .animate-fade-slide-in-right .expanded .animate-ripple.item .item, .animate-fade-slide-in-right .animate-ripple.expanded.item .item, .animate-ripple.done - .animate-fade-slide-in-right .expanded.item, - .animate-fade-slide-in-right .animate-ripple.done .expanded.item, .animate-ripple.done .expanded .item, .animate-ripple .expanded .animate-ripple.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .item, .animate-ripple.done .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-fade-slide-in .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-fade-slide-in-right .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-ripple.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .card.card-comment { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-ripple .has-mask-reverse:after, .animate-ripple .has-mask-reverse:after { - opacity: 0; - filter: alpha(opacity=0); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; } - - .animate-ripple.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-ripple.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-ripple.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-ripple.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-ripple.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-ripple.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .has-mask-reverse:after, .animate-ripple.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-ripple.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-ripple.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-ripple.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-ripple.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-ripple.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .has-mask-reverse:after { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-ripple .out, .animate-ripple .out { - -ms-transform: scale3d(0, 0, 1); - -webkit-transform: scale3d(0, 0, 1); - transform: scale3d(0, 0, 1); } - - /* Motion: Slide / Fade In - ==================================*/ - .animate-fade-slide-in .item, - .animate-fade-slide-in .item { - visibility: hidden; } - - .animate-fade-slide-in .item, - .animate-fade-slide-in .item { - -ms-transform: scale3d(0.8, 0, 1); - -webkit-transform: scale3d(0.8, 0, 1); - transform: scale3d(0.8, 0, 1); - -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } - - .animate-fade-slide-in .item-bg-image img.background, - .animate-fade-slide-in .item-bg-image img.background { - box-shadow: none; - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); } - - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, - .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-fade-slide-in .item, - .animate-ripple .expanded - .animate-fade-slide-in .item, - .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item { - -ms-transform: translate3d(0, 0, 0); - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, - .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-fade-slide-in .item, - .animate-ripple .expanded - .animate-fade-slide-in .item, - .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item { - visibility: visible; } - - .list .item.item-bg-image, - .list .item.item-bg-image { - max-height: 150px; } - - .animate-fade-slide-in .item, - .animate-fade-slide-in .item { - visibility: hidden; } - - .animate-fade-slide-in .item, - .animate-fade-slide-in .item { - -ms-transform: translate3d(-250px, 250px, 0); - -webkit-transform: translate3d(-250px, 250px, 0); - transform: translate3d(-250px, 250px, 0); - -webkit-transition: -webkit-transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); - opacity: 0; - filter: alpha(opacity=0); } - - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, - .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment, - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-fade-slide-in .item, - .animate-ripple .expanded - .animate-fade-slide-in .item, - .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment { - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); - -webkit-transition: all 0.5s ease-in-out; - transition: all 0.5s ease-in-out; - opacity: 1; - filter: alpha(opacity=100); } - - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, - .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment, - .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-fade-slide-in .item, - .animate-ripple .expanded - .animate-fade-slide-in .item, - .animate-fade-slide-in - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in .card.card-comment, - .animate-fade-slide-in.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment { - visibility: visible; } - - .animate-fade-slide-in.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .in, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .in, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .in, - .animate-ripple .expanded .animate-fade-slide-in.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .in, .animate-fade-slide-in.done .expanded .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, .animate-fade-slide-in.done .expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.done .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in.done .item, .animate-ripple .expanded .animate-fade-slide-in.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, .animate-fade-slide-in.done - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.done .card.card-comment, - .animate-ripple .animate-fade-slide-in.expanded.item .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.expanded.item .card.card-comment, - .animate-fade-slide-in-right - .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .card.card-comment, - .animate-fade-slide-in.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .in, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .in, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .in, - .animate-ripple .expanded .animate-fade-slide-in.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .in, .animate-fade-slide-in.done .expanded .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, .animate-fade-slide-in.done .expanded.item, - .animate-fade-slide-in-right .expanded - .animate-fade-slide-in.done .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, - .animate-ripple .expanded - .animate-fade-slide-in.done .item, .animate-ripple .expanded .animate-fade-slide-in.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, - .animate-fade-slide-in.done - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in.done .card.card-comment, - .animate-ripple .animate-fade-slide-in.expanded.item .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.expanded.item .card.card-comment, - .animate-fade-slide-in-right - .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .card.card-comment { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-fade-slide-in .has-mask-reverse:after, - .animate-fade-slide-in .has-mask-reverse:after { - opacity: 0; - filter: alpha(opacity=0); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; } - - .animate-fade-slide-in.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-fade-slide-in.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .has-mask-reverse:after, - .animate-fade-slide-in.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-fade-slide-in.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-fade-slide-in.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .has-mask-reverse:after { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-fade-slide-in .out, - .animate-fade-slide-in .out { - -ms-transform: scale3d(0, 0, 1); - -webkit-transform: scale3d(0, 0, 1); - transform: scale3d(0, 0, 1); } - - /* Motion: Slide In Right - ==================================*/ - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right .item { - visibility: hidden; } - - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right .item { - -ms-transform: scale3d(0.8, 0, 1); - -webkit-transform: scale3d(0.8, 0, 1); - transform: scale3d(0.8, 0, 1); - -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } - - .animate-fade-slide-in-right .item-bg-image > img.background, - .animate-fade-slide-in-right .item-bg-image > img.background { - box-shadow: none; - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); } - - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done > *, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item > *, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item > *, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item > *, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item > *, - .animate-ripple .expanded .animate-fade-slide-in-right.item > *, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment > *, - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded - .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done > *, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item > *, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item > *, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item > *, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item > *, - .animate-ripple .expanded .animate-fade-slide-in-right.item > *, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment > * { - -ms-transform: translate3d(0, 0, 0); - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in-right.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item, - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded - .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, - .animate-ripple .expanded .animate-fade-slide-in-right.item .item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item { - visibility: visible; } - - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right .item { - visibility: hidden; } - - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right .item { - -ms-transform: translate3d(250px, 250px, 0); - -webkit-transform: translate3d(250px, 250px, 0); - transform: translate3d(250px, 250px, 0); - -webkit-transition: -webkit-transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); - transition: transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); - opacity: 0; - filter: alpha(opacity=0); } - - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment, - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded - .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment { - -ms-transform: scale3d(1, 1, 1); - -webkit-transform: scale3d(1, 1, 1); - transform: scale3d(1, 1, 1); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; - opacity: 1; - filter: alpha(opacity=100); } - - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment, - .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded - .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, - .animate-ripple .expanded - .animate-fade-slide-in-right .item, - .animate-fade-slide-in-right - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in-right .card.card-comment, - .animate-fade-slide-in-right.done, - .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right.item, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment { - visibility: visible; } - - .animate-fade-slide-in-right.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .in, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .in, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .in, - .animate-ripple .expanded .animate-fade-slide-in-right.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded.item, - .animate-ripple .expanded .animate-fade-slide-in-right.done .item, .animate-ripple .expanded .animate-fade-slide-in-right.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item, .animate-fade-slide-in-right.done - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.done .card.card-comment, - .animate-fade-slide-in - .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .animate-fade-slide-in-right.expanded.item .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.expanded.item .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in-right.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .card.card-comment, - .animate-fade-slide-in-right.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .in, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .in, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .in, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .in, - .animate-ripple .expanded .animate-fade-slide-in-right.item .in, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .in, .animate-fade-slide-in .expanded - .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded.item, - .animate-ripple .expanded - .animate-fade-slide-in-right.done .item, .animate-ripple .expanded .animate-fade-slide-in-right.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item, - .animate-fade-slide-in-right.done - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded - .animate-fade-slide-in-right.done .card.card-comment, - .animate-fade-slide-in - .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .animate-fade-slide-in-right.expanded.item .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, - .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.expanded.item .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in-right.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .card.card-comment { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-fade-slide-in-right .has-mask-reverse:after, - .animate-fade-slide-in-right .has-mask-reverse:after { - opacity: 0; - filter: alpha(opacity=0); - -webkit-transition: all 0.3s ease-in-out; - transition: all 0.3s ease-in-out; } - - .animate-fade-slide-in-right.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .has-mask-reverse:after, - .animate-fade-slide-in-right.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, - .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, - .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, - .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, - .animate-ripple .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, - .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .has-mask-reverse:after { - opacity: 1; - filter: alpha(opacity=100); } - - .animate-fade-slide-in-right .out, - .animate-fade-slide-in-right .out { - -ms-transform: scale3d(0, 0, 1); - -webkit-transform: scale3d(0, 0, 1); - transform: scale3d(0, 0, 1); } - - /* Motion: Slide Up - ==================================*/ - .slide-up, - .slide-up, - .hero.slide-up { - height: 100%; - overflow: hidden; - text-align: center; } - - .slide-up { - -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); - -webkit-transform: translate3d(0, 0, 0); - transform: translate3d(0, 0, 0); } - - .slide-up *, - .slide-up *, - .hero.slide-up * { - opacity: 0; - filter: alpha(opacity=0); } - - .hero.slide-up + .mid-bar, - .slide-up + .mid-bar, - .slide-up + .mid-bar { - height: 100%; - opacity: 0.7; - filter: alpha(opacity=70); - -webkit-transform: translate3d(100%, -240px, 0); - transform: translate3d(100%, -240px, 0); } - - /*! - * Waves v0.5.4 - * http://fian.my.id/Waves - * - * Copyright 2014 Alfiana E. Sibuea and other contributors - * Forked by Zach Fitzgerald and other contributors for Ionic Material - * - * Released under the MIT license - * https://github.com/fians/Waves/blob/master/LICENSE - * - */ - .ink, .button-fab, .button-flat, .button-raised, .button-clear, .button-text, .popup .button { - position: relative; - cursor: pointer; - /*display: inline-block;*/ - overflow: hidden; - -webkit-user-select: none; - -moz-user-select: none; - -ms-user-select: none; - user-select: none; - -webkit-tap-highlight-color: transparent; - -webkit-transition: all 0.3s ease-out; - -moz-transition: all 0.3s ease-out; - -o-transition: all 0.3s ease-out; - transition: all 0.3s ease-out; } - - .ink-ripple { - position: absolute; - border-radius: 50%; - width: 100px; - height: 100px; - margin-top: -50px; - margin-left: -50px; - opacity: 0; - background-color: rgba(255, 255, 255, 0.4); - -webkit-transition: all 0.5s ease-out; - -moz-transition: all 0.5s ease-out; - -o-transition: all 0.5s ease-out; - transition: all 0.5s ease-out; - -webkit-transition-property: -webkit-transform, opacity; - -moz-transition-property: -moz-transform, opacity; - -o-transition-property: -o-transform, opacity; - transition-property: transform, opacity; - -webkit-transform: scale(0); - -moz-transform: scale(0); - -ms-transform: scale(0); - -o-transform: scale(0); - transform: scale(0); - pointer-events: none; } - - .ink-notransition { - -webkit-transition: none !important; - -moz-transition: none !important; - -o-transition: none !important; - transition: none !important; } - - .button-fab, - .button-flat, - .button-clear, - .button-text, - .button-raised, - .ink-button, - .ink-circle { - -webkit-transform: translateZ(0); - -moz-transform: translateZ(0); - -ms-transform: translateZ(0); - -o-transform: translateZ(0); - transform: translateZ(0); } - - .button-fab.activated, - .button-flat.activated, - .button-raised.activated, - .button-clear.activated, - .activated.button-text, - .ink-button.activated, - .ink.activated, - .ink-circle.activated, - .popup .button.activated, - .button-fab:active, - .button-flat:active, - .button-raised:active, - .button-clear:active, - .button-text:active, - .ink-button:active, - .ink:active, - .ink-circle:active, - .popup .button:active { - -webkit-mask-image: -webkit-radial-gradient(circle, #ffffff 100%, #000000 100%); } - - .ink-button, - .ink-button:visited, - .ink-button:link, - .button-fab, - .button-fab:visited, - .button-fab:link, - .button-flat, - .button-flat:visited, - .button-flat:link, - .button-raised, - .button-raised:visited, - .button-raised:link, - .button-clear, - .button-text, - .button-clear:visited, - .button-text:visited, - .button-clear:link, - .button-text:link { - white-space: nowrap; - vertical-align: middle; - cursor: pointer; - border: none; - outline: none; - /* color: inherit; */ - /* background-color: rgba(0, 0, 0, 0); */ - font-size: 14px; - text-align: center; - text-decoration: none; - z-index: 1; } - - - /* Ionic Overrides - ==================================*/ - * { - font-family: "RobotoDraft","Roboto","Helvetica Neue", "Segoe UI", sans-serif; } - - .rounded { - border-radius: 4px; } - - a { - cursor: pointer; } - - .has-header.expanded { - /* Expanded modifier */ - top: 76px; } - - /* Bar Overrides - ==================================*/ - .bar { - border-bottom: none; - padding: 0; } - - .bar .button { - min-height: 44px; - min-width: 44px; - max-width: 48px; - margin-bottom: 0; - max-height: 44px; - width: 48px; } - - .bar .title + .buttons.buttons-right { - right: 0; - top: 0; } - - /* Title Overrides - ==================================*/ - .title-left, - .title.title-left { - left: 48px; } - - .title-right, - .title.title-right { - left: 48px; } - - /* Background Colors - ==================================*/ - .positive-bg, - .button-positive, - .button-text, - .bar .button-positive, - .bar .button-text, - .header-positive, - .button-bar-positive, - .bar-positive, - .positive-border, - .positive-bg:hover, - .bar .button-positive:hover, - .bar .button-text:hover, - .button-positive:hover, - .button-text:hover, - .header-positive:hover, - .button-bar-positive:hover, - .bar-positive:hover, - .positive-border:hover, - .positive-bg:active, - .bar .button-positive:active, - .bar .button-text:active, - .button-positive:active, - .button-text:active, - .header-positive:active, - .button-bar-positive:active, - .bar-positive:active, - .positive-border:active, - .positive-bg.activated, - .bar .button-positive.activated, - .bar .activated.button-text, - .button-positive.activated, - .activated.button-text, - .header-positive.activated, - .button-bar-positive.activated, - .bar-positive.activated, - .positive-border.activated { - background-color: #3F51B5; - color: #fff; } - - .positive-900-bg, - .button-positive-900, - .bar .button-positive-900, - .header-positive-900, - .button-bar-positive-900, - .bar-positive-900, - .positive-900-border, - .positive-900-bg:hover, - .button-positive-900:hover, - .bar .button-positive-900:hover, - .header-positive-900:hover, - .button-bar-positive-900:hover, - .bar-positive-900:hover, - .positive-900-border:hover, - .positive-900-bg:active, - .bar .button-positive-900:active, - .button-positive-900:active, - .header-positive-900:active, - .button-bar-positive-900:active, - .bar-positive-900:active, - .positive-900-border:active, - .positive-900-bg.activated, - .button-positive-900.activated, - .bar .button-positive-900.activated, - .header-positive-900.activated, - .button-bar-positive-900.activated, - .bar-positive-900.activated, - .positive-900-border.activated { - background-color: #1A237E; - color: #fff; } - - .positive-100-bg, - .button-positive-100, - .bar .button-positive-100, - .header-positive-100, - .button-bar-positive-100, - .bar-positive-100, - .positive-100-border, - .positive-100-bg:hover, - .button-positive-100:hover, - .bar .button-positive-100:hover, - .header-positive-100:hover, - .button-bar-positive-100:hover, - .bar-positive-100:hover, - .positive-100-border:hover, - .positive-100-bg:active, - .button-positive-100:active, - .bar .button-positive-100:active, - .header-positive-100:active, - .button-bar-positive-100:active, - .bar-positive-100:active, - .positive-100-border:active, - .positive-100-bg.activated, - .button-positive-100.activated, - .bar .button-positive-100.activated, - .header-positive-100.activated, - .button-bar-positive-100.activated, - .bar-positive-100.activated, - .positive-100-border.activated { - background-color: #C5CAE9; - color: #fff; } - - .calm-bg, - .button-calm, - .bar .button-calm, - .header-calm, - .button-bar-calm, - .bar-calm, - .calm-border, - .calm-bg:hover, - .button-calm:hover, - .bar .button-calm:hover, - .header-calm:hover, - .button-bar-calm:hover, - .bar-calm:hover, - .calm-border:hover, - .calm-bg:active, - .button-calm:active, - .bar .button-calm:active, - .header-calm:active, - .button-bar-calm:active, - .bar-calm:active, - .calm-border:active, - .calm-bg.activated, - .button-calm.activated, - .bar .button-calm.activated, - .header-calm.activated, - .button-bar-calm.activated, - .bar-calm.activated, - .calm-border.activated { - background-color: #2196F3; - color: #fff; } - - .calm-900-bg, - .button-calm-900, - .bar .button-calm-900, - .header-calm-900, - .button-bar-calm-900, - .bar-calm-900, - .calm-900-border, - .calm-900-bg:hover, - .button-calm-900:hover, - .bar .button-calm-900:hover, - .header-calm-900:hover, - .button-bar-calm-900:hover, - .bar-calm-900:hover, - .calm-900-border:hover, - .calm-900-bg:active, - .button-calm-900:active, - .bar .button-calm-900:active, - .header-calm-900:active, - .button-bar-calm-900:active, - .bar-calm-900:active, - .calm-900-border:active, - .calm-900-bg.activated, - .button-calm-900.activated, - .bar .button-calm-900.activated, - .header-calm-900.activated, - .button-bar-calm-900.activated, - .bar-calm-900.activated, - .calm-900-border.activated { - background-color: #0D47A1; - color: #fff; } - - .calm-100-bg, - .button-calm-100, - .bar .button-calm-100, - .header-calm-100, - .button-bar-calm-100, - .bar-calm-100, - .calm-100-border, - .calm-100-bg:hover, - .button-calm-100:hover, - .bar .button-calm-100:hover, - .header-calm-100:hover, - .button-bar-calm-100:hover, - .bar-calm-100:hover, - .calm-100-border:hover, - .calm-100-bg:active, - .button-calm-100:active, - .bar .button-calm-100:active, - .header-calm-100:active, - .button-bar-calm-100:active, - .bar-calm-100:active, - .calm-100-border:active, - .calm-100-bg.activated, - .button-calm-100.activated, - .bar .button-calm-100.activated, - .header-calm-100.activated, - .button-bar-calm-100.activated, - .bar-calm-100.activated, - .calm-100-border.activated { - background-color: #BBDEFB; - color: #fff; } - - .royal-bg, - .button-royal, - .bar .button-royal, - .header-royal, - .button-bar-royal, - .bar-royal, - .royal-border, - .royal-bg:hover, - .button-royal:hover, - .bar .button-royal:hover, - .header-royal:hover, - .button-bar-royal:hover, - .bar-royal:hover, - .royal-border:hover, - .royal-bg:active, - .button-royal:active, - .bar .button-royal:active, - .header-royal:active, - .button-bar-royal:active, - .bar-royal:active, - .royal-border:active, - .royal-bg.activated, - .button-royal.activated, - .bar .button-royal.activated, - .header-royal.activated, - .button-bar-royal.activated, - .bar-royal.activated, - .royal-border.activated { - background-color: #673AB7; - color: #fff; } - - .royal-900-bg, - .button-royal-900, - .bar .button-royal-900, - .header-royal-900, - .button-bar-royal-900, - .bar-royal-900, - .royal-900-border, - .royal-900-bg:hover, - .button-royal-900:hover, - .bar .button-royal-900:hover, - .header-royal-900:hover, - .button-bar-royal-900:hover, - .bar-royal-900:hover, - .royal-900-border:hover, - .royal-900-bg:active, - .button-royal-900:active, - .bar .button-royal-900:active, - .header-royal-900:active, - .button-bar-royal-900:active, - .bar-royal-900:active, - .royal-900-border:active, - .royal-900-bg.activated, - .button-royal-900.activated, - .bar .button-royal-900.activated, - .header-royal-900.activated, - .button-bar-royal-900.activated, - .bar-royal-900.activated, - .royal-900-border.activated { - background-color: #311B92; - color: #fff; } - - .royal-100-bg, - .button-royal-100, - .bar .button-royal-100, - .header-royal-100, - .button-bar-royal-100, - .bar-royal-100, - .royal-100-border, - .royal-100-bg:hover, - .button-royal-100:hover, - .bar .button-royal-100:hover, - .header-royal-100:hover, - .button-bar-royal-100:hover, - .bar-royal-100:hover, - .royal-100-border:hover, - .royal-100-bg:active, - .button-royal-100:active, - .bar .button-royal-100:active, - .header-royal-100:active, - .button-bar-royal-100:active, - .bar-royal-100:active, - .royal-100-border:active, - .royal-100-bg.activated, - .button-royal-100.activated, - .bar .button-royal-100.activated, - .header-royal-100.activated, - .button-bar-royal-100.activated, - .bar-royal-100.activated, - .royal-100-border.activated { - background-color: #D1C4E9; - color: #fff; } - - .balanced-bg, - .button-balanced, - .bar .button-balanced, - .header-balanced, - .button-bar-balanced, - .bar-balanced, - .balanced-border, - .balanced-bg:hover, - .button-balanced:hover, - .bar .button-balanced:hover, - .header-balanced:hover, - .button-bar-balanced:hover, - .bar-balanced:hover, - .balanced-border:hover, - .balanced-bg:active, - .button-balanced:active, - .bar .button-balanced:active, - .header-balanced:active, - .button-bar-balanced:active, - .bar-balanced:active, - .balanced-border:active, - .balanced-bg.activated, - .button-balanced.activated, - .bar .button-balanced.activated, - .header-balanced.activated, - .button-bar-balanced.activated, - .bar-balanced.activated, - .balanced-border.activated { - background-color: #4CAF50; - color: #fff; } - - .balanced-900-bg, - .button-balanced-900, - .bar .button-balanced-900, - .header-balanced-900, - .button-bar-balanced-900, - .bar-balanced-900, - .balanced-900-border, - .balanced-900-bg:hover, - .button-balanced-900:hover, - .bar .button-balanced-900:hover, - .header-balanced-900:hover, - .button-bar-balanced-900:hover, - .bar-balanced-900:hover, - .balanced-900-border:hover, - .balanced-900-bg:active, - .button-balanced-900:active, - .bar .button-balanced-900:active, - .header-balanced-900:active, - .button-bar-balanced-900:active, - .bar-balanced-900:active, - .balanced-900-border:active, - .balanced-900-bg.activated, - .button-balanced-900.activated, - .bar .button-balanced-900.activated, - .header-balanced-900.activated, - .button-bar-balanced-900.activated, - .bar-balanced-900.activated, - .balanced-900-border.activated { - background-color: #1B5E20; - color: #fff; } - - .balanced-100-bg, - .button-balanced-100, - .bar .button-balanced-100, - .header-balanced-100, - .button-bar-balanced-100, - .bar-balanced-100, - .balanced-100-border, - .balanced-100-bg:hover, - .button-balanced-100:hover, - .bar .balanced-100-bg:hover, - .header-balanced-100:hover, - .button-bar-balanced-100:hover, - .bar-balanced-100:hover, - .balanced-100-border:hover, - .balanced-100-bg:active, - .button-balanced-100:active, - .bar .button-balanced-100:active, - .header-balanced-100:active, - .button-bar-balanced-100:active, - .bar-balanced-100:active, - .balanced-100-border:active, - .balanced-100-bg.activated, - .button-balanced-100.activated, - .bar .button-balanced-100.activated, - .header-balanced-100.activated, - .button-bar-balanced-100.activated, - .bar-balanced-100.activated, - .balanced-100-border.activated { - background-color: #C8E6C9; - color: #fff; } - - .energized-bg, - .button-energized, - .bar .button-energized, - .header-energized, - .button-bar-energized, - .bar-energized, - .energized-border, - .energized-bg:hover, - .button-energized:hover, - .bar .button-energized:hover, - .header-energized:hover, - .button-bar-energized:hover, - .bar-energized:hover, - .energized-border:hover, - .energized-bg:active, - .button-energized:active, - .bar .button-energized:active, - .header-energized:active, - .button-bar-energized:active, - .bar-energized:active, - .energized-border:active, - .energized-bg.activated, - .button-energized.activated, - .bar .button-energized.activated, - .header-energized.activated, - .button-bar-energized.activated, - .bar-energized.activated, - .energized-border.activated { - background-color: #FF9800; - color: #fff; } - - .energized-900-bg, - .button-energized-900, - .bar .button-energized-900, - .header-energized-900, - .button-bar-energized-900, - .bar-energized-900, - .energized-900-border, - .energized-900-bg:hover, - .button-energized-900:hover, - .bar .button-energized-900:hover, - .header-energized-900:hover, - .button-bar-energized-900:hover, - .bar-energized-900:hover, - .energized-900-border:hover, - .energized-900-bg:active, - .button-energized-900:active, - .bar .button-energized-900:active, - .header-energized-900:active, - .button-bar-energized-900:active, - .bar-energized-900:active, - .energized-900-border:active, - .energized-900-bg.activated, - .button-energized-900.activated, - .bar .button-energized-900.activated, - .header-energized-900.activated, - .button-bar-energized-900.activated, - .bar-energized-900.activated, - .energized-900-border.activated { - background-color: #E65100; - color: #fff; } - - .energized-100-bg, - .button-energized-100, - .bar .button-energized-100, - .header-energized-100, - .button-bar-energized-100, - .bar-energized-100, - .energized-100-border, - .energized-100-bg:hover, - .button-energized-100:hover, - .bar .button-energized-100:hover, - .header-energized-100:hover, - .button-bar-energized-100:hover, - .bar-energized-100:hover, - .energized-100-border:hover, - .energized-100-bg:active, - .button-energized-100:active, - .bar .button-energized-100:active, - .header-energized-100:active, - .button-bar-energized-100:active, - .bar-energized-100:active, - .energized-100-border:active, - .energized-100-bg.activated, - .button-energized-100.activated, - .bar .button-energized-100.activated, - .header-energized-100.activated, - .button-bar-energized-100.activated, - .bar-energized-100.activated, - .energized-100-border.activated { - background-color: #FFE0B2; } - - .assertive-bg, - .button-assertive, - .bar .button-assertive, - .header-assertive, - .button-bar-assertive, - .bar-assertive, - .assertive-border, - .assertive-bg:hover, - .button-assertive:hover, - .bar .button-assertive:hover, - .header-assertive:hover, - .button-bar-assertive:hover, - .bar-assertive:hover, - .assertive-border:hover, - .assertive-bg:active, - .button-assertive:active, - .bar .button-assertive:active, - .header-assertive:active, - .button-bar-assertive:active, - .bar-assertive:active, - .assertive-border:active, - .assertive-bg.activated, - .button-assertive.activated, - .bar .button-assertive.activated, - .header-assertive.activated, - .button-bar-assertive.activated, - .bar-assertive.activated, - .assertive-border.activated { - background-color: #F44336; - color: #fff; } - - .assertive-900-bg, - .button-assertive-900, - .bar .button-assertive-900, - .header-assertive-900, - .button-bar-assertive-900, - .bar-assertive-900, - .assertive-900-border, - .assertive-900-bg:hover, - .button-assertive-900:hover, - .bar .button-assertive-900:hover, - .header-assertive-900:hover, - .button-bar-assertive-900:hover, - .bar-assertive-900:hover, - .assertive-900-border:hover, - .assertive-900-bg:active, - .button-assertive-900:active, - .bar .button-assertive-900:active, - .header-assertive-900:active, - .button-bar-assertive-900:active, - .bar-assertive-900:active, - .assertive-900-border:active, - .assertive-900-bg.activated, - .button-assertive-900.activated, - .bar .button-assertive-900.activated, - .header-assertive-900.activated, - .button-bar-assertive-900.activated, - .bar-assertive-900.activated, - .assertive-900-border.activated { - background-color: #B71C1C; - color: #fff; } - - .assertive-100-bg, - .button-assertive-100, - .bar .button-assertive-100, - .header-assertive-100, - .button-bar-assertive-100, - .bar-assertive-100, - .assertive-100-border, - .assertive-100-bg:hover, - .button-assertive-100:hover, - .bar .button-assertive-100:hover, - .header-assertive-100:hover, - .button-bar-assertive-100:hover, - .bar-assertive-100:hover, - .assertive-100-border:hover, - .assertive-100-bg:active, - .button-assertive-100:active, - .bar .button-assertive-100:active, - .header-assertive-100:active, - .button-bar-assertive-100:active, - .bar-assertive-100:active, - .assertive-100-border:active, - .assertive-100-bg.activated, - .bar .button-assertive-100.activated, - .button-assertive-100.activated, - .header-assertive-100.activated, - .button-bar-assertive-100.activated, - .bar-assertive-100.activated, - .assertive-100-border.activated { - background-color: #FFCDD2; - color: #fff; } - - .stable-bg, - .button-stable, - .bar .button-stable, - .header-stable, - .button-bar-stable, - .bar-stable, - .stable-border, - .stable-bg:hover, - .button-stable:hover, - .bar .button-stable:hover, - .header-stable:hover, - .button-bar-stable:hover, - .bar-stable:hover, - .stable-border:hover, - .stable-bg:active, - .button-stable:active, - .bar .button-stable:active, - .header-stable:active, - .button-bar-stable:active, - .bar-stable:active, - .stable-border:active, - .stable-bg.activated, - .button-stable.activated, - .bar .button-stable.activated, - .header-stable.activated, - .button-bar-stable.activated, - .bar-stable.activated, - .stable-border.activated { - background-color: #E0E0E0; - color: #fff; } - - /* Text Colors - ==================================*/ - .positive, .icon-help, - .positive *, .icon-help *, - *.positive, - *.icon-help, - .positive:hover, - .icon-help:hover, - .positive:hover *, .icon-help:hover *, - *.positive:hover, - *.icon-help:hover, - .positive:active, - .icon-help:active, - .positive:active *, .icon-help:active *, - *.positive:active, - *.icon-help:active { - color: #3F51B5; } - - .positive-900, - .positive-900 *, - *.positive-900, - .positive-900:hover, - .positive-900:hover *, - *.positive-900:hover, - .positive-900:active, - .positive-900:active *, - *.positive-900:active { - color: #3F51B5; } - - .positive-100, - .positive-100 *, - *.positive-100, - .positive-100:hover, - .positive-100:hover *, - *.positive-100:hover, - .positive-100:active, - .positive-100:active *, - *.positive-100:active { - color: #C5CAE9; } - - .calm-100, - .calm-100 *, - *.calm-100, - .calm-100:hover, - .calm-100:hover *, - *.calm-100:hover, - .calm-100:active, - .calm-100:active *, - *.calm-100:active { - color: #2196F3; } - - .calm-900, - .calm-900 *, - *.calm-900, - .calm-900:hover, - .calm-900:hover *, - *.calm-900:hover, - .calm-900:active, - .calm-900:active *, - *.calm-900:active { - color: #0D47A1; } - - .calm-100, - .calm-100 *, - *.calm-100, - .calm-100:hover, - .calm-100:hover *, - *.calm-100:hover, - .calm-100:active, - .calm-100:active *, - *.calm-100:active { - color: #BBDEFB; } - - .royal, - .royal *, - *.royal, - .royal:hover, - .royal:hover *, - *.royal:hover, - .royal:active, - .royal:active *, - *.royal:active { - color: #673AB7; } - - .royal-900, - .royal-900 *, - *.royal-900, - .royal-900:hover, - .royal-900:hover *, - *.royal-900:hover, - .royal-900:active, - .royal-900:active *, - *.royal-900:active { - color: #311B92; } - - .royal-100, - .royal-100 *, - *.royal-100, - .royal-100:hover, - .royal-100:hover *, - *.royal-100:hover, - .royal-100:active, - .royal-100:active *, - *.royal-100:active { - color: #D1C4E9; } - - .balanced, - .balanced *, - *.balanced, - .balanced:hover, - .balanced:hover *, - *.balanced:hover, - .balanced:active, - .balanced:active *, - *.balanced:active { - color: #4CAF50; } - - .balanced-900, - .balanced-900 *, - *.balanced-900, - .balanced-900:hover, - .balanced-900:hover *, - *.balanced-900:hover, - .balanced-900:active, - .balanced-900:active *, - *.balanced-900:active { - color: #1B5E20; } - - .balanced-100, - .balanced-100 *, - *.balanced-100, - .balanced-100:hover, - .balanced-100:hover *, - *.balanced-100:hover, - .balanced-100:active, - .balanced-100:active *, - *.balanced-100:active { - color: #C8E6C9; } - - .energized, - .energized *, - *.energized, - .energized:hover, - .energized:hover *, - *.energized:hover, - .energized:active, - .energized:active *, - *.energized:active { - color: #FF9800; } - - .energized-900, - .energized-900 *, - *.energized-900, - .energized-900:hover, - .energized-900:hover *, - *.energized-900:hover, - .energized-900:active, - .energized-900:active *, - *.energized-900:active { - color: #E65100; } - - .energized-100, - .energized-100 *, - *.energized-100, - .energized-100:hover, - .energized-100:hover *, - *.energized-100:hover, - .energized-100:active, - .energized-100:active *, - *.energized-100:active { - color: #FFE0B2; } - - .assertive, .icon-alert, - .assertive *, .icon-alert *, - *.assertive, - *.icon-alert, - .assertive:hover, - .icon-alert:hover, - .assertive:hover *, .icon-alert:hover *, - *.assertive:hover, - *.icon-alert:hover, - .assertive:active, - .icon-alert:active, - .assertive:active *, .icon-alert:active *, - *.assertive:active, - *.icon-alert:active { - color: #F44336; } - - .assertive-900, - .assertive-900 *, - *.assertive-900, - .assertive-900:hover, - .assertive-900:hover *, - *.assertive-900:hover, - .assertive-900:active, - .assertive-900:active *, - *.assertive-900:active { - color: #B71C1C; } - - .assertive-100, - .assertive-100 *, - *.assertive-100, - .assertive-100:hover, - .assertive-100:hover *, - *.assertive-100:hover, - .assertive-100:active, - .assertive-100:active *, - *.assertive-100:active { - color: #FFCDD2; } - - .stable, - .stable *, - *.stable, - .stable:hover, - .stable:hover *, - *.stable:hover, - .stable:active, - .stable:active *, - *.stable:active { - color: #E0E0E0; } - - .light, - .light *, - *.light, - .light:hover, - .light:hover *, - *.light:hover, - .light:active, - .light:active *, - *.light:active { - color: #fff; } - - .dark, - .dark *, - *.dark, - .dark:hover, - .dark:hover *, - *.dark:hover, - .dark:active, - .dark:active *, - *.dark:active { - color: #444; } - - .light-border { - border-color: #ddd; } - - .navbar-default .navbar-nav > li > a { - margin: 0; - padding-right: 26px; - padding-left: 26px; - border-top: 3px solid transparent; - color: #BFD5C9; - opacity: 1; } - - /* Mid-Bar - ==================================*/ - .mid-bar { - padding: 16px; } - - .mid-bar h1, - .mid-bar h2, - .mid-bar h3, - .mid-bar h4, - .mid-bar h5, - .mid-bar h6 { - color: #fff; - margin-bottom: 5px; } - - .mid-bar p { - color: rgba(255, 255, 255, 0.5); - margin-bottom: 0; } - - /* Item - ==================================*/ - .item-avatar, - .item-avatar .item-content, - .item-avatar-left, - .item-avatar-left .item-content, - .card > .item-avatar { - padding-left: 95px; } - - .item, - .item-complex .item-content, - .item-radio .item-content { - background-color: transparent; } - - .dark-bg h2, - .item.dark-bg h2 { - color: #fff; } - - .tabs-striped .tabs { - box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); } - - .bar .button.button-clear, .bar .button.button-text { - color: #fff; } - - .bar .button.button-icon .icon:before, .bar .button.button-icon .icon-help:before, .bar .button.button-icon .icon-alert:before, .bar .button.button-icon #menu .footer .icon-help:before, #menu .footer .bar .button.button-icon .icon-help:before, - .bar .button.button-icon.icon-left:before, - .bar .button.button-icon.icon-right:before, - .bar .button.button-icon:before { - vertical-align: top; - font-size: 24px; } - - .button-icon.button.active, - .button-icon.button.activated { - opacity: initial; } - - /* Button - ==================================*/ - .button { - overflow: hidden !important; } - - @font-face { - font-family: "Cesiumicons"; - src: url("../fonts/cesiumicons.eot?v=1.2"); - src: url("../fonts/cesiumicons.eot?v=1.2#iefix") format("embedded-opentype"), url("../fonts/cesiumicons.ttf?v=1.2") format("truetype"), url("../fonts/cesiumicons.woff?v=1.2") format("woff"), url("../fonts/cesiumicons.woff") format("woff"), url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBzdGFuZGFsb25lPSJubyI/Pgo8IURPQ1RZUEUgc3ZnIFBVQkxJQyAiLS8vVzNDLy9EVEQgU1ZHIDEuMS8vRU4iICJodHRwOi8vd3d3LnczLm9yZy9HcmFwaGljcy9TVkcvMS4xL0RURC9zdmcxMS5kdGQiID4KPHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciPgo8bWV0YWRhdGE+R2VuZXJhdGVkIGJ5IEljb01vb248L21ldGFkYXRhPgo8ZGVmcz4KPGZvbnQgaWQ9ImNlc2l1bWljb25zIiBob3Jpei1hZHYteD0iMTAyNCI+Cjxmb250LWZhY2UgdW5pdHMtcGVyLWVtPSIxMDI0IiBhc2NlbnQ9Ijk2MCIgZGVzY2VudD0iLTY0IiAvPgo8bWlzc2luZy1nbHlwaCBob3Jpei1hZHYteD0iMTAyNCIgLz4KPGdseXBoIHVuaWNvZGU9IiYjeDIwOyIgaG9yaXotYWR2LXg9IjUxMiIgZD0iIiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkwMDsiIGQ9Ik03MDQuNjMxIDU3MS4xMjhjLTg4LjYxMy0xNDEuNDA5LTIyMC40MDMtMjYzLjQwOC0zODUuNTIxLTMwNS40NzMtNjcuMjUyLTE4LjQ5My0xMzcuMTEyLTI1LjgyMi0yMDYuNzA1LTI3LjAyMiA4NS41NDktMTYzLjA5NSAyNzUuODEtMjYyLjI3MiA0NTguNTQ1LTIzNy45NSAxODEuMTY2IDE5LjA3NiAzNDEuNjYyIDE1Ny4yMTUgMzg4LjMzOCAzMzMuMTYyLTU0LjYwMSAxMDMuOTI0LTE0MS40OSAxOTcuODc0LTI1NC42NTYgMjM3LjI4M3oiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTAxOyIgZD0iTTcwNS45MzcgNTcyLjA4OGMtMTYuNzU2LTExMS44MTUtODUuMDg2LTIwNy4yLTEwNy4yNjctMzE3LjIyMS05LjE4OC01MS41NDEgMjYuNzg2LTExMC42NDItMTQuNTc5LTE1NS41MTQtNDYuMjEtNTIuNDAyLTExOC45MDctNjUuMzc5LTE4MS45MzgtODcuNDY2IDE3Ny43My00Ny45MSAzNzkuNjYgMjQuMjE1IDQ4NS45ODIgMTc0Ljc4IDUwLjU1IDY5LjExNiA4MC45OTIgMTUyLjY4OSA4Ni42OTYgMjM4LjEyNS00Ni4wODUgOS40ODMtNjkuNDM2IDU1LjAyMC0xMTAuNTU2IDc0LjIwMS00OS45MTcgMjkuOTkyLTEwMy41NzEgNTMuNTE0LTE1OC4zMzkgNzMuMDk1eiIgLz4KPGdseXBoIHVuaWNvZGU9IiYjeGU5MDI7IiBkPSJNNzA0LjQyMiA1NjMuNGMtMjMuNzg2LTUxLjQwOC0yNi40MzYtMTA5Ljc1Ny01NC4wODQtMTU5LjcyNi03LjIxMS0zMi45MDktNDEuMDkxLTg0LjI4OC0yOS40OTgtMTA2LjQ4OSAxMDEuNTktMTYuNTQyIDIwMy4xODEtMzMuMDg1IDMwNC43NzEtNDkuNjI3IDE4LjI3OCAzNS42NDkgMzEuODQ1IDczLjcxMSA0MC4yMDEgMTEyLjg5NC03My44NjggNzkuMjY5LTE1MC40NzEgMTYzLjQyNi0yNTUuMzU4IDIwMC42MzVsLTYuMDMyIDIuMzE0eiIgLz4KPGdseXBoIHVuaWNvZGU9IiYjeGU5MDM7IiBkPSJNNjY4LjYxNiA0NDUuNzk0Yy0yMC45ODMtMzEuNjc5LTI3LjI0NS03MC40MDgtNDQuMTA0LTEwNC40MjYtMTYuMjIxLTM1LjY4OS01MS40MTUtMTE5LjQ3MiAxOS42ODQtMTA5Ljg5IDUwLjQ1OSAyNC42NTQgOTUuMDUwLTEzLjM3OSAxNDUuODE5LTE2LjkyOSA0MC4zMTctNS41NjEgODIuMTM3LTQuNjY1IDEyMS4yOTUgNy4yMDEgMTQuOTgzIDMxLjcxNiA1MC42ODYgODIuNTk0IDM4LjcyNiAxMDguMzA1LTQ4LjU2MyA5Ljg2Ny05OS40NzMgMTMuODc5LTE0NS4yMTUgMzMuNjkgMTkuMjczIDQ1LjU5NS01NC45NzQgNTUuNDY1LTg5Ljk3NiA2OC44NC0xNS4zNDEgNC42MzctMzAuNzY1IDguOTk4LTQ2LjIyOSAxMy4yMDh6IiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkwNDsiIGQ9Ik02MDcuNzE0IDE2MC44NzNjLTMwLjE4My04NS4wMDItMTIzLjE4Ni0xMTkuOTQ4LTIwMS44MzItMTQ1Ljk0MyA1LjAyMC0xMS4xODYgNzEuMTg2LTE2LjMyNSAxMDAuNzg3LTE3LjMyNiAxMTcuMTE1LTMuMDY3IDIzNC41NDIgNDEuMDg2IDMyMC40OCAxMjAuNzUyLTc0LjgzMSAyLjgzNC0xNDkuNTQ5IDE0Ljc4NC0yMTkuNDM1IDQyLjUxN3oiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTA1OyIgZD0iTTU5OC40ODUgMjY4LjUzNGMtNC4xNzQtNDIuNzY0LTI1LjE4OC0xMTUuOTM2IDM0LjAyOC0xMjguNDk0IDU4LjUzNy0xNC41ODEgMTE5LjQzNC0xOC41MTEgMTc3LjA2Ny0zNy4wMjAgNDUuMjAxIDM3LjQ0NyA4My4xNjcgODMuNTk5IDExMS4xMiAxMzUuMjE3LTEwNy45OC0xLjY4MS0yMTkuNDMxLTEwLjIwNC0zMjIuMjE1IDMwLjI5N3oiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTA2OyIgZD0iTTYwNy4yMSA2MDAuNzhjMTguOTU5IDIuMjkgMzUuMjM0IDEuODk2IDI4LjI4OC0yMS43MiA1LjkzNS0xNC42NDQtMzYuNjE0LTE0LjYwNy0yOC4yODggMS44NDEgMCA2LjYyNiAwIDEzLjI1MiAwIDE5Ljg3OXoiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTA3OyIgZD0iTTQ0NC41NDEgNzAyLjk2MmMyMS4wODggMy45NTggMzIuNjU0LTEuNjA0IDI2Ljk1NC0yNC40MzUtNS42NzMtMTEuMDcxLTM3LjY5Ny03LjY1LTI2Ljk1NCAxNC4wMzYgMCAzLjQ2NiAwIDYuOTMzIDAgMTAuMzk5eiIgLz4KPGdseXBoIHVuaWNvZGU9IiYjeGU5MDg7IiBkPSJNNTA0LjUzMiA2ODEuNzhjMTMuNDc4LTMuMjk4IDQzLjIzNiA5LjY4MiAzNi43NTUtMTMuNTc1IDYuMjU0LTIzLjAzMC03LjA1Ny0yNi40OTYtMjcuMTUtMjMuMTgtMTguNTY3LTUuNDk2LTUuODcxIDI2LjUyNy05LjYwNSAzNi43NTV6IiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkwOTsiIGQ9Ik01MDIuNTM5IDQxNy40MjZjMTguNzg2IDAgMzcuNTcyIDAgNTYuMzU4IDAgMC0xOC43ODYgMC0zNy41NzIgMC01Ni4zNTgtMTguNzg2IDAtMzcuNTcyIDAtNTYuMzU4IDAgMCAxOC43ODYgMCAzNy41NzIgMCA1Ni4zNTh6IiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkwYTsiIGQ9Ik01MDguODU2IDYzMC4xNTdjMTMuNDc4LTMuMjk4IDQzLjIzNiA5LjY4MiAzNi43NTUtMTMuNTc1IDYuMjU0LTIzLjAzMC03LjA1Ny0yNi40OTYtMjcuMTUtMjMuMTgtMTguNTY3LTUuNDk2LTUuODcxIDI2LjUyNy05LjYwNSAzNi43NTV6IiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkwYjsiIGQ9Ik01NjkuMTAzIDQ3NC4xNGMxOC45NyAxLjkyOSAzNi40OTIgMi40MDkgMjkuNDA0LTIxLjcyIDYuODk0LTE1LjkzMi0zNy4yMzUtMTUuODMzLTI5LjQwNCAwLjcyNSAwIDYuOTk4IDAgMTMuOTk2IDAgMjAuOTk0eiIgLz4KPGdseXBoIHVuaWNvZGU9IiYjeGU5MGM7IiBkPSJNNjI3LjA4NyA1MzEuNDAzYzE5LjM1IDcuODY5IDI5LjQwOC05LjU4MiAxOS4zMzgtMjIuMDUzLTIxLjU4My00LjE0Ny0yMC44MDEgMy4zODctMTkuMzM4IDIyLjA1M3oiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTBkOyIgZD0iTTU0Ny42MjcgNTI1LjIzMWMxMy40NzgtMy4yOTggNDMuMjM2IDkuNjgyIDM2Ljc1NS0xMy41NzUgNi4yNTQtMjMuMDMwLTcuMDU3LTI2LjQ5Ni0yNy4xNS0yMy4xOC0xOC41NjctNS40OTYtNS44NzEgMjYuNTI3LTkuNjA1IDM2Ljc1NXoiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTBlOyIgZD0iTTQwMy44MDIgMzM1LjAyNGMxNC40MjYtMS45OTggMzYuNzMgMy44ODIgNDYuNTU2LTIuNzE1LTIuMDIwLTE0LjMxNSAzLjkxMy0zNi41ODQtMi43MTUtNDYuMjkyLTEyLjg3OSAzLjU0NS00MC4wMTctNy4xNjktNDMuODQxIDUuNDMgMCAxNC41MjYgMCAyOS4wNTEgMCA0My41Nzd6IiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkwZjsiIGQ9Ik00NDcuOTA3IDQwNi4wODRjMTYuODM2LTAuMTE3IDM5LjY3NyA1Ljc3IDMxLjg1NC0xOS4wMDUgNy44Mi0yNC4zNTUtMTYuMDgwLTE2Ljk5LTMxLjg1NC0xNy43NSAwIDEyLjI1MiAwIDI0LjUwMyAwIDM2Ljc1NXoiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTEwOyIgZD0iTTUxNS4yOTEgNDY4LjU2N2MtMTguOTcxIDEuODk0LTM2LjYxMiAyLjQ1OS0yOS41MS0yMS43Mi0yLjUzOC0xMi45OSAzOS42MjctMTMuMzc0IDI5LjUxIDUuODkxIDAgNS4yNzYgMCAxMC41NTMgMCAxNS44Mjl6IiAvPgo8Z2x5cGggdW5pY29kZT0iJiN4ZTkxMTsiIGQ9Ik01NzguNDg2IDU3Mi44NjZjMjEuMjMyIDQuOTIgMjkuNDY5LTMuMTU2IDI0LjUwMy0yNC40MzUtMTMuMTA0LTguNTMyLTMyLjk1NS0yLjEwNi0yNC41MDMgMTkuMjAxdjUuMjM0eiIgLz4KPGdseXBoIHVuaWNvZGU9IiYjeGU5MTI7IiBkPSJNNjAxLjQ3MSA2NDEuNjg5Yy0xOC45NzEgMS44OTQtMzYuNjEyIDIuNDU5LTI5LjUxLTIxLjcyLTIuNTM4LTEyLjk5IDM5LjYyNy0xMy4zNzQgMjkuNTEgNS44OTEgMCA1LjI3NiAwIDEwLjU1MyAwIDE1LjgyOXoiIC8+CjxnbHlwaCB1bmljb2RlPSImI3hlOTE1OyIgZ2x5cGgtbmFtZT0iZGlhc3BvcmEiIGhvcml6LWFkdi14PSIxMDY0IiBkPSJNNjcwLjEgMjIuMTA5Yy0yOC42OTIgNDAuMDQ2LTczLjQ1OSAxMDIuNTcxLTk5LjQ4MiAxMzguOTQ1LTI2LjU3OSAzNy4xNTEtNDguMjAyIDY2LjE0Ni00OS4zNCA2Ni4xNjEtMS4xNjkgMC4wMTUtNDIuOTg2LTU2LjE5Ni05OC45MTItMTMyLjk2MS01My4yODgtNzMuMTQzLTk3LjMzLTEzMi45ODctOTcuODcyLTEzMi45ODctMS41MTYgMC0xOTIuMTE0IDEzNC4yNS0xOTIuNjA1IDEzNS42NjMtMC4yMzcgMC42ODQgNDIuODUgNjMuOTM1IDk1Ljc1MSAxNDAuNTU5czk2LjE4MiAxNDAuNTIzIDk2LjE4MiAxNDEuOTk5YzAgMi4zOC0xNy4xNDIgOC4zOTgtMTUxLjYwNSA1My4yMjYtODMuMzgzIDI3Ljc5OC0xNTIuNDIyIDUwLjg2MS0xNTMuNDIgNTEuMjUxLTEuMzg4IDAuNTQyIDYuNTUgMjcuMTAzIDMzLjc4MyAxMTMuMDMyIDE5LjU3OCA2MS43NzggMzYuMDQwIDExMi44MjggMzYuNTgxIDExMy40NDRzNzMuMDExLTIyLjU3NCAxNjEuMDQzLTUxLjUzNGM4OC4wMzMtMjguOTYgMTYwLjc5Ni01Mi42NTQgMTYxLjY5Ni01Mi42NTNzMS45MTkgMS4zNDggMi4yNjMgMi45OTRjMC4zNDQgMS42NDYgMS4xMDQgNzYuNjAxIDEuNjg4IDE2Ni41NjZzMS41NDYgMTY0LjMyMiAyLjEzNyAxNjUuMjM2YzAuODQ3IDEuMzA5IDI1LjU3NiAxLjY1OCAxMTYuMzY0IDEuNjQyIDYzLjQwOC0wLjAxMSAxMTUuOTQ4LTAuNDYgMTE2Ljc1Ni0wLjk5NyAxLjAxOC0wLjY3NyAyLjc1Ny01MC4xMzggNS42Ny0xNjEuMjI2IDQuNzgzLTE4Mi40MDggNC44ODYtMTg0Ljg1MiA3Ljc5NS0xODQuODUyIDEuMTI4IDAgNzAuODk2IDIzLjM1MyAxNTUuMDQxIDUxLjg5NnMxNTMuNCA1MS40MzMgMTUzLjkwMSA1MC44NjhjMS43LTEuOTE4IDcwLjcxNS0yMjYuNjAyIDY5Ljg2NS0yMjcuNDUxLTAuNDY0LTAuNDY0LTcwLjY5LTI0LjUxNC0xNTYuMDU2LTUzLjQ0NC0xMTcuNDMyLTM5Ljc5Ny0xNTUuMzE1LTUzLjE0My0xNTUuNjMzLTU0LjgyNy0wLjIzNS0xLjI0NCA0MC4xNDktNjEuNTk0IDkxLjQ5OS0xMzYuNzM3IDUwLjU1NS03My45ODEgOTEuNzMzLTEzNS4wNjIgOTEuNTA3LTEzNS43MzctMC41MjEtMS41NDktMTg5LjY4MS0xNDAuODg0LTE5MS4yNjMtMTQwLjg4NC0wLjY0MiAwLTI0LjY0MyAzMi43NjUtNTMuMzM1IDcyLjgxeiIgLz4KPC9mb250PjwvZGVmcz48L3N2Zz4=) format("svg"); - font-weight: normal; - font-style: normal; } - - .cion, .cesiumicons, - .ion-social-duniter:before, - .ion-social-diaspora:before, - .ion-office:before, - .ion-library:before { - display: inline-block; - font-family: "Cesiumicons"; - speak: none; - font-style: normal; - font-weight: normal; - font-variant: normal; - text-transform: none; - text-rendering: auto; - line-height: 1; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; } - - .ion-social-duniter:before { - content: ""; } - - .ion-social-diaspora:before { - content: ""; } - - .ion-office:before { - content: ""; } - - .ion-library:before { - content: ""; } - - @media screen and (max-width: 400px) { - @-ms-viewport { - width: 320px; } - .item .badge { - right: 16px; } } - - @media screen and (max-width: 767px) { - .hidden-xs { - display: none !important; - visibility: hidden !important; } - .badge { - text-overflow: ellipsis !important; - white-space: nowrap; - overflow: hidden !important; - max-width: 300px !important; - display: block !important; } - .badge:empty { - display: none !important; } - .item .badge { - right: 16px; } - .padding-top-xs { - padding-top: 10px; } } - - @media screen and (min-width: 768px) { - .hidden-xs { - display: inherit; - visibility: visible; } - .row.hidden-xs { - display: flex !important; } - .button.hidden-xs { - display: inline-block; } - .item-toggle .toggle { - right: 32px; } } - - @media screen and (max-width: 767px) { - .visible-xs { - display: inherit !important; - visibility: visible !important; } } - - @media screen and (min-width: 768px) { - .visible-xs { - display: none !important; - visibility: hidden !important; } } - - @media screen and (max-width: 767px) { - .padding-xs { - padding: 16px !important; } } - - @media screen and (min-width: 768px) { - .padding-xs { - padding: inherit; } } - - @media screen and (max-width: 767px) { - .no-padding-xs { - padding: 0px !important; } } - - @media screen and (min-width: 768px) { - .no-padding-xs { - padding: inherit; } } - - @media screen and (max-width: 767px) { - .no-margin-xs { - margin: 0px !important; } } - - @media screen and (min-width: 768px) { - .no-margin-xs { - margin: inherit; } } - - @media screen and (max-width: 991px) and (min-width: 768px) { - .hidden-sm, .row-header.hidden-sm { - display: none !important; - visibility: hidden !important; } - .badge { - text-overflow: ellipsis !important; - white-space: nowrap; - overflow: hidden !important; - max-width: 400px !important; - display: block !important; } - .badge:empty { - display: none !important; } } - - @media screen and (min-width: 992px) { - .hidden-sm { - display: inherit; - visibility: visible; } - .row.hidden-sm { - display: flex !important; } - .button.hidden-sm { - display: inline-block; } } - - @media screen and (max-width: 767px) { - .hidden-sm { - display: inherit; - visibility: visible; } } - - @media screen and (max-width: 991px) { - .visible-sm { - display: inherit !important; - visibility: visible !important; } } - - @media screen and (min-width: 992px) { - .visible-sm { - display: none; - visibility: hidden; } } - - @media screen and (max-width: 767px) { - .visible-sm { - display: none; - visibility: hidden; } } - - @media screen and (max-width: 991px) { - body { - cursor: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAN1wAADdcBQiibeAAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAAEbSURBVDiNndMxK4ZRGMbx3zmRMrwvM2XCQFFik/IJpCw+hJLPgfIhLBY+gEEGJQPFwGZg9TIoBrfhOfSQHl7XeM7/fw3nvk+KCPWklEawgGlMlOMrnOMoIm6/8B8FKaWMFazhGQ94LFwbg+jHDvYi4u2zIKWUsIk5XNfE72ljHKfYiIjI5WIZMzhrkJW7s8IuQ8IwdnGDpwa5nhbGsJqxiNcuZIV9xWJWvXY3cr1kOmMSnX8UdDCZEb+RDYmMS9WMu80gLrNqw1r/KGjhPOMQvaol+Wva6MFhjoh7bGEUuVGrkgu7HRH3H8IBTjCLgQZ5oDAnxfnymRKWsI4X1Zw7NbGFPmxjP4qYfvjOQ5jHlGpHqCZ1geOIuKvz76QSW1T3cwmnAAAAAElFTkSuQmCC), auto; } } - - @media screen and (min-width: 992px) { - body { - cursor: inherit; } } - - @media screen and (min-width: 992px) and (max-width: 1199px) { - .hidden-md { - display: none !important; - visibility: hidden !important; } - .badge { - text-overflow: ellipsis !important; - white-space: nowrap; - overflow: hidden !important; - max-width: 400px !important; - display: block !important; } - .badge:empty { - display: none !important; } - /* - see issue # - html{ - -webkit-user-selectuser-select: all !important; - -moz-user-select: all !important; - -ms-user-select: all !important; - user-select: all !important; - }*/ } - - @media screen and (min-width: 1200px) { - .hidden-md { - display: inherit; - visibility: visible; } } - - @media screen and (max-width: 991px) { - .hidden-md { - display: inherit; - visibility: visible; } } - - @media screen and (min-width: 992px) and (max-width: 1199px) { - .visible-md { - display: inherit !important; - visibility: visible !important; } } - - @media screen and (min-width: 1200px) { - .visible-md { - display: none; - visibility: hidden; } } - - @media screen and (max-width: 991px) { - .visible-md { - display: none; - visibility: hidden; } } - - @media screen and (min-width: 1200px) { - .hidden-lg { - display: none !important; - visibility: hidden !important; } - .visible-lg { - display: inherit !important; - visibility: visible !important; } - .badge { - text-overflow: ellipsis !important; - white-space: nowrap; - overflow: hidden !important; - max-width: 450px !important; - display: block !important; } - .badge:empty { - display: none !important; } - /*html{ - -webkit-user-select: all !important; - -moz-user-select: all !important; - -ms-user-select: all !important; - user-select: all !important; - }*/ } - - @media screen and (max-width: 1199px) { - .hidden-lg { - display: inherit; - visibility: visible; } } - - @media screen and (max-width: 1199px) { - .visible-lg { - display: none; - visibility: hidden; } } - - @media screen and (max-width: 768px) { - .no-padding-xs { - padding: inherit; } } - - @media screen and (max-width: 767px) { - .no-margin-xs { - margin: 0px !important; } } - - /********** - Notifications view - **********/ - .view-notification .item.unread { - background-color: #ecf0f7 !important; - border-color: #dddfe2 !important; } - - .view-notification ion-item h4 i.icon, .view-notification ion-item h4 i.icon-help, .view-notification ion-item h4 i.icon-alert, .view-notification ion-item h4 #menu .footer i.icon-help, #menu .footer .view-notification ion-item h4 i.icon-help { - font-size: 18px !important; - line-height: 12px !important; - vertical-align: middle !important; } - - /* ============ - Buttons - =============== */ - .bar.bar-header .button.button-clear.button-icon.ion-android-more-vertical, .bar.bar-header .button.button-icon.ion-android-more-vertical.button-text, - .bar.bar-header .button.button-clear.button-icon i.ion-android-more-vertical, - .bar.bar-header .button.button-icon.button-text i.ion-android-more-vertical { - padding-left: 8px; } - - .bar.bar-header .buttons .secondary-buttons > - .button.button-clear.button-icon.ion-android-more-vertical:first-child, .bar.bar-header .buttons .secondary-buttons > - .button.button-icon.ion-android-more-vertical.button-text:first-child { - padding-left: 0px !important; } - - .bar .buttons.pull-right, .bar .popover-helptip .buttons.icon.icon-right, .popover-helptip .bar .buttons.icon.icon-right, .bar .popover-helptip .buttons.icon-right.icon-help, .popover-helptip .bar .buttons.icon-right.icon-help, .bar .popover-helptip .buttons.icon-right.icon-alert, .popover-helptip .bar .buttons.icon-right.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-right.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-right.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-right.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-right.icon-help, .bar .popover-helptip .buttons.icon.icon-center, .popover-helptip .bar .buttons.icon.icon-center, .bar .popover-helptip .buttons.icon-center.icon-help, .popover-helptip .bar .buttons.icon-center.icon-help, .bar .popover-helptip .buttons.icon-center.icon-alert, .popover-helptip .bar .buttons.icon-center.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-center.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-center.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-center.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-center.icon-help, .bar .popover-helptip .buttons.icon.icon-bottom-right, .popover-helptip .bar .buttons.icon.icon-bottom-right, .bar .popover-helptip .buttons.icon-bottom-right.icon-help, .popover-helptip .bar .buttons.icon-bottom-right.icon-help, .bar .popover-helptip .buttons.icon-bottom-right.icon-alert, .popover-helptip .bar .buttons.icon-bottom-right.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-bottom-right.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-bottom-right.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-bottom-right.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-bottom-right.icon-help, .bar .popover-helptip .buttons.icon.icon-bottom-center, .popover-helptip .bar .buttons.icon.icon-bottom-center, .bar .popover-helptip .buttons.icon-bottom-center.icon-help, .popover-helptip .bar .buttons.icon-bottom-center.icon-help, .bar .popover-helptip .buttons.icon-bottom-center.icon-alert, .popover-helptip .bar .buttons.icon-bottom-center.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-bottom-center.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-bottom-center.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-bottom-center.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-bottom-center.icon-help, .bar .title + .button:last-child, .bar .title + .buttons, .bar > .button + .button:last-child, .bar > .button.pull-right, .popover-helptip .bar > .button.icon.icon-right, .popover-helptip .bar > .button.icon-right.icon-help, .popover-helptip .bar > .button.icon-right.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-right.icon-help, #menu .footer .popover-helptip .bar > .button.icon-right.icon-help, .popover-helptip .bar > .button.icon.icon-center, .popover-helptip .bar > .button.icon-center.icon-help, .popover-helptip .bar > .button.icon-center.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-center.icon-help, #menu .footer .popover-helptip .bar > .button.icon-center.icon-help, .popover-helptip .bar > .button.icon.icon-bottom-right, .popover-helptip .bar > .button.icon-bottom-right.icon-help, .popover-helptip .bar > .button.icon-bottom-right.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-bottom-right.icon-help, #menu .footer .popover-helptip .bar > .button.icon-bottom-right.icon-help, .popover-helptip .bar > .button.icon.icon-bottom-center, .popover-helptip .bar > .button.icon-bottom-center.icon-help, .popover-helptip .bar > .button.icon-bottom-center.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-bottom-center.icon-help, #menu .footer .popover-helptip .bar > .button.icon-bottom-center.icon-help { - top: 0px !important; } - - .bar.bar-header { - padding-right: 5px !important; } - .bar.bar-header .buttons-right span { - margin-left: 0px !important; } - - .bar .title + .buttons.buttons-right { - right: 5px; } - - .button-icon { - border-color: transparent; - box-shadow: none !important; } - - .button-small-padding { - padding: 0 7px !important; } - - .button-text { - color: grey !important; - font-size: 12px; } - - .button-text.button-small { - padding: 5px 2px; - font-size: 12px !important; } - - .button-text-positive { - color: #387ef5 !important; } - - .button-text-stable { - color: #b2b2b2 !important; } - - - .gray, .popover-share .bar-header span, .popover-share .bar-footer .button-close, .popover-helptip .button-close { - color: grey !important; } - .gray b, .popover-share .bar-header span b, .popover-share .bar-footer .button-close b, .popover-helptip .button-close b { - color: grey !important; } - - .gray a, .popover-share .bar-header span a, .popover-share .bar-footer .button-close a, .popover-helptip .button-close a, .positive a, .icon-help a { - color: inherit; } - - .gray a:hover, .popover-share .bar-header span a:hover, .popover-share .bar-footer .button-close a:hover, .popover-helptip .button-close a:hover, .positive a:hover, .icon-help a:hover { - color: inherit; } - - .gray a:visited, .popover-share .bar-header span a:visited, .popover-share .bar-footer .button-close a:visited, .popover-helptip .button-close a:visited, .positive a:visited, .icon-help a:visited { - color: inherit; } - - .item a { - text-decoration: none; } - - .no-padding { - padding: 0px !important; } - - .avatar-member { - background-image: url(data:image/png;base64,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); } - - .avatar.disable { - opacity: 0.7; } - - .avatar-wallet { - background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAIAAAACACAQAAABpN6lAAAAAAmJLR0QA/4ePzL8AAAAJcEhZcwAADdcAAA3XAUIom3gAAAAHdElNRQfgBA0JMBOJU4OdAAAB2klEQVR42u3au0ocUQCH8e+IxFRKUohrkG0Eu4BiLrCFDxDBNtjoI1gYTJnGQrRInc4HyBuopNHgBVZIwCJFsHBXLURtQiIyKYyFN3B3zJyd3e/X7e7MmXP+e26zOyBJkiRJkiRJkiRJkiRJkppVSF9E0sk0r+jKuOYnrLMQTqMHkHSzQTHS17fLy3CYroi21JWYj9Z8KDIfvwdUKAC7bGfc+CH6gGrojTyJJBemMr/u+4sLpy2njRZnAAZgAK2t/YHKGUzeZlzz5w2yFU6/EKVsQHAIGIABGIABGIABRN4ILbOVcc1fM9IQCfp7gHOAARiAAbgMXpuhH/GJ0brK/sG78OVfKR0cXPnsQ/iYkwAYY6LOM4eZpXR5q3/t77bH+RkCO5HObZQeEL4nQ7ypcwh8boIAIJQpuwoYQMsugwBJ/73L/h1+3vr+OYtXXn/Lzz7gCV8ZqOH4LUrhz4155IzJvA6B0VqaDwzHu7f/PwEsc1zT8RXWmmoOCJWkl9L95wBWbw6AvO8DfrHkMmgABmAABtDw0j8gUaWHeA9K7odC7GVwhXGgGOmB2ZX4PaDAJs8i9d89XoRq5AAgecoMg3Rm3PhTysyFIyRJkiRJkiRJkiRJkiRJku7wFyuiadmIs5Q9AAAAAElFTkSuQmCC); } - - .item.item-icon-left > i.avatar:first-child { - position: absolute; - display: flex; - height: 100%; - align-items: center; - font-size: 16px; - left: 16px; - top: 8px; - max-height: 32px; - max-width: 32px; } - - .item.item-checkbox.item-avatar > i.avatar:first-child, - .item.item-checkbox.item-avatar > i.item-image:first-child, - .item.item-checkbox.item-avatar .item-content > i.avatar:first-child, - .item.item-checkbox.item-avatar .item-content > i.item-image:first-child, - .item.item-checkbox.item-avatar * > i.avatar:first-child, - .item.item-checkbox.item-avatar * > i.item-image:first-child, - .item.item-checkbox.item-avatar * .item-content > i.avatar:first-child, - .item.item-checkbox.item-avatar * .item-content > i.item-image:first-child { - left: 65px; } - - .item.item-checkbox.item-avatar .item-content, - .item.item-checkbox .item-content .item-avatar { - padding-left: 65px; } - - .card .card-header, - .card .card-header { - font-size: 90%; - opacity: 0.8; - filter: alpha(opacity=80); } - - .card.stable-900-bg, - .card .stable-900-bg, - .item.stable-900-bg, - .item .stable-900-bg, - .item-complex .item-content .stable-900-bg, - .item-radio .item-content .stable-900-bg { - background-color: #e0e0e0 !important; } - - .card .item { - background: inherit; } - - .card.stable-bg, - .card .stable-bg, - .item.stable-bg, - .item .stable-bg, - .item-complex .item-content .stable-bg, - .item-radio .item-content .stable-bg { - background-color: #f8f8f8 !important; } - - .card .card-header { - padding-top: 5px !important; - padding-bottom: 0px !important; - min-height: 25px; } - - .card .item .card-footer { - margin-bottom: 5px; } - - .card .card-avatar .avatar, - .card.card-avatar .avatar { - box-shadow: 0px 3px 4px 0 rgba(0, 0, 0, 0.26); - top: 7px; - background-color: #D9D9D9; } - - .card .card-avatar img.avatar, - .card.card-avatar img.avatar { - border: 0; - min-height: 54px; - min-width: 54px; } - - .card .card-avatar .item.item-avatar, - .card.card-avatar .item.item-avatar { - padding-top: 10px; - padding-bottom: 2px; - min-height: 45px !important; } - - .card .card-avatar .card-footer, - .card.card-avatar .card-footer { - padding-left: 88px; } - .card .card-avatar .card-footer .pull-right a, .card .card-avatar .card-footer .popover-helptip .icon.icon-right a, .popover-helptip .card .card-avatar .card-footer .icon.icon-right a, .card .card-avatar .card-footer .popover-helptip .icon-right.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-right.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-right.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-right.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-right.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-right.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon.icon-center a, .popover-helptip .card .card-avatar .card-footer .icon.icon-center a, .card .card-avatar .card-footer .popover-helptip .icon-center.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-center.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-center.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-center.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-center.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-center.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-center.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-center.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon.icon-bottom-right a, .popover-helptip .card .card-avatar .card-footer .icon.icon-bottom-right a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-right.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-right.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-bottom-right.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-right.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-bottom-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon.icon-bottom-center a, .popover-helptip .card .card-avatar .card-footer .icon.icon-bottom-center a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-center.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-center.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-center.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-bottom-center.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-center.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-bottom-center.icon-help a, - .card.card-avatar .card-footer .pull-right a, - .card.card-avatar .card-footer .popover-helptip .icon.icon-right a, .popover-helptip - .card.card-avatar .card-footer .icon.icon-right a, - .card.card-avatar .card-footer .popover-helptip .icon-right.icon-help a, .popover-helptip - .card.card-avatar .card-footer .icon-right.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon-right.icon-alert a, .popover-helptip - .card.card-avatar .card-footer .icon-right.icon-alert a, - .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-right.icon-help a, .popover-helptip #menu .footer - .card.card-avatar .card-footer .icon-right.icon-help a, - .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-right.icon-help a, #menu .footer .popover-helptip - .card.card-avatar .card-footer .icon-right.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon.icon-center a, .popover-helptip - .card.card-avatar .card-footer .icon.icon-center a, - .card.card-avatar .card-footer .popover-helptip .icon-center.icon-help a, .popover-helptip - .card.card-avatar .card-footer .icon-center.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon-center.icon-alert a, .popover-helptip - .card.card-avatar .card-footer .icon-center.icon-alert a, - .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-center.icon-help a, .popover-helptip #menu .footer - .card.card-avatar .card-footer .icon-center.icon-help a, - .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-center.icon-help a, #menu .footer .popover-helptip - .card.card-avatar .card-footer .icon-center.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon.icon-bottom-right a, .popover-helptip - .card.card-avatar .card-footer .icon.icon-bottom-right a, - .card.card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-help a, .popover-helptip - .card.card-avatar .card-footer .icon-bottom-right.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-alert a, .popover-helptip - .card.card-avatar .card-footer .icon-bottom-right.icon-alert a, - .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-right.icon-help a, .popover-helptip #menu .footer - .card.card-avatar .card-footer .icon-bottom-right.icon-help a, - .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-right.icon-help a, #menu .footer .popover-helptip - .card.card-avatar .card-footer .icon-bottom-right.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon.icon-bottom-center a, .popover-helptip - .card.card-avatar .card-footer .icon.icon-bottom-center a, - .card.card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-help a, .popover-helptip - .card.card-avatar .card-footer .icon-bottom-center.icon-help a, - .card.card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-alert a, .popover-helptip - .card.card-avatar .card-footer .icon-bottom-center.icon-alert a, - .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-center.icon-help a, .popover-helptip #menu .footer - .card.card-avatar .card-footer .icon-bottom-center.icon-help a, - .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-center.icon-help a, #menu .footer .popover-helptip - .card.card-avatar .card-footer .icon-bottom-center.icon-help a { - margin-right: 8px; } - - a.underline:focus, - .underline a:focus, - .a.underline:active, - .underline a:active, - a.underline:hover, - .underline a:hover { - outline: 1px !important; - text-decoration: underline !important; } - - .card-avatar-small.card, - .card-avatar-small .card, - .card-avatar-small .card.card-avatar, - .card-avatar-small .card .card-avatar { - min-height: 45px; } - .card-avatar-small.card .avatar, - .card-avatar-small.card .item-avatar .avatar, - .card-avatar-small .card .avatar, - .card-avatar-small .card .item-avatar .avatar, - .card-avatar-small .card.card-avatar .avatar, - .card-avatar-small .card.card-avatar .item-avatar .avatar, - .card-avatar-small .card .card-avatar .avatar, - .card-avatar-small .card .card-avatar .item-avatar .avatar { - box-shadow: 0px 2px 2px 0 rgba(0, 0, 0, 0.26); - height: 30px !important; - width: 30px !important; - left: 5px !important; } - .card-avatar-small.card .item.item-avatar, - .card-avatar-small .card .item.item-avatar, - .card-avatar-small .card.card-avatar .item.item-avatar, - .card-avatar-small .card .card-avatar .item.item-avatar { - min-height: 25px !important; - padding-left: 42px !important; } - .card-avatar-small.card .card-footer, - .card-avatar-small .card .card-footer, - .card-avatar-small .card.card-avatar .card-footer, - .card-avatar-small .card .card-avatar .card-footer { - padding-top: 0px; - padding-left: 42px !important; } - - .list .item.text-left { - text-align: left !important; } - - .list .item.text-center, .list .item.large-button-bar { - text-align: center !important; } - - .list .item.text-right { - text-align: right !important; } - - .list .item-divider.item-divider-top-border { - border-top: solid 1px rgba(0, 0, 0, 0.12); } - - /********** - Item > Avatar - **********/ - .item-avatar { - min-height: 80px !important; } - - .item-avatar > i:first-child, - .item-avatar > img:first-child, - .item-avatar i.item-image:first-child, - .item-avatar img.item-image:first-child, - .item-avatar .item-content > i:first-child, - .item-avatar .item-content > img:first-child, - .item-avatar .item-content i.item-image:first-child, - .item-avatar .item-content img.item-image:first-child, - .item-avatar-left > i:first-child, - .item-avatar-left > img:first-child, - .item-avatar-left i.item-image:first-child, - .item-avatar-left img.item-image:first-child, - .item-avatar-left .item-content > i:first-child, - .item-avatar-left .item-content > img:first-child, - .item-avatar-left .item-content i.item-image:first-child, - .item-avatar-left .item-content img.item-image:first-child { - color: #D9D9D9; - background-color: #f8f8f8; - border: solid 1px #D9D9D9; - overflow: hidden !important; - font-size: 45px !important; - /*padding-left: 0px; - padding-top: 0px; - text-align: center; - vertical-align: middle;*/ - line-height: 56px; - width: 100% !important; - /*display: block !important;*/ - max-height: 56px !important; - max-width: 56px !important; - top: 12px !important; } - - .item-avatar > .icon:first-child:before, .item-avatar > .icon-help:first-child:before, .item-avatar > .icon-alert:first-child:before, #menu .footer .item-avatar > .icon-help:first-child:before, - .item-avatar .icon.item-image:first-child:before, - .item-avatar .item-image.icon-help:first-child:before, - .item-avatar .item-image.icon-alert:first-child:before, - .item-avatar #menu .footer .item-image.icon-help:first-child:before, #menu .footer - .item-avatar .item-image.icon-help:first-child:before, - .item-avatar .item-content > .icon:first-child:before, - .item-avatar .item-content > .icon-help:first-child:before, - .item-avatar .item-content > .icon-alert:first-child:before, - .item-avatar #menu .footer .item-content > .icon-help:first-child:before, #menu .footer - .item-avatar .item-content > .icon-help:first-child:before, - .item-avatar.item-icon-right .icon:first-child:before, - .item-avatar.item-icon-right .icon-help:first-child:before, - .item-avatar.item-icon-right .icon-alert:first-child:before, - .item-avatar.item-icon-right #menu .footer .icon-help:first-child:before, #menu .footer - .item-avatar.item-icon-right .icon-help:first-child:before, - .item-avatar.item-icon-right .icon-help:first-child:before, - .item-avatar.item-icon-right .icon-alert:first-child:before, - .item-avatar.item-icon-right #menu .footer .icon-help:first-child:before, - #menu .footer .item-avatar.item-icon-right .icon-help:first-child:before { - width: 56px !important; } - - .item-avatar.item-icon-right .icon:last-child, .item-avatar.item-icon-right .icon-help:last-child, .item-avatar.item-icon-right .icon-alert:last-child, .item-avatar.item-icon-right #menu .footer .icon-help:last-child, #menu .footer .item-avatar.item-icon-right .icon-help:last-child, - .item-avatar.item-icon-right .icon-help:last-child, - .item-avatar.item-icon-right .icon-alert:last-child, - .item-avatar.item-icon-right #menu .footer .icon-help:last-child, - #menu .footer .item-avatar.item-icon-right .icon-help:last-child { - left: auto; } - - /********** - Item > Other - **********/ - .item em { - font-weight: bold !important; } - - @media screen and (min-width: 992px) { - .list .item.item-border-large { - border-bottom: solid 1px #ccc !important; } - .list.item-border-large .item { - border-bottom: solid 1px #ccc !important; - margin: 0px; } - .list.item-border-large .item-divider { - border-top: 0; } } - - .list .item.item-border { - border-bottom: solid 1px #ccc !important; } - - .list .item.item-small-height { - padding-top: 2px; - padding-bottom: 0px; - min-height: 24px; } - .list .item.item-small-height .badge { - padding-top: 0px !important; - top: inherit; } - .list .item.item-small-height .badge.badge-balanced, - .list .item.item-small-height .badge.badge-positive, - .list .item.item-small-height .badge.badge-assertive, - .list .item.item-small-height .badge.badge-editable:hover, - .list .item.item-small-height .badge.badge-royal, - .list .item.item-small-height .badge.badge-calm, - .list .item.item-small-height .badge.badge-energized { - top: 1px !important; - padding-top: 3px !important; - padding-bottom: 2px !important; } - - /* - * Badge - */ - .item.item-icon-right .badge, - .item.item-button-right .badge { - right: 43px; } - - .badge-editable:hover { - cursor: pointer; } - - .badge-editable:hover:before { - content: " "; } - - .badge-button { - position: absolute !important; - margin: 0 !important; - padding: 0px 4px !important; - font-size: 10px; - top: 5px !important; - right: 3px !important; } - - /********** - Item buttons - **********/ - .item-button-right > .button, - .item-button-right .item-content > .button, - .item-button-right > .buttons, - .item-button-right .item-content > .buttons { - top: 16px; } - - .item-button-right > .button.button-small, - .item-button-right .item-content > .button.button-small, - .item-button-right > .buttons .button-small, - .item-button-right .item-content > .buttons .button-small { - font-size: 14px; } - - .item.large-button-bar { - margin-bottom: 10px; } - - /********** - Item avatar - **********/ - .item-avatar-left-padding { - padding-left: 95px; } - - /********** - Item thumbnail - **********/ - .item-thumbnail-left-padding { - padding-left: 106px; } - - .item.item-thumbnail-left, .item-thumbnail-left { - min-height: 100px !important; } - - .item-thumbnail-left > i:first-child, - .item-thumbnail-left i.item-image, - .item-thumbnail-left .item-content > i:first-child, - .item-thumbnail-left .item-content i.item-image { - color: #D9D9D9; - background-color: #f8f8f8; - overflow: hidden !important; - font-size: 50px !important; - line-height: 80px; - padding: 0 15px; - background-position: center; - background-size: cover; - display: inline-block; } - - @media screen and (max-width: 400px) { - .card > .item.item-thumbnail-left, - .item-thumbnail-left, - .item-thumbnail-left .item-content { - padding-left: 84px !important; } - .item-thumbnail-left > img:first-child, - .item-thumbnail-left img.item-image, - .item-thumbnail-left .item-content > img:first-child, - .item-thumbnail-left .item-content img.item-image { - max-width: 70px; - max-height: 70px; } - .item h2 { - font-size: 13px !important; } } - - /********** - Item icons - **********/ - .item-icon-left-padding { - padding-left: 40px; } - - .item-icon-right-padding { - padding-right: 40px; } - - .text-keep-lines { - white-space: pre-line !important; } - - .text-italic { - font-style: italic !important; } - - /********** - Help - **********/ - .icon-help { - font-size: 38px; - vertical-align: middle; } - - .icon-alert { - font-size: 38px; - vertical-align: middle; } - - .bar-header .button-icon .avatar { - height: 35px; - width: 35px; - position: relative; - left: 0px; - top: 4px; - border: solid 1px #D9D9D9; } - - .bar-header .button-icon .avatar.active { - background-color: #e0e0e0; } - - .bar-header .button-icon .avatar { - height: 31px; - width: 31px; - position: relative; - left: 0px; - top: 6px; } - - .badge.badge-secondary, - .badge .badge-secondary { - font-size: 12px; - font-style: italic; - top: 37px !important; - font-weight: normal !important; - margin-right: 0px; - padding-right: 0px; } - - .ion-es-user-api:before { - content: url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPHN2ZyB3aWR0aD0iMzBweCIgaGVpZ2h0PSIzMHB4IiB2aWV3Qm94PSIwIDAgNTAwIDUwMCIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczpieD0iaHR0cHM6Ly9ib3h5LXN2Zy5jb20iPgogIDxwYXRoIGQ9Ik0gMzkwLjk5MiAxMjIuMjc2IEMgNDE4LjIwOSAxNTcuMTgzIDQzNC45MjQgMjAwLjI3OCA0NDEuMjAyIDI0OC4yNCBDIDQ0MS4xODcgMjc2LjcxMSA0MzUuNzEgMzA1LjI5MiA0MjUuMzA4IDMzMy4yODIgQyAzNzEuMTAxIDQxNi43MjcgMjc0Ljk1MSA0NDkuMjEzIDE2Ny42NjkgNDM0LjI3OSBDIDEzNi40MDYgNDE2LjAzNiAxMDkuMDA5IDM5MC4wNzYgODYuMjcyIDM1OC4wMTYgQyA2Ny43NzggMzIzLjc5NCA1Ny45NDIgMjg0LjQgNTYuNDg5IDI0Mi4wNjIgQyA2Mi4yMDYgMTk5Ljc2NiA3OS42MTcgMTYxLjkxMiAxMDYuMzAxIDEyOS42MTcgQyAxNjIuNTI0IDg1LjQzOSAyMzkuMDMgNzAuMTgxIDMyMS45NjggODIuNTIgQyAzNDUuNTA4IDkyLjUzNSAzNjguNjU3IDEwNS44ODMgMzkwLjk5MiAxMjIuMjc2IFoiIHN0eWxlPSJmaWxsOiByZ2IoMjU1LCAyNTUsIDI1NSk7IiBieDpvcmlnaW49IjAgMCIvPgogIDxnIHRyYW5zZm9ybT0ibWF0cml4KDAuNTE3ODQ4LCAwLCAwLCAwLjUxNzg0OCwgLTUzLjMwNjYyNSwgLTU5OS45MzEyMTMpIiBzdHlsZT0ib3BhY2l0eTogMTsiPgogICAgPGcgaWQ9ImctMTQiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDAuNTsiIHRyYW5zZm9ybT0ibWF0cml4KDEsIDAsIDAsIDEsIDE0NC41NzA3MjQsIDEwMDcuMDk5NDI2KSI+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OmlubGluZTtmaWxsOiNmZmQwODY7ZmlsbC1vcGFjaXR5OjE7ZmlsbC1ydWxlOmV2ZW5vZGQ7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjFweDtzdHJva2UtbGluZWNhcDpidXR0O3N0cm9rZS1saW5lam9pbjptaXRlcjtzdHJva2Utb3BhY2l0eToxIiBkPSJNIDU4NS4xNjk5Miw1MjQuOTEyMTEgQyA0NDcuNDE3NDUsNzM4Ljg0MDE1IDI4NS45MzA5Myw3OTcuNjgxNDIgOTMuMzQ3NjU2LDgwMS4wMzcxMSAxNTguNjY0NTIsOTIxLjg0MDgzIDI4Ny4yMDAzOCwxMDAzLjE2OTggNDM0LjAzOTA2LDEwMDEuMjU1OSA2MDcuNTY4NTEsOTk4Ljk5NDQ2IDc1Mi41MTMxNyw4ODEuMTg0ODggNzk2LjY1MjM0LDcyMS45NjY4IGMgLTIuNjg2NCwtNi41Nzc2NCAtNi4yMDEwNiwtMTMuNjIwMzcgLTEwLjgxNjQsLTIxLjEzNDc3IEMgNzY4LjY4OTg3LDY3Ny4wMzg3OCA3MDkuMTA0NzgsNTY4LjQ3NzIxIDU4NS4xNjk5Miw1MjQuOTEyMTEgWiIgaWQ9InBhdGgtMTAyIi8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OmlubGluZTtmaWxsOiMyNzBiMGI7ZmlsbC1vcGFjaXR5OjAuOTkzOTM5Mzk7ZmlsbC1ydWxlOmV2ZW5vZGQ7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjFweDtzdHJva2UtbGluZWNhcDpidXR0O3N0cm9rZS1saW5lam9pbjptaXRlcjtzdHJva2Utb3BhY2l0eToxIiBkPSJNIDU4Ni4yNTQgNTI0LjExNSBDIDU2NC42ODIgNjUwLjQ1MyA0NzcuOTc0IDc1NC40NzIgNDk4LjU4NCA4MzUuNjAyIEMgNTI0LjY0OCA5MzguMTk5IDQxOS40NTggOTYxLjUxNSAzMzMuOTczIDk4OS4zNDIgQyAzNjUuOTIzIDk5Ny41NDUgMzk5LjQ3NSAxMDAxLjcwNiA0MzQuMDM5IDEwMDEuMjU2IEMgNjM0LjA1MiA5OTguNjQ5IDc5Ni4wOTMgODQyLjUzNiA4MDkuNTYxIDY0Ni40MzkgQyA3OTMuMTk3IDY0MS43NzcgNzc3LjQyNSA2MzQuNjg5IDc2Ni43MjMgNjIyLjc0NiBDIDczNC4wMzUgNTg2LjI3MiA2NTAuMTE3IDU0Ni41NjQgNTg2LjI1NCA1MjQuMTE1IFoiIGlkPSJwYXRoLTEwMyIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYigyMDMsIDEzNywgMyk7IGZpbGwtb3BhY2l0eTogMC45OTM5Mzk7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyIgZD0iTSA1ODYuMjU0IDUyNC4xMTUgQyA1NjQuNjgyIDY1MC40NTMgNDc3Ljk3NCA3NTQuNDcyIDQ5OC41ODQgODM1LjYwMiBDIDUyNC42NDggOTM4LjE5OSA0MTkuNDU4IDk2MS41MTUgMzMzLjk3MyA5ODkuMzQyIEMgMzY1LjkyMyA5OTcuNTQ1IDM5OS40NzUgMTAwMS43MDYgNDM0LjAzOSAxMDAxLjI1NiBDIDYzNC4wNTIgOTk4LjY0OSA3OTYuMDkzIDg0Mi41MzYgODA5LjU2MSA2NDYuNDM5IEMgNzkzLjE5NyA2NDEuNzc3IDc3Ny40MjUgNjM0LjY4OSA3NjYuNzIzIDYyMi43NDYgQyA3MzQuMDM1IDU4Ni4yNzIgNjUwLjExNyA1NDYuNTY0IDU4Ni4yNTQgNTI0LjExNSBaIiBpZD0icGF0aC0xMDQiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgPC9nPgogICAgPGcgaWQ9ImctMTUiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDAuNTsiIHRyYW5zZm9ybT0ibWF0cml4KDEsIDAsIDAsIDEsIDE0NC41NzA3MjQsIDEwMDcuMDk5NDI2KSI+CiAgICAgIDxyZWN0IHg9IjUwNC4yNjUiIHk9IjUwMC4yODciIHdpZHRoPSIyMy40OTIiIGhlaWdodD0iMjUuNTI3IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2ZhYmIzNztmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMDUiLz4KICAgICAgPHJlY3QgeD0iMzY5LjE3NCIgeT0iNDE1LjQyOSIgd2lkdGg9IjIyLjM4NCIgaGVpZ2h0PSIyNC40MTkiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojZmZkMDg2O2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTEwNiIvPgogICAgICA8cmVjdCB4PSI0MTguOTk1IiB5PSI0MzMuMDE5IiB3aWR0aD0iMzAuNTI0IiBoZWlnaHQ9IjMwLjUyNCIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmYWJiMzc7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTA3Ii8+CiAgICAgIDxyZWN0IHg9IjQxNy4zNCIgeT0iNjUyLjU1NiIgd2lkdGg9IjQ2LjgwMyIgaGVpZ2h0PSI0Ni44MDMiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojZmJjMTRjO2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTEwOCIvPgogICAgICA8cmVjdCB4PSI0MjIuNTg2IiB5PSI0NzUuODkxIiB3aWR0aD0iMzAuNTI0IiBoZWlnaHQ9IjMwLjUyNCIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmYWJiMzc7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTA5Ii8+CiAgICAgIDxyZWN0IHg9IjQ3Mi42MTgiIHk9IjYwNS40NTciIHdpZHRoPSIyNC40MTkiIGhlaWdodD0iMjYuNDU0IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2NjODkwMjtmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMTAiLz4KICAgICAgPHJlY3QgeD0iNTIwLjc3MiIgeT0iNTU3LjkwMiIgd2lkdGg9IjE4LjMxNCIgaGVpZ2h0PSIxOC4zMTQiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojZmFiYjM3O2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTExMSIvPgogICAgICA8cmVjdCB4PSI0NTQuNzg0IiB5PSI1NjMuMDI4IiB3aWR0aD0iMzAuNTI0IiBoZWlnaHQ9IjMwLjUyNCIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmYWJiMzc7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTEyIi8+CiAgICAgIDxyZWN0IHg9IjMzNS4zNDIiIHk9IjcyMC45ODciIHdpZHRoPSIzOC42NjMiIGhlaWdodD0iNDAuNjk4IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2ZiYzE0YztmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMTMiLz4KICAgICAgPHJlY3QgeD0iMzcxLjk3IiB5PSI2NjEuOTc1IiB3aWR0aD0iMjYuNDU0IiBoZWlnaHQ9IjMwLjUyNCIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmYmMxNGM7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTE0Ii8+CiAgICAgIDxyZWN0IHg9Ii00MjcuOTMiIHk9IjYxMC4wODUiIHdpZHRoPSIyNC41MDciIGhlaWdodD0iMjQuNDE5IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2NjODkwMjtmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMTUiIHRyYW5zZm9ybT0ic2NhbGUoLTEsMSkiLz4KICAgICAgPHJlY3QgeD0iNDgwLjQxMSIgeT0iNTIzLjQ2OSIgd2lkdGg9IjIwLjM0OSIgaGVpZ2h0PSIyMi4zODQiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojZmZkMDg2O2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTExNiIvPgogICAgICA8cmVjdCB4PSItNDk5LjUiIHk9IjQ2Ni4zMTQiIHdpZHRoPSIyNC41MDciIGhlaWdodD0iMjQuNDE5IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2NjODkwMjtmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMTciIHRyYW5zZm9ybT0ic2NhbGUoLTEsMSkiLz4KICAgIDwvZz4KICAgIDxnIGlkPSJnLTE2IiBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBvcGFjaXR5OiAxOyIgdHJhbnNmb3JtPSJtYXRyaXgoMS4wMDAwMDAwNzAwMDIwNDk1LCAwLCAwLCAxLjAwMDAwMDA3MDAwMjA0OTUsIDE0NC41NzA3MjM5MTA3OTYxMiwgMTAwNy4wOTk0Mzk4NzE1OTk0KSI+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IG5vbmU7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IHJnYigyNTUsIDEyMiwgMCk7IHN0cm9rZS13aWR0aDogMi4zNDg4ODsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyIgZD0iTSA1ODUuMTY5OTIsNTI0LjkxMjExIEMgNDQ3LjQxNzQ1LDczOC44NDAxNSAyODUuOTMwOTMsNzk3LjY4MTQyIDkzLjM0NzY1Niw4MDEuMDM3MTEgMTU4LjY2NDUyLDkyMS44NDA4MyAyODcuMjAwMzgsMTAwMy4xNjk4IDQzNC4wMzkwNiwxMDAxLjI1NTkgNjA3LjU2ODUxLDk5OC45OTQ0NiA3NTIuNTEzMTcsODgxLjE4NDg4IDc5Ni42NTIzNCw3MjEuOTY2OCBjIC0yLjY4NjQsLTYuNTc3NjQgLTYuMjAxMDYsLTEzLjYyMDM3IC0xMC44MTY0LC0yMS4xMzQ3NyBDIDc2OC42ODk4Nyw2NzcuMDM4NzggNzA5LjEwNDc4LDU2OC40NzcyMSA1ODUuMTY5OTIsNTI0LjkxMjExIFoiIGlkPSJwYXRoLTExOCIvPgogICAgICA8cGF0aCBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBmaWxsOiBub25lOyBmaWxsLW9wYWNpdHk6IDAuOTkzOTM5OyBmaWxsLXJ1bGU6IGV2ZW5vZGQ7IHN0cm9rZTogcmdiKDI1NSwgMTIyLCAwKTsgc3Ryb2tlLXdpZHRoOiAyLjM0ODg4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IiBkPSJNIDU4Ni4yNTQgNTI0LjExNSBDIDU2NC42ODIgNjUwLjQ1MyA0NzcuOTc0IDc1NC40NzIgNDk4LjU4NCA4MzUuNjAyIEMgNTI0LjY0OCA5MzguMTk5IDQxOS40NTggOTYxLjUxNSAzMzMuOTczIDk4OS4zNDIgQyAzNjUuOTIzIDk5Ny41NDUgMzk5LjQ3NSAxMDAxLjcwNiA0MzQuMDM5IDEwMDEuMjU2IEMgNjM0LjA1MiA5OTguNjQ5IDc5Ni4wOTMgODQyLjUzNiA4MDkuNTYxIDY0Ni40MzkgQyA3OTMuMTk3IDY0MS43NzcgNzc3LjQyNSA2MzQuNjg5IDc2Ni43MjMgNjIyLjc0NiBDIDczNC4wMzUgNTg2LjI3MiA2NTAuMTE3IDU0Ni41NjQgNTg2LjI1NCA1MjQuMTE1IFoiIGlkPSJwYXRoLTExOSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IG5vbmU7IGZpbGwtb3BhY2l0eTogMC45OTM5Mzk7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiByZ2IoMjU1LCAxMjIsIDApOyBzdHJva2Utd2lkdGg6IDIuMzQ4ODg7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2Utb3BhY2l0eTogMTsiIGQ9Ik0gNTg2LjI1NCA1MjQuMTE1IEMgNTY0LjY4MiA2NTAuNDUzIDQ3Ny45NzQgNzU0LjQ3MiA0OTguNTg0IDgzNS42MDIgQyA1MjQuNjQ4IDkzOC4xOTkgNDE5LjQ1OCA5NjEuNTE1IDMzMy45NzMgOTg5LjM0MiBDIDM2NS45MjMgOTk3LjU0NSAzOTkuNDc1IDEwMDEuNzA2IDQzNC4wMzkgMTAwMS4yNTYgQyA2MzQuMDUyIDk5OC42NDkgNzk2LjA5MyA4NDIuNTM2IDgwOS41NjEgNjQ2LjQzOSBDIDc5My4xOTcgNjQxLjc3NyA3NzcuNDI1IDYzNC42ODkgNzY2LjcyMyA2MjIuNzQ2IEMgNzM0LjAzNSA1ODYuMjcyIDY1MC4xMTcgNTQ2LjU2NCA1ODYuMjU0IDUyNC4xMTUgWiIgaWQ9InBhdGgtMTIwIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgIDwvZz4KICA8L2c+CiAgPGcgdHJhbnNmb3JtPSJtYXRyaXgoMC42MDgyNjEsIDAsIDAsIDAuNjA4MjYxLCAtMjAuMDg0OTc2LCAzLjI1NTczNikiPgogICAgPGc+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYig2NCwgMTc4LCAyNTUpOyBmaWxsLW9wYWNpdHk6IDE7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyBvcGFjaXR5OiAwLjc4OyIgaWQ9InBhdGgtMTAiIGQ9Ik0gNDE5LjEzMyA4NS43MzggQyA0MTcuMjk1IDg1LjczOSA0MTUuNDU1IDg1Ljc1NCA0MTMuNjExIDg1Ljc3OCBDIDM1NC44NTIgODYuNTYxIDI5OS4wMjMgOTkuMTc0IDI0OC4zNTggMTIxLjMyIEMgNTg0LjEyMSAxMy4yMDYgNzk2LjE3NSAyMTEuMjYgODMxLjcgNTA4Ljc3MyBMIDg0Ni4zODMgNTA3LjEyNCBDIDgxNS41ODYgMjM1LjgwMSA2NTIuNDc3IDg1LjU4OCA0MTkuMTMzIDg1LjczOCBaIiB0cmFuc2Zvcm09Im1hdHJpeCgwLjk2MzczLCAwLjI2Njg3OCwgLTAuMjY2ODc4LCAwLjk2MzczLCA4OS41MDUyMzEsIC0xMzYuNjE1MDYxKSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYig2NCwgMTc4LCAyNTUpOyBmaWxsLW9wYWNpdHk6IDE7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyBvcGFjaXR5OiAxOyIgaWQ9InBhdGgtMyIgZD0iTSAxNDYuMDM3IDE5NC4wNjUgQyAxNDQuMjAxIDE5NC4wNjYgMTQyLjM2MSAxOTQuMDgxIDE0MC41MTcgMTk0LjEwNCBDIDgxLjc2IDE5NC44ODggMjUuOTMxIDIwNy41MDMgLTI0LjczNiAyMjkuNjQ4IEMgMzExLjAyOSAxMjEuNTMxIDUyMy4wODMgMzE5LjU4MyA1NTguNjA0IDYxNy4wOTMgTCA1NzMuMjg0IDYxNS40NDMgQyA1NDIuNDkgMzQ0LjEyMiAzNzkuMzg1IDE5My45MTMgMTQ2LjAzNyAxOTQuMDY1IFoiIHRyYW5zZm9ybT0ibWF0cml4KC0wLjE4MDk4OCwgLTAuOTgzNDg1LCAwLjk4MzQ4NSwgLTAuMTgwOTg4LCAtMzkuMjk4NTgxLCA3MDUuODk3NTQ0KSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYig2NCwgMTc4LCAyNTUpOyBmaWxsLW9wYWNpdHk6IDE7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyBvcGFjaXR5OiAwLjQ7IiBpZD0icGF0aC0xMSIgZD0iTSA0ODcuNzQ0IDE5My4wODQgQyA0ODUuOTA2IDE5My4wODUgNDg0LjA2NyAxOTMuMSA0ODIuMjIzIDE5My4xMjMgQyA0MjMuNDY0IDE5My45MDcgMzY3LjYzNSAyMDYuNTIgMzE2Ljk3NSAyMjguNjY4IEMgNjUyLjcyNiAxMjAuNTUyIDg2NC43NzUgMzE4LjYwNSA5MDAuMjk5IDYxNi4xMDkgTCA5MTQuOTc5IDYxNC40NiBDIDg4NC4xODQgMzQzLjE0NCA3MjEuMDgxIDE5Mi45MzUgNDg3Ljc0NCAxOTMuMDg0IFoiIHRyYW5zZm9ybT0ibWF0cml4KDAuNjE5OTk3LCAwLjc4NDYwNSwgLTAuNzg0NjA1LCAwLjYxOTk5NywgNTIzLjA2NzIzMSwgLTM0My4zMzE1MzkpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDAuODQ7IiBpZD0icGF0aC0xMiIgZD0iTSAyOTUuOTY5IDQzMi43NTcgQyAyOTQuMTMxIDQzMi43NTggMjkyLjI5MyA0MzIuNzczIDI5MC40NDkgNDMyLjc5NyBDIDIzMS42ODggNDMzLjU4IDE3NS44NiA0NDYuMTk0IDEyNS4xOTggNDY4LjM0MSBDIDQ2MC45NTcgMzYwLjIyNiA2NzMuMDA1IDU1OC4yOCA3MDguNTI2IDg1NS43ODYgTCA3MjMuMjA3IDg1NC4xMzkgQyA2OTIuNDEyIDU4Mi44MTkgNTI5LjMxIDQzMi42MDkgMjk1Ljk2OSA0MzIuNzU3IFoiIHRyYW5zZm9ybT0ibWF0cml4KC0wLjk3ODg0OSwgMC4yMDQ1ODQsIC0wLjIwNDU4NCwgLTAuOTc4ODQ5LCA5NjMuODIwNzk2LCAxMTE2LjM2NzI5MykiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgICA8cGF0aCBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBmaWxsOiByZ2IoNjQsIDE3OCwgMjU1KTsgZmlsbC1vcGFjaXR5OiAxOyBmaWxsLXJ1bGU6IGV2ZW5vZGQ7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAxcHg7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2Utb3BhY2l0eTogMTsgb3BhY2l0eTogMC40OyIgaWQ9InBhdGgtMTMiIGQ9Ik0gMTU4LjAzNiAxNDguODMgQyAxNTYuMiAxNDguODMzIDE1NC4zNiAxNDguODQ2IDE1Mi41MTYgMTQ4Ljg3MiBDIDkzLjc1OSAxNDkuNjUzIDM3LjkzIDE2Mi4yNjcgLTEyLjczNCAxODQuNDExIEMgMzIzLjAyMiA3Ni4zMDEgNTM1LjA3NiAyNzQuMzUyIDU3MC42MDQgNTcxLjg1NyBMIDU4NS4yODYgNTcwLjIwOCBDIDU1NC40ODMgMjk4Ljg5MSAzOTEuMzggMTQ4LjY4MyAxNTguMDM2IDE0OC44MyBaIiB0cmFuc2Zvcm09Im1hdHJpeCgwLjMzODYxNCwgLTAuOTQwOTI2LCAwLjk0MDkyNiwgMC4zMzg2MTQsIC0xMTUuNTk1MTU4LCA0ODMuNzA1OTg0KSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYig2NCwgMTc4LCAyNTUpOyBmaWxsLW9wYWNpdHk6IDE7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyBvcGFjaXR5OiAxOyIgaWQ9InBhdGgtMTQiIGQ9Ik0gNTAxLjAxOSAyODguMjkyIEMgNDk5LjE4MSAyODguMjkzIDQ5Ny4zNCAyODguMzA4IDQ5NS40OTYgMjg4LjMzMSBDIDQzNi43MzYgMjg5LjExNSAzODAuOTA4IDMwMS43MjYgMzMwLjI0MyAzMjMuODc0IEMgNjY2LjAwNCAyMTUuNzU3IDg3OC4wNjIgNDEzLjgxMiA5MTMuNTg2IDcxMS4zMjIgTCA5MjguMjY4IDcwOS42NzMgQyA4OTcuNDY4IDQzOC4zNTIgNzM0LjM2MSAyODguMTQxIDUwMS4wMTkgMjg4LjI5MiBaIiB0cmFuc2Zvcm09Im1hdHJpeCgwLjE2NDY1NiwgMC45ODYzNTEsIC0wLjk4NjM1MSwgMC4xNjQ2NTYsIDk4Mi44NTcyNjYsIC0yMzMuNDUyMDYzKSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYig2NCwgMTc4LCAyNTUpOyBmaWxsLW9wYWNpdHk6IDE7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyBvcGFjaXR5OiAwLjQ7IiBpZD0icGF0aC0xNSIgZD0iTSAxOTIuOTU0IDM3Ni41MjEgQyAxOTEuMTE1IDM3Ni41MjIgMTg5LjI3NyAzNzYuNTM3IDE4Ny40MzMgMzc2LjU2MSBDIDEyOC42NzQgMzc3LjM0NCA3Mi44NDUgMzg5Ljk1OCAyMi4xODEgNDEyLjEwMyBDIDM1Ny45NCAzMDMuOTg5IDU2OS45ODcgNTAyLjA0NSA2MDUuNTA1IDc5OS41NTMgTCA2MjAuMTg4IDc5Ny45MDUgQyA1ODkuMzkyIDUyNi41ODQgNDI2LjI5NSAzNzYuMzY5IDE5Mi45NTQgMzc2LjUyMSBaIiB0cmFuc2Zvcm09Im1hdHJpeCgtMC45MzE0MDQsIC0wLjM2Mzk4OCwgMC4zNjM5ODgsIC0wLjkzMTQwNCwgNDE5LjQ5OTAwNiwgMTE4Mi41OTk4MDkpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDE7IiBpZD0icGF0aC0xNiIgZD0iTSAyMjEuNjE0IDYzLjk3MyBDIDIxOS43NzcgNjMuOTc2IDIxNy45MzggNjMuOTkgMjE2LjA5NCA2NC4wMTUgQyAxNTcuMzM0IDY0Ljc5NSAxMDEuNTA2IDc3LjQwOSA1MC44NDEgOTkuNTU0IEMgMzg2LjU5OCAtOC41NTkgNTk4LjY1MSAxODkuNDk3IDYzNC4xNzMgNDg3LjAwMSBMIDY0OC44NTQgNDg1LjM1NCBDIDYxOC4wNTggMjE0LjAzNyA0NTQuOTU0IDYzLjgyNCAyMjEuNjE0IDYzLjk3MyBaIiB0cmFuc2Zvcm09Im1hdHJpeCgwLjc1MDQ1MiwgLTAuNjYwOTI1LCAwLjY2MDkyNSwgMC43NTA0NTIsIC03MC44MDMzMjIsIDI5MC45MjAyNzMpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDAuNDsiIGlkPSJwYXRoLTE3IiBkPSJNIDQ0MS44ODggMzQyLjA5OCBDIDQ0MC4wNDcgMzQyLjA5OCA0MzguMjA3IDM0Mi4xMTQgNDM2LjM2MiAzNDIuMTM4IEMgMzc3LjYwMSAzNDIuOTIyIDMyMS43NjUgMzU1LjUzNSAyNzEuMTAzIDM3Ny42ODEgQyA2MDYuODg2IDI2OS41NjcgODE4Ljk0NCA0NjcuNjE1IDg1NC40NjIgNzY1LjEzMyBMIDg2OS4xNDIgNzYzLjQ4MyBDIDgzOC4zNTQgNDkyLjE1NCA2NzUuMjQyIDM0MS45NDggNDQxLjg4OCAzNDIuMDk4IFoiIHRyYW5zZm9ybT0ibWF0cml4KC0wLjM1NjU4MiwgMC45MzQyNjQsIC0wLjkzNDI2NCwgLTAuMzU2NTgyLCAxMjU2Ljc1OTQ5MywgMTY5LjE4MjU1MikiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgICA8ZyB0cmFuc2Zvcm09Im1hdHJpeCgwLjc2MjQ5MywgMCwgMCwgMC43NjI0OTMsIDE5LjI5NDY0NywgLTcxNi4zMjI4MTUpIj4KICAgICAgICA8cGF0aCBkPSJNIDU2Ni40NjYgMTA4OS4xMzcgTCA1NjYuNDY2IDExNDguMjIyIEMgNTY1Ljg2NyAxMTQ4LjIxOSA1NjUuMjY3IDExNDguMjE3IDU2NC42NjcgMTE0OC4yMTcgQyA1NjEuNDgxIDExNDguMjE3IDU1OC4zMSAxMTQ4LjI1OSA1NTUuMTU0IDExNDguMzQxIEwgNTU1LjE1NCAxMDg5LjEzNyBaIE0gOTA4LjI5IDE0NzcuMTQ2IEwgOTU2Ljk4NyAxNDc3LjE0NiBMIDk1Ni45ODcgMTQ4OC40NTggTCA5MDguNTY5IDE0ODguNDU4IEMgOTA4LjUzOSAxNDg0LjY5MyA5MDguNDQ3IDE0ODAuOTIyIDkwOC4yOSAxNDc3LjE0NiBaIE0gNTY2LjQ2NiAxODM1LjAxMyBMIDU2Ni40NjYgMTg3Ni40NjcgTCA1NTUuMTU0IDE4NzYuNDY3IEwgNTU1LjE1NCAxODM0Ljg4NiBDIDU1OC4zMTggMTgzNC45NzMgNTYxLjQ5IDE4MzUuMDE3IDU2NC42NjcgMTgzNS4wMTcgQyA1NjUuMjY3IDE4MzUuMDE3IDU2NS44NjcgMTgzNS4wMTYgNTY2LjQ2NiAxODM1LjAxMyBaIE0gMjIwLjc3NSAxNDg4LjQ1OCBMIDE2OS42NTcgMTQ4OC40NTggTCAxNjkuNjU3IDE0NzcuMTQ2IEwgMjIxLjA4OSAxNDc3LjE0NiBDIDIyMC45MjEgMTQ4MC45MDkgMjIwLjgxNiAxNDg0LjY4IDIyMC43NzUgMTQ4OC40NTggWiIgc3R5bGU9ImZpbGw6IHJnYig2NCwgNDAsIDApOyBzdHJva2U6IHJnYig2NCwgNDAsIDApOyBzdHJva2Utd2lkdGg6IDEwLjc4MDY7IiBieDpvcmlnaW49IjAgMCIvPgogICAgICAgIDxwYXRoIGQ9Ik0gNzYwLjEyMSAxMTQzLjMxMSBMIDcxNy4zMzYgMTIxNy40MTcgQyA3MTUuNzIgMTIxNi41MTggNzE0LjA5NiAxMjE1LjYzMiA3MTIuNDY1IDEyMTQuNzYxIEwgNzU1LjMxOCAxMTQwLjUzNyBaIE0gODMwLjEyMiAxMzI0LjIwMyBMIDkwMC40MjQgMTI4My42MTQgTCA5MDMuMTk4IDEyODguNDE3IEwgODMzLjA0IDEzMjguOTIzIEMgODMyLjA4IDEzMjcuMzM5IDgzMS4xMDcgMTMyNS43NjYgODMwLjEyMiAxMzI0LjIwMyBaIE0gODQwLjU5OSAxNjQxLjEzNyBMIDkwMy4xOTcgMTY3Ny4yNzggTCA5MDAuNDI0IDE2ODIuMDgyIEwgODM3LjkxMSAxNjQ1Ljk5IEMgODM4LjgyMSAxNjQ0LjM4IDgzOS43MTcgMTY0Mi43NjIgODQwLjU5OSAxNjQxLjEzNyBaIE0gNzI1LjE2OCAxNzYxLjg0MyBMIDc2MC4xMjIgMTgyMi4zODUgTCA3NTUuMzE4IDE4MjUuMTU4IEwgNzIwLjM3OSAxNzY0LjY0MSBDIDcyMS45ODUgMTc2My43MjIgNzIzLjU4MiAxNzYyLjc4OSA3MjUuMTY4IDE3NjEuODQzIFogTSA0MDMuNTM4IDE3NjAuOTMyIEwgMzY2LjQ1NiAxODI1LjE1OSBMIDM2MS42NTMgMTgyMi4zODYgTCAzOTguODAyIDE3NTguMDQyIEMgNDAwLjM3MiAxNzU5LjAxOSA0MDEuOTUxIDE3NTkuOTgyIDQwMy41MzggMTc2MC45MzIgWiBNIDI4OS41OCAxNjQyLjY4OSBMIDIyMS4zNSAxNjgyLjA4MiBMIDIxOC41NzcgMTY3Ny4yNzkgTCAyODYuOTQ0IDE2MzcuODA3IEMgMjg3LjgwNyAxNjM5LjQzNyAyODguNjg1IDE2NDEuMDY1IDI4OS41OCAxNjQyLjY4OSBaIE0gMjk0LjM1NCAxMzMyLjE2OCBMIDIxOC41NzYgMTI4OC40MTcgTCAyMjEuMzUgMTI4My42MTQgTCAyOTcuMjEzIDEzMjcuNDE0IEMgMjk2LjI0NiAxMzI4Ljk5IDI5NS4yOTMgMTMzMC41NzUgMjk0LjM1NCAxMzMyLjE2OCBaIE0gNDA2LjA3OSAxMjIwLjI1OSBMIDM2MS42NTMgMTE0My4zMTEgTCAzNjYuNDU3IDExNDAuNTM3IEwgNDEwLjg4OSAxMjE3LjQ5NSBDIDQwOS4yNzYgMTIxOC40MDMgNDA3LjY3MyAxMjE5LjMyNCA0MDYuMDc5IDEyMjAuMjU5IFoiIHN0eWxlPSJmaWxsOiByZ2IoNjQsIDQwLCAwKTsgc3Ryb2tlOiByZ2IoNjQsIDQwLCAwKTsgc3Ryb2tlLXdpZHRoOiAxMC43ODA2OyIgYng6b3JpZ2luPSIwIDAiLz4KICAgICAgPC9nPgogICAgPC9nPgogICAgPHJlY3QgeD0iNDI5LjM2NyIgeT0iNDI1Ljk3OSIgd2lkdGg9IjIwLjAwMyIgaGVpZ2h0PSIyMS43MzUiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDE7IGZpbGw6IHJnYig4MCwgMTUwLCAyMDApOyBmaWxsLW9wYWNpdHk6IDE7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAzNC45OyBzdHJva2UtbWl0ZXJsaW1pdDogNDsgc3Ryb2tlLWRhc2hhcnJheTogbm9uZTsgc3Ryb2tlLWRhc2hvZmZzZXQ6IDA7IHN0cm9rZS1vcGFjaXR5OiAwLjk5NjA3ODsiIGlkPSJwYXRoLTkxIiB0cmFuc2Zvcm09Im1hdHJpeCgxLCAwLCAwLCAwLjk5OTk5OSwgLTMwMS4zNzcwNzUsIC0yMTkuNDk4NDM0KSIvPgogICAgPHJlY3QgeD0iNDQ2Ljg1MyIgeT0iMjUuNDYyIiB3aWR0aD0iMjUuOTkiIGhlaWdodD0iMjUuOTkiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDE7IGZpbGw6IHJnYig4MCwgMTUwLCAyMDApOyBmaWxsLW9wYWNpdHk6IDE7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAzNC45OyBzdHJva2UtbWl0ZXJsaW1pdDogNDsgc3Ryb2tlLWRhc2hhcnJheTogbm9uZTsgc3Ryb2tlLWRhc2hvZmZzZXQ6IDA7IHN0cm9rZS1vcGFjaXR5OiAwLjk5NjA3ODsiIGlkPSJwYXRoLTkzIi8+CiAgICA8cmVjdCB4PSItMTM1Ljk1NiIgeT0iNjA1LjE0MSIgd2lkdGg9IjM5Ljg1MSIgaGVpZ2h0PSIzOS44NTEiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDE7IGZpbGw6IHJnYig4MCwgMTUwLCAyMDApOyBmaWxsLW9wYWNpdHk6IDE7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAzNC45OyBzdHJva2UtbWl0ZXJsaW1pdDogNDsgc3Ryb2tlLWRhc2hhcnJheTogbm9uZTsgc3Ryb2tlLWRhc2hvZmZzZXQ6IDA7IHN0cm9rZS1vcGFjaXR5OiAwLjk5NjA3ODsiIGlkPSJwYXRoLTk0IiB0cmFuc2Zvcm09Im1hdHJpeCgwLjk5OTk5OSwgMCwgMCwgMSwgODQ2LjE3Njc1NiwgLTM5NC45NTYwMjQpIi8+CiAgICA8cmVjdCB4PSIyMjUuNzc5IiB5PSI2ODkuODM2IiB3aWR0aD0iMjUuOTkiIGhlaWdodD0iMjUuOTkiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDE7IGZpbGw6IHJnYig4MCwgMTUwLCAyMDApOyBmaWxsLW9wYWNpdHk6IDE7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAzNC45OyBzdHJva2UtbWl0ZXJsaW1pdDogNDsgc3Ryb2tlLWRhc2hhcnJheTogbm9uZTsgc3Ryb2tlLWRhc2hvZmZzZXQ6IDA7IHN0cm9rZS1vcGFjaXR5OiAwLjk5NjA3ODsiIGlkPSJwYXRoLTk1Ii8+CiAgICA8cmVjdCB4PSI1OTQuMDEzIiB5PSI3MzcuMTQyIiB3aWR0aD0iMjAuNzkyIiBoZWlnaHQ9IjIyLjUyNSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTYiLz4KICAgIDxyZWN0IHg9IjM4Ny4yMzUiIHk9IjQ3OS4zOTciIHdpZHRoPSIyNS45OSIgaGVpZ2h0PSIyNS45OSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTgiIHRyYW5zZm9ybT0ibWF0cml4KDEuMDAwMDAxLCAwLCAwLCAwLjk5OTk5OSwgMzY1LjI3NDk2NiwgNzkuMjk1MjQxKSIvPgogICAgPHJlY3QgeD0iNDIxLjk2MiIgeT0iNzc1LjkzNyIgd2lkdGg9IjMyLjkyIiBoZWlnaHQ9IjM0LjY1MyIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTkiLz4KICAgIDxyZWN0IHg9Ii02Ni45MzkiIHk9Ii01NjguOTc4IiB3aWR0aD0iMjAuODY3IiBoZWlnaHQ9IjIwLjc5MiIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtMTAxIiB0cmFuc2Zvcm09Im1hdHJpeCgtMSwgMCwgMCwgMC45OTk5OTcsIDMxLjM1NjkyOCwgOTg0LjczNjMwNikiLz4KICA8L2c+CiAgPHBhdGggZD0iTSAyMTMuOTAzIDE4Ni43NzIgQyAxOTUuODU2IDE4Ni43NzIgMTgxLjYxIDE5Mi43ODIgMTcxLjE2MyAyMDQuODAyIEMgMTYwLjcyMyAyMTYuODE1IDE1NS41MDMgMjMzLjI2OSAxNTUuNTAzIDI1NC4xNjIgQyAxNTUuNTAzIDI3NS42NDkgMTYwLjUzNiAyOTIuMjQ5IDE3MC42MDMgMzAzLjk2MiBDIDE4MC42NzYgMzE1LjY4MiAxOTUuMDMzIDMyMS41NDIgMjEzLjY3MyAzMjEuNTQyIEMgMjI1LjEyNiAzMjEuNTQyIDIzOC4xOTMgMzE5LjQ4MiAyNTIuODczIDMxNS4zNjIgTCAyNTIuODczIDMzMi4wOTIgQyAyNDEuNDkzIDMzNi4zNjUgMjI3LjQ1MyAzMzguNTAyIDIxMC43NTMgMzM4LjUwMiBDIDE4Ni41NzMgMzM4LjUwMiAxNjcuOTEzIDMzMS4xNjIgMTU0Ljc3MyAzMTYuNDgyIEMgMTQxLjYzMyAzMDEuODA5IDEzNS4wNjMgMjgwLjk1OSAxMzUuMDYzIDI1My45MzIgQyAxMzUuMDYzIDIzNy4wMTIgMTM4LjIyNiAyMjIuMTg5IDE0NC41NTMgMjA5LjQ2MiBDIDE1MC44OCAxOTYuNzM1IDE2MC4wMTMgMTg2LjkyNSAxNzEuOTUzIDE4MC4wMzIgQyAxODMuODkzIDE3My4xNDUgMTk3Ljk1IDE2OS43MDIgMjE0LjEyMyAxNjkuNzAyIEMgMjMxLjM0MyAxNjkuNzAyIDI0Ni4zOTMgMTcyLjg0OSAyNTkuMjczIDE3OS4xNDIgTCAyNTEuMTgzIDE5NS41MzIgQyAyMzguNzU2IDE4OS42OTIgMjI2LjMzIDE4Ni43NzIgMjEzLjkwMyAxODYuNzcyIFogTSAzNjUuMTU2IDMwMi42NzIgQyAzNjUuMTU2IDMxNC4xMjUgMzYwLjg4NiAzMjIuOTU5IDM1Mi4zNDYgMzI5LjE3MiBDIDM0My44MTIgMzM1LjM5MiAzMzEuODMyIDMzOC41MDIgMzE2LjQwNiAzMzguNTAyIEMgMzAwLjA4NiAzMzguNTAyIDI4Ny4zNTkgMzM1LjkxOSAyNzguMjI2IDMzMC43NTIgTCAyNzguMjI2IDMxMy40NTIgQyAyODQuMTM5IDMxNi40NDUgMjkwLjQ4NiAzMTguODA1IDI5Ny4yNjYgMzIwLjUzMiBDIDMwNC4wMzkgMzIyLjI1MiAzMTAuNTY5IDMyMy4xMTIgMzE2Ljg1NiAzMjMuMTEyIEMgMzI2LjU4OSAzMjMuMTEyIDMzNC4wNzYgMzIxLjU1OSAzMzkuMzE2IDMxOC40NTIgQyAzNDQuNTYyIDMxNS4zNDUgMzQ3LjE4NiAzMTAuNjA5IDM0Ny4xODYgMzA0LjI0MiBDIDM0Ny4xODYgMjk5LjQ1NSAzNDUuMTA2IDI5NS4zNTkgMzQwLjk0NiAyOTEuOTUyIEMgMzM2Ljc5MiAyODguNTQ1IDMyOC42ODkgMjg0LjUxOSAzMTYuNjM2IDI3OS44NzIgQyAzMDUuMTgyIDI3NS42MDUgMjk3LjAzOSAyNzEuODgyIDI5Mi4yMDYgMjY4LjcwMiBDIDI4Ny4zNzkgMjY1LjUyMiAyODMuNzg2IDI2MS45MDkgMjgxLjQyNiAyNTcuODYyIEMgMjc5LjA2NiAyNTMuODIyIDI3Ny44ODYgMjQ4Ljk5MiAyNzcuODg2IDI0My4zNzIgQyAyNzcuODg2IDIzMy4zNDUgMjgxLjk2OSAyMjUuNDI5IDI5MC4xMzYgMjE5LjYyMiBDIDI5OC4yOTYgMjEzLjgyMiAzMDkuNDg2IDIxMC45MjIgMzIzLjcwNiAyMTAuOTIyIEMgMzM2Ljk1OSAyMTAuOTIyIDM0OS45MTIgMjEzLjYxNSAzNjIuNTY2IDIxOS4wMDIgTCAzNTUuOTQ2IDIzNC4xNzIgQyAzNDMuNTkyIDIyOS4wNzkgMzMyLjM5OSAyMjYuNTMyIDMyMi4zNjYgMjI2LjUzMiBDIDMxMy41MzIgMjI2LjUzMiAzMDYuODY5IDIyNy45MTUgMzAyLjM3NiAyMzAuNjgyIEMgMjk3Ljg4MiAyMzMuNDU1IDI5NS42MzYgMjM3LjI3NSAyOTUuNjM2IDI0Mi4xNDIgQyAyOTUuNjM2IDI0NS40MzUgMjk2LjQ3OSAyNDguMjQyIDI5OC4xNjYgMjUwLjU2MiBDIDI5OS44NDYgMjUyLjg4MiAzMDIuNTU5IDI1NS4wOTIgMzA2LjMwNiAyNTcuMTkyIEMgMzEwLjA0NiAyNTkuMjg1IDMxNy4yMzIgMjYyLjMxNSAzMjcuODY2IDI2Ni4yODIgQyAzNDIuNDY2IDI3MS42MDIgMzUyLjMyOSAyNzYuOTU1IDM1Ny40NTYgMjgyLjM0MiBDIDM2Mi41ODkgMjg3LjczNSAzNjUuMTU2IDI5NC41MTIgMzY1LjE1NiAzMDIuNjcyIFoiIHN0eWxlPSJmaWxsOiByZ2IoNjQsIDQwLCAwKTsiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgo8L3N2Zz4K); } - - .row-header { - border-bottom: solid 1px #ccc !important; - margin: 0px; - min-height: 28px !important; } - - .col-header { - text-align: center; - display: block !important; } - - .col-15 { - -webkit-box-flex: 0; - -webkit-flex: 0 0 15%; - -moz-box-flex: 0; - -moz-flex: 0 0 15%; - -ms-flex: 0 0 15%; - flex: 0 0 15%; - max-width: 15%; } - - .col-border-left { - border-left: solid 1px #ccc !important; } - - .col-border-right { - border-right: solid 1px #ccc !important; } - - .text-no-transform { - text-transform: inherit; } - ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css_logo.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css_logo.st deleted file mode 100644 index 8a554e95b6ad236c0bf36e70d93f8accb5d2329f..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/css_logo.st +++ /dev/null @@ -1,22 +0,0 @@ -css_logo() ::= << - @media screen and (max-width: 767px) { - .logo { - height: 96px; - background-image: url(data:image/png;base64,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); - background-size: 96px 96px; } - } - - @media screen and (min-width: 768px) and (max-width: 991px) { - .logo { - height: 144px; - background-image: url(data:image/png;base64,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); - background-size: 144px 144px; } - } - - @media screen and (min-width: 992px) { - .logo { - height: 200px; - background-image: url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiPz4KPHN2ZyB3aWR0aD0iNTEycHgiIGhlaWdodD0iNTEycHgiIHZpZXdCb3g9IjAgMCA1MDAgNTAwIiB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHhtbG5zOmJ4PSJodHRwczovL2JveHktc3ZnLmNvbSI+CiAgPHBhdGggZD0iTSAzOTAuOTkyIDEyMi4yNzYgQyA0MTguMjA5IDE1Ny4xODMgNDM0LjkyNCAyMDAuMjc4IDQ0MS4yMDIgMjQ4LjI0IEMgNDQxLjE4NyAyNzYuNzExIDQzNS43MSAzMDUuMjkyIDQyNS4zMDggMzMzLjI4MiBDIDM3MS4xMDEgNDE2LjcyNyAyNzQuOTUxIDQ0OS4yMTMgMTY3LjY2OSA0MzQuMjc5IEMgMTM2LjQwNiA0MTYuMDM2IDEwOS4wMDkgMzkwLjA3NiA4Ni4yNzIgMzU4LjAxNiBDIDY3Ljc3OCAzMjMuNzk0IDU3Ljk0MiAyODQuNCA1Ni40ODkgMjQyLjA2MiBDIDYyLjIwNiAxOTkuNzY2IDc5LjYxNyAxNjEuOTEyIDEwNi4zMDEgMTI5LjYxNyBDIDE2Mi41MjQgODUuNDM5IDIzOS4wMyA3MC4xODEgMzIxLjk2OCA4Mi41MiBDIDM0NS41MDggOTIuNTM1IDM2OC42NTcgMTA1Ljg4MyAzOTAuOTkyIDEyMi4yNzYgWiIgc3R5bGU9ImZpbGw6IHJnYigyNTUsIDI1NSwgMjU1KTsiIGJ4Om9yaWdpbj0iMCAwIi8+CiAgPGcgdHJhbnNmb3JtPSJtYXRyaXgoMC41MTc4NDgsIDAsIDAsIDAuNTE3ODQ4LCAtNTMuMzA2NjI1LCAtNTk5LjkzMTIxMykiIHN0eWxlPSJvcGFjaXR5OiAxOyI+CiAgICA8ZyBpZD0iZy0xNCIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMC41OyIgdHJhbnNmb3JtPSJtYXRyaXgoMSwgMCwgMCwgMSwgMTQ0LjU3MDcyNCwgMTAwNy4wOTk0MjYpIj4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6aW5saW5lO2ZpbGw6I2ZmZDA4NjtmaWxsLW9wYWNpdHk6MTtmaWxsLXJ1bGU6ZXZlbm9kZDtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MXB4O3N0cm9rZS1saW5lY2FwOmJ1dHQ7c3Ryb2tlLWxpbmVqb2luOm1pdGVyO3N0cm9rZS1vcGFjaXR5OjEiIGQ9Ik0gNTg1LjE2OTkyLDUyNC45MTIxMSBDIDQ0Ny40MTc0NSw3MzguODQwMTUgMjg1LjkzMDkzLDc5Ny42ODE0MiA5My4zNDc2NTYsODAxLjAzNzExIDE1OC42NjQ1Miw5MjEuODQwODMgMjg3LjIwMDM4LDEwMDMuMTY5OCA0MzQuMDM5MDYsMTAwMS4yNTU5IDYwNy41Njg1MSw5OTguOTk0NDYgNzUyLjUxMzE3LDg4MS4xODQ4OCA3OTYuNjUyMzQsNzIxLjk2NjggYyAtMi42ODY0LC02LjU3NzY0IC02LjIwMTA2LC0xMy42MjAzNyAtMTAuODE2NCwtMjEuMTM0NzcgQyA3NjguNjg5ODcsNjc3LjAzODc4IDcwOS4xMDQ3OCw1NjguNDc3MjEgNTg1LjE2OTkyLDUyNC45MTIxMSBaIiBpZD0icGF0aC0xMDIiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6aW5saW5lO2ZpbGw6IzI3MGIwYjtmaWxsLW9wYWNpdHk6MC45OTM5MzkzOTtmaWxsLXJ1bGU6ZXZlbm9kZDtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MXB4O3N0cm9rZS1saW5lY2FwOmJ1dHQ7c3Ryb2tlLWxpbmVqb2luOm1pdGVyO3N0cm9rZS1vcGFjaXR5OjEiIGQ9Ik0gNTg2LjI1NCA1MjQuMTE1IEMgNTY0LjY4MiA2NTAuNDUzIDQ3Ny45NzQgNzU0LjQ3MiA0OTguNTg0IDgzNS42MDIgQyA1MjQuNjQ4IDkzOC4xOTkgNDE5LjQ1OCA5NjEuNTE1IDMzMy45NzMgOTg5LjM0MiBDIDM2NS45MjMgOTk3LjU0NSAzOTkuNDc1IDEwMDEuNzA2IDQzNC4wMzkgMTAwMS4yNTYgQyA2MzQuMDUyIDk5OC42NDkgNzk2LjA5MyA4NDIuNTM2IDgwOS41NjEgNjQ2LjQzOSBDIDc5My4xOTcgNjQxLjc3NyA3NzcuNDI1IDYzNC42ODkgNzY2LjcyMyA2MjIuNzQ2IEMgNzM0LjAzNSA1ODYuMjcyIDY1MC4xMTcgNTQ2LjU2NCA1ODYuMjU0IDUyNC4xMTUgWiIgaWQ9InBhdGgtMTAzIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDIwMywgMTM3LCAzKTsgZmlsbC1vcGFjaXR5OiAwLjk5MzkzOTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IiBkPSJNIDU4Ni4yNTQgNTI0LjExNSBDIDU2NC42ODIgNjUwLjQ1MyA0NzcuOTc0IDc1NC40NzIgNDk4LjU4NCA4MzUuNjAyIEMgNTI0LjY0OCA5MzguMTk5IDQxOS40NTggOTYxLjUxNSAzMzMuOTczIDk4OS4zNDIgQyAzNjUuOTIzIDk5Ny41NDUgMzk5LjQ3NSAxMDAxLjcwNiA0MzQuMDM5IDEwMDEuMjU2IEMgNjM0LjA1MiA5OTguNjQ5IDc5Ni4wOTMgODQyLjUzNiA4MDkuNTYxIDY0Ni40MzkgQyA3OTMuMTk3IDY0MS43NzcgNzc3LjQyNSA2MzQuNjg5IDc2Ni43MjMgNjIyLjc0NiBDIDczNC4wMzUgNTg2LjI3MiA2NTAuMTE3IDU0Ni41NjQgNTg2LjI1NCA1MjQuMTE1IFoiIGlkPSJwYXRoLTEwNCIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICA8L2c+CiAgICA8ZyBpZD0iZy0xNSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMC41OyIgdHJhbnNmb3JtPSJtYXRyaXgoMSwgMCwgMCwgMSwgMTQ0LjU3MDcyNCwgMTAwNy4wOTk0MjYpIj4KICAgICAgPHJlY3QgeD0iNTA0LjI2NSIgeT0iNTAwLjI4NyIgd2lkdGg9IjIzLjQ5MiIgaGVpZ2h0PSIyNS41MjciIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojZmFiYjM3O2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTEwNSIvPgogICAgICA8cmVjdCB4PSIzNjkuMTc0IiB5PSI0MTUuNDI5IiB3aWR0aD0iMjIuMzg0IiBoZWlnaHQ9IjI0LjQxOSIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmZmQwODY7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTA2Ii8+CiAgICAgIDxyZWN0IHg9IjQxOC45OTUiIHk9IjQzMy4wMTkiIHdpZHRoPSIzMC41MjQiIGhlaWdodD0iMzAuNTI0IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2ZhYmIzNztmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMDciLz4KICAgICAgPHJlY3QgeD0iNDE3LjM0IiB5PSI2NTIuNTU2IiB3aWR0aD0iNDYuODAzIiBoZWlnaHQ9IjQ2LjgwMyIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmYmMxNGM7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTA4Ii8+CiAgICAgIDxyZWN0IHg9IjQyMi41ODYiIHk9IjQ3NS44OTEiIHdpZHRoPSIzMC41MjQiIGhlaWdodD0iMzAuNTI0IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2ZhYmIzNztmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMDkiLz4KICAgICAgPHJlY3QgeD0iNDcyLjYxOCIgeT0iNjA1LjQ1NyIgd2lkdGg9IjI0LjQxOSIgaGVpZ2h0PSIyNi40NTQiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojY2M4OTAyO2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTExMCIvPgogICAgICA8cmVjdCB4PSI1MjAuNzcyIiB5PSI1NTcuOTAyIiB3aWR0aD0iMTguMzE0IiBoZWlnaHQ9IjE4LjMxNCIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmYWJiMzc7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTExIi8+CiAgICAgIDxyZWN0IHg9IjQ1NC43ODQiIHk9IjU2My4wMjgiIHdpZHRoPSIzMC41MjQiIGhlaWdodD0iMzAuNTI0IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2ZhYmIzNztmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMTIiLz4KICAgICAgPHJlY3QgeD0iMzM1LjM0MiIgeT0iNzIwLjk4NyIgd2lkdGg9IjM4LjY2MyIgaGVpZ2h0PSI0MC42OTgiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojZmJjMTRjO2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTExMyIvPgogICAgICA8cmVjdCB4PSIzNzEuOTciIHk9IjY2MS45NzUiIHdpZHRoPSIyNi40NTQiIGhlaWdodD0iMzAuNTI0IiBzdHlsZT0iZGlzcGxheTppbmxpbmU7b3BhY2l0eToxO2ZpbGw6I2ZiYzE0YztmaWxsLW9wYWNpdHk6MTtzdHJva2U6bm9uZTtzdHJva2Utd2lkdGg6MzQuOTAwMDAxNTM7c3Ryb2tlLW1pdGVybGltaXQ6NDtzdHJva2UtZGFzaGFycmF5Om5vbmU7c3Ryb2tlLWRhc2hvZmZzZXQ6MDtzdHJva2Utb3BhY2l0eTowLjk5NjA3ODQzIiBpZD0icGF0aC0xMTQiLz4KICAgICAgPHJlY3QgeD0iLTQyNy45MyIgeT0iNjEwLjA4NSIgd2lkdGg9IjI0LjUwNyIgaGVpZ2h0PSIyNC40MTkiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojY2M4OTAyO2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTExNSIgdHJhbnNmb3JtPSJzY2FsZSgtMSwxKSIvPgogICAgICA8cmVjdCB4PSI0ODAuNDExIiB5PSI1MjMuNDY5IiB3aWR0aD0iMjAuMzQ5IiBoZWlnaHQ9IjIyLjM4NCIgc3R5bGU9ImRpc3BsYXk6aW5saW5lO29wYWNpdHk6MTtmaWxsOiNmZmQwODY7ZmlsbC1vcGFjaXR5OjE7c3Ryb2tlOm5vbmU7c3Ryb2tlLXdpZHRoOjM0LjkwMDAwMTUzO3N0cm9rZS1taXRlcmxpbWl0OjQ7c3Ryb2tlLWRhc2hhcnJheTpub25lO3N0cm9rZS1kYXNob2Zmc2V0OjA7c3Ryb2tlLW9wYWNpdHk6MC45OTYwNzg0MyIgaWQ9InBhdGgtMTE2Ii8+CiAgICAgIDxyZWN0IHg9Ii00OTkuNSIgeT0iNDY2LjMxNCIgd2lkdGg9IjI0LjUwNyIgaGVpZ2h0PSIyNC40MTkiIHN0eWxlPSJkaXNwbGF5OmlubGluZTtvcGFjaXR5OjE7ZmlsbDojY2M4OTAyO2ZpbGwtb3BhY2l0eToxO3N0cm9rZTpub25lO3N0cm9rZS13aWR0aDozNC45MDAwMDE1MztzdHJva2UtbWl0ZXJsaW1pdDo0O3N0cm9rZS1kYXNoYXJyYXk6bm9uZTtzdHJva2UtZGFzaG9mZnNldDowO3N0cm9rZS1vcGFjaXR5OjAuOTk2MDc4NDMiIGlkPSJwYXRoLTExNyIgdHJhbnNmb3JtPSJzY2FsZSgtMSwxKSIvPgogICAgPC9nPgogICAgPGcgaWQ9ImctMTYiIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IG9wYWNpdHk6IDE7IiB0cmFuc2Zvcm09Im1hdHJpeCgxLjAwMDAwMDA3MDAwMjA0OTUsIDAsIDAsIDEuMDAwMDAwMDcwMDAyMDQ5NSwgMTQ0LjU3MDcyMzkxMDc5NjEyLCAxMDA3LjA5OTQzOTg3MTU5OTQpIj4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogbm9uZTsgZmlsbC1vcGFjaXR5OiAxOyBmaWxsLXJ1bGU6IGV2ZW5vZGQ7IHN0cm9rZTogcmdiKDI1NSwgMTIyLCAwKTsgc3Ryb2tlLXdpZHRoOiAyLjM0ODg4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IiBkPSJNIDU4NS4xNjk5Miw1MjQuOTEyMTEgQyA0NDcuNDE3NDUsNzM4Ljg0MDE1IDI4NS45MzA5Myw3OTcuNjgxNDIgOTMuMzQ3NjU2LDgwMS4wMzcxMSAxNTguNjY0NTIsOTIxLjg0MDgzIDI4Ny4yMDAzOCwxMDAzLjE2OTggNDM0LjAzOTA2LDEwMDEuMjU1OSA2MDcuNTY4NTEsOTk4Ljk5NDQ2IDc1Mi41MTMxNyw4ODEuMTg0ODggNzk2LjY1MjM0LDcyMS45NjY4IGMgLTIuNjg2NCwtNi41Nzc2NCAtNi4yMDEwNiwtMTMuNjIwMzcgLTEwLjgxNjQsLTIxLjEzNDc3IEMgNzY4LjY4OTg3LDY3Ny4wMzg3OCA3MDkuMTA0NzgsNTY4LjQ3NzIxIDU4NS4xNjk5Miw1MjQuOTEyMTEgWiIgaWQ9InBhdGgtMTE4Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IG5vbmU7IGZpbGwtb3BhY2l0eTogMC45OTM5Mzk7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiByZ2IoMjU1LCAxMjIsIDApOyBzdHJva2Utd2lkdGg6IDIuMzQ4ODg7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2Utb3BhY2l0eTogMTsiIGQ9Ik0gNTg2LjI1NCA1MjQuMTE1IEMgNTY0LjY4MiA2NTAuNDUzIDQ3Ny45NzQgNzU0LjQ3MiA0OTguNTg0IDgzNS42MDIgQyA1MjQuNjQ4IDkzOC4xOTkgNDE5LjQ1OCA5NjEuNTE1IDMzMy45NzMgOTg5LjM0MiBDIDM2NS45MjMgOTk3LjU0NSAzOTkuNDc1IDEwMDEuNzA2IDQzNC4wMzkgMTAwMS4yNTYgQyA2MzQuMDUyIDk5OC42NDkgNzk2LjA5MyA4NDIuNTM2IDgwOS41NjEgNjQ2LjQzOSBDIDc5My4xOTcgNjQxLjc3NyA3NzcuNDI1IDYzNC42ODkgNzY2LjcyMyA2MjIuNzQ2IEMgNzM0LjAzNSA1ODYuMjcyIDY1MC4xMTcgNTQ2LjU2NCA1ODYuMjU0IDUyNC4xMTUgWiIgaWQ9InBhdGgtMTE5IiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogbm9uZTsgZmlsbC1vcGFjaXR5OiAwLjk5MzkzOTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IHJnYigyNTUsIDEyMiwgMCk7IHN0cm9rZS13aWR0aDogMi4zNDg4ODsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyIgZD0iTSA1ODYuMjU0IDUyNC4xMTUgQyA1NjQuNjgyIDY1MC40NTMgNDc3Ljk3NCA3NTQuNDcyIDQ5OC41ODQgODM1LjYwMiBDIDUyNC42NDggOTM4LjE5OSA0MTkuNDU4IDk2MS41MTUgMzMzLjk3MyA5ODkuMzQyIEMgMzY1LjkyMyA5OTcuNTQ1IDM5OS40NzUgMTAwMS43MDYgNDM0LjAzOSAxMDAxLjI1NiBDIDYzNC4wNTIgOTk4LjY0OSA3OTYuMDkzIDg0Mi41MzYgODA5LjU2MSA2NDYuNDM5IEMgNzkzLjE5NyA2NDEuNzc3IDc3Ny40MjUgNjM0LjY4OSA3NjYuNzIzIDYyMi43NDYgQyA3MzQuMDM1IDU4Ni4yNzIgNjUwLjExNyA1NDYuNTY0IDU4Ni4yNTQgNTI0LjExNSBaIiBpZD0icGF0aC0xMjAiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgPC9nPgogIDwvZz4KICA8ZyB0cmFuc2Zvcm09Im1hdHJpeCgwLjYwODI2MSwgMCwgMCwgMC42MDgyNjEsIC0yMC4wODQ5NzYsIDMuMjU1NzM2KSI+CiAgICA8Zz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDAuNzg7IiBpZD0icGF0aC0xMCIgZD0iTSA0MTkuMTMzIDg1LjczOCBDIDQxNy4yOTUgODUuNzM5IDQxNS40NTUgODUuNzU0IDQxMy42MTEgODUuNzc4IEMgMzU0Ljg1MiA4Ni41NjEgMjk5LjAyMyA5OS4xNzQgMjQ4LjM1OCAxMjEuMzIgQyA1ODQuMTIxIDEzLjIwNiA3OTYuMTc1IDIxMS4yNiA4MzEuNyA1MDguNzczIEwgODQ2LjM4MyA1MDcuMTI0IEMgODE1LjU4NiAyMzUuODAxIDY1Mi40NzcgODUuNTg4IDQxOS4xMzMgODUuNzM4IFoiIHRyYW5zZm9ybT0ibWF0cml4KDAuOTYzNzMsIDAuMjY2ODc4LCAtMC4yNjY4NzgsIDAuOTYzNzMsIDg5LjUwNTIzMSwgLTEzNi42MTUwNjEpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDE7IiBpZD0icGF0aC0zIiBkPSJNIDE0Ni4wMzcgMTk0LjA2NSBDIDE0NC4yMDEgMTk0LjA2NiAxNDIuMzYxIDE5NC4wODEgMTQwLjUxNyAxOTQuMTA0IEMgODEuNzYgMTk0Ljg4OCAyNS45MzEgMjA3LjUwMyAtMjQuNzM2IDIyOS42NDggQyAzMTEuMDI5IDEyMS41MzEgNTIzLjA4MyAzMTkuNTgzIDU1OC42MDQgNjE3LjA5MyBMIDU3My4yODQgNjE1LjQ0MyBDIDU0Mi40OSAzNDQuMTIyIDM3OS4zODUgMTkzLjkxMyAxNDYuMDM3IDE5NC4wNjUgWiIgdHJhbnNmb3JtPSJtYXRyaXgoLTAuMTgwOTg4LCAtMC45ODM0ODUsIDAuOTgzNDg1LCAtMC4xODA5ODgsIC0zOS4yOTg1ODEsIDcwNS44OTc1NDQpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDAuNDsiIGlkPSJwYXRoLTExIiBkPSJNIDQ4Ny43NDQgMTkzLjA4NCBDIDQ4NS45MDYgMTkzLjA4NSA0ODQuMDY3IDE5My4xIDQ4Mi4yMjMgMTkzLjEyMyBDIDQyMy40NjQgMTkzLjkwNyAzNjcuNjM1IDIwNi41MiAzMTYuOTc1IDIyOC42NjggQyA2NTIuNzI2IDEyMC41NTIgODY0Ljc3NSAzMTguNjA1IDkwMC4yOTkgNjE2LjEwOSBMIDkxNC45NzkgNjE0LjQ2IEMgODg0LjE4NCAzNDMuMTQ0IDcyMS4wODEgMTkyLjkzNSA0ODcuNzQ0IDE5My4wODQgWiIgdHJhbnNmb3JtPSJtYXRyaXgoMC42MTk5OTcsIDAuNzg0NjA1LCAtMC43ODQ2MDUsIDAuNjE5OTk3LCA1MjMuMDY3MjMxLCAtMzQzLjMzMTUzOSkiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgICA8cGF0aCBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBmaWxsOiByZ2IoNjQsIDE3OCwgMjU1KTsgZmlsbC1vcGFjaXR5OiAxOyBmaWxsLXJ1bGU6IGV2ZW5vZGQ7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAxcHg7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2Utb3BhY2l0eTogMTsgb3BhY2l0eTogMC44NDsiIGlkPSJwYXRoLTEyIiBkPSJNIDI5NS45NjkgNDMyLjc1NyBDIDI5NC4xMzEgNDMyLjc1OCAyOTIuMjkzIDQzMi43NzMgMjkwLjQ0OSA0MzIuNzk3IEMgMjMxLjY4OCA0MzMuNTggMTc1Ljg2IDQ0Ni4xOTQgMTI1LjE5OCA0NjguMzQxIEMgNDYwLjk1NyAzNjAuMjI2IDY3My4wMDUgNTU4LjI4IDcwOC41MjYgODU1Ljc4NiBMIDcyMy4yMDcgODU0LjEzOSBDIDY5Mi40MTIgNTgyLjgxOSA1MjkuMzEgNDMyLjYwOSAyOTUuOTY5IDQzMi43NTcgWiIgdHJhbnNmb3JtPSJtYXRyaXgoLTAuOTc4ODQ5LCAwLjIwNDU4NCwgLTAuMjA0NTg0LCAtMC45Nzg4NDksIDk2My44MjA3OTYsIDExMTYuMzY3MjkzKSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxwYXRoIHN0eWxlPSJkaXNwbGF5OiBpbmxpbmU7IGZpbGw6IHJnYig2NCwgMTc4LCAyNTUpOyBmaWxsLW9wYWNpdHk6IDE7IGZpbGwtcnVsZTogZXZlbm9kZDsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDFweDsgc3Ryb2tlLWxpbmVjYXA6IGJ1dHQ7IHN0cm9rZS1saW5lam9pbjogbWl0ZXI7IHN0cm9rZS1vcGFjaXR5OiAxOyBvcGFjaXR5OiAwLjQ7IiBpZD0icGF0aC0xMyIgZD0iTSAxNTguMDM2IDE0OC44MyBDIDE1Ni4yIDE0OC44MzMgMTU0LjM2IDE0OC44NDYgMTUyLjUxNiAxNDguODcyIEMgOTMuNzU5IDE0OS42NTMgMzcuOTMgMTYyLjI2NyAtMTIuNzM0IDE4NC40MTEgQyAzMjMuMDIyIDc2LjMwMSA1MzUuMDc2IDI3NC4zNTIgNTcwLjYwNCA1NzEuODU3IEwgNTg1LjI4NiA1NzAuMjA4IEMgNTU0LjQ4MyAyOTguODkxIDM5MS4zOCAxNDguNjgzIDE1OC4wMzYgMTQ4LjgzIFoiIHRyYW5zZm9ybT0ibWF0cml4KDAuMzM4NjE0LCAtMC45NDA5MjYsIDAuOTQwOTI2LCAwLjMzODYxNCwgLTExNS41OTUxNTgsIDQ4My43MDU5ODQpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDE7IiBpZD0icGF0aC0xNCIgZD0iTSA1MDEuMDE5IDI4OC4yOTIgQyA0OTkuMTgxIDI4OC4yOTMgNDk3LjM0IDI4OC4zMDggNDk1LjQ5NiAyODguMzMxIEMgNDM2LjczNiAyODkuMTE1IDM4MC45MDggMzAxLjcyNiAzMzAuMjQzIDMyMy44NzQgQyA2NjYuMDA0IDIxNS43NTcgODc4LjA2MiA0MTMuODEyIDkxMy41ODYgNzExLjMyMiBMIDkyOC4yNjggNzA5LjY3MyBDIDg5Ny40NjggNDM4LjM1MiA3MzQuMzYxIDI4OC4xNDEgNTAxLjAxOSAyODguMjkyIFoiIHRyYW5zZm9ybT0ibWF0cml4KDAuMTY0NjU2LCAwLjk4NjM1MSwgLTAuOTg2MzUxLCAwLjE2NDY1NiwgOTgyLjg1NzI2NiwgLTIzMy40NTIwNjMpIiBieDpvcmlnaW49IjAuNSAwLjUiLz4KICAgICAgPHBhdGggc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgZmlsbDogcmdiKDY0LCAxNzgsIDI1NSk7IGZpbGwtb3BhY2l0eTogMTsgZmlsbC1ydWxlOiBldmVub2RkOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMXB4OyBzdHJva2UtbGluZWNhcDogYnV0dDsgc3Ryb2tlLWxpbmVqb2luOiBtaXRlcjsgc3Ryb2tlLW9wYWNpdHk6IDE7IG9wYWNpdHk6IDAuNDsiIGlkPSJwYXRoLTE1IiBkPSJNIDE5Mi45NTQgMzc2LjUyMSBDIDE5MS4xMTUgMzc2LjUyMiAxODkuMjc3IDM3Ni41MzcgMTg3LjQzMyAzNzYuNTYxIEMgMTI4LjY3NCAzNzcuMzQ0IDcyLjg0NSAzODkuOTU4IDIyLjE4MSA0MTIuMTAzIEMgMzU3Ljk0IDMwMy45ODkgNTY5Ljk4NyA1MDIuMDQ1IDYwNS41MDUgNzk5LjU1MyBMIDYyMC4xODggNzk3LjkwNSBDIDU4OS4zOTIgNTI2LjU4NCA0MjYuMjk1IDM3Ni4zNjkgMTkyLjk1NCAzNzYuNTIxIFoiIHRyYW5zZm9ybT0ibWF0cml4KC0wLjkzMTQwNCwgLTAuMzYzOTg4LCAwLjM2Mzk4OCwgLTAuOTMxNDA0LCA0MTkuNDk5MDA2LCAxMTgyLjU5OTgwOSkiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgICA8cGF0aCBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBmaWxsOiByZ2IoNjQsIDE3OCwgMjU1KTsgZmlsbC1vcGFjaXR5OiAxOyBmaWxsLXJ1bGU6IGV2ZW5vZGQ7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAxcHg7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2Utb3BhY2l0eTogMTsgb3BhY2l0eTogMTsiIGlkPSJwYXRoLTE2IiBkPSJNIDIyMS42MTQgNjMuOTczIEMgMjE5Ljc3NyA2My45NzYgMjE3LjkzOCA2My45OSAyMTYuMDk0IDY0LjAxNSBDIDE1Ny4zMzQgNjQuNzk1IDEwMS41MDYgNzcuNDA5IDUwLjg0MSA5OS41NTQgQyAzODYuNTk4IC04LjU1OSA1OTguNjUxIDE4OS40OTcgNjM0LjE3MyA0ODcuMDAxIEwgNjQ4Ljg1NCA0ODUuMzU0IEMgNjE4LjA1OCAyMTQuMDM3IDQ1NC45NTQgNjMuODI0IDIyMS42MTQgNjMuOTczIFoiIHRyYW5zZm9ybT0ibWF0cml4KDAuNzUwNDUyLCAtMC42NjA5MjUsIDAuNjYwOTI1LCAwLjc1MDQ1MiwgLTcwLjgwMzMyMiwgMjkwLjkyMDI3MykiIGJ4Om9yaWdpbj0iMC41IDAuNSIvPgogICAgICA8cGF0aCBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBmaWxsOiByZ2IoNjQsIDE3OCwgMjU1KTsgZmlsbC1vcGFjaXR5OiAxOyBmaWxsLXJ1bGU6IGV2ZW5vZGQ7IHN0cm9rZTogbm9uZTsgc3Ryb2tlLXdpZHRoOiAxcHg7IHN0cm9rZS1saW5lY2FwOiBidXR0OyBzdHJva2UtbGluZWpvaW46IG1pdGVyOyBzdHJva2Utb3BhY2l0eTogMTsgb3BhY2l0eTogMC40OyIgaWQ9InBhdGgtMTciIGQ9Ik0gNDQxLjg4OCAzNDIuMDk4IEMgNDQwLjA0NyAzNDIuMDk4IDQzOC4yMDcgMzQyLjExNCA0MzYuMzYyIDM0Mi4xMzggQyAzNzcuNjAxIDM0Mi45MjIgMzIxLjc2NSAzNTUuNTM1IDI3MS4xMDMgMzc3LjY4MSBDIDYwNi44ODYgMjY5LjU2NyA4MTguOTQ0IDQ2Ny42MTUgODU0LjQ2MiA3NjUuMTMzIEwgODY5LjE0MiA3NjMuNDgzIEMgODM4LjM1NCA0OTIuMTU0IDY3NS4yNDIgMzQxLjk0OCA0NDEuODg4IDM0Mi4wOTggWiIgdHJhbnNmb3JtPSJtYXRyaXgoLTAuMzU2NTgyLCAwLjkzNDI2NCwgLTAuOTM0MjY0LCAtMC4zNTY1ODIsIDEyNTYuNzU5NDkzLCAxNjkuMTgyNTUyKSIgYng6b3JpZ2luPSIwLjUgMC41Ii8+CiAgICAgIDxnIHRyYW5zZm9ybT0ibWF0cml4KDAuNzYyNDkzLCAwLCAwLCAwLjc2MjQ5MywgMTkuMjk0NjQ3LCAtNzE2LjMyMjgxNSkiPgogICAgICAgIDxwYXRoIGQ9Ik0gNTY2LjQ2NiAxMDg5LjEzNyBMIDU2Ni40NjYgMTE0OC4yMjIgQyA1NjUuODY3IDExNDguMjE5IDU2NS4yNjcgMTE0OC4yMTcgNTY0LjY2NyAxMTQ4LjIxNyBDIDU2MS40ODEgMTE0OC4yMTcgNTU4LjMxIDExNDguMjU5IDU1NS4xNTQgMTE0OC4zNDEgTCA1NTUuMTU0IDEwODkuMTM3IFogTSA5MDguMjkgMTQ3Ny4xNDYgTCA5NTYuOTg3IDE0NzcuMTQ2IEwgOTU2Ljk4NyAxNDg4LjQ1OCBMIDkwOC41NjkgMTQ4OC40NTggQyA5MDguNTM5IDE0ODQuNjkzIDkwOC40NDcgMTQ4MC45MjIgOTA4LjI5IDE0NzcuMTQ2IFogTSA1NjYuNDY2IDE4MzUuMDEzIEwgNTY2LjQ2NiAxODc2LjQ2NyBMIDU1NS4xNTQgMTg3Ni40NjcgTCA1NTUuMTU0IDE4MzQuODg2IEMgNTU4LjMxOCAxODM0Ljk3MyA1NjEuNDkgMTgzNS4wMTcgNTY0LjY2NyAxODM1LjAxNyBDIDU2NS4yNjcgMTgzNS4wMTcgNTY1Ljg2NyAxODM1LjAxNiA1NjYuNDY2IDE4MzUuMDEzIFogTSAyMjAuNzc1IDE0ODguNDU4IEwgMTY5LjY1NyAxNDg4LjQ1OCBMIDE2OS42NTcgMTQ3Ny4xNDYgTCAyMjEuMDg5IDE0NzcuMTQ2IEMgMjIwLjkyMSAxNDgwLjkwOSAyMjAuODE2IDE0ODQuNjggMjIwLjc3NSAxNDg4LjQ1OCBaIiBzdHlsZT0iZmlsbDogcmdiKDY0LCA0MCwgMCk7IHN0cm9rZTogbm9uZTsiIGJ4Om9yaWdpbj0iMCAwIi8+CiAgICAgICAgPHBhdGggZD0iTSA3NjAuMTIxIDExNDMuMzExIEwgNzE3LjMzNiAxMjE3LjQxNyBDIDcxNS43MiAxMjE2LjUxOCA3MTQuMDk2IDEyMTUuNjMyIDcxMi40NjUgMTIxNC43NjEgTCA3NTUuMzE4IDExNDAuNTM3IFogTSA4MzAuMTIyIDEzMjQuMjAzIEwgOTAwLjQyNCAxMjgzLjYxNCBMIDkwMy4xOTggMTI4OC40MTcgTCA4MzMuMDQgMTMyOC45MjMgQyA4MzIuMDggMTMyNy4zMzkgODMxLjEwNyAxMzI1Ljc2NiA4MzAuMTIyIDEzMjQuMjAzIFogTSA4NDAuNTk5IDE2NDEuMTM3IEwgOTAzLjE5NyAxNjc3LjI3OCBMIDkwMC40MjQgMTY4Mi4wODIgTCA4MzcuOTExIDE2NDUuOTkgQyA4MzguODIxIDE2NDQuMzggODM5LjcxNyAxNjQyLjc2MiA4NDAuNTk5IDE2NDEuMTM3IFogTSA3MjUuMTY4IDE3NjEuODQzIEwgNzYwLjEyMiAxODIyLjM4NSBMIDc1NS4zMTggMTgyNS4xNTggTCA3MjAuMzc5IDE3NjQuNjQxIEMgNzIxLjk4NSAxNzYzLjcyMiA3MjMuNTgyIDE3NjIuNzg5IDcyNS4xNjggMTc2MS44NDMgWiBNIDQwMy41MzggMTc2MC45MzIgTCAzNjYuNDU2IDE4MjUuMTU5IEwgMzYxLjY1MyAxODIyLjM4NiBMIDM5OC44MDIgMTc1OC4wNDIgQyA0MDAuMzcyIDE3NTkuMDE5IDQwMS45NTEgMTc1OS45ODIgNDAzLjUzOCAxNzYwLjkzMiBaIE0gMjg5LjU4IDE2NDIuNjg5IEwgMjIxLjM1IDE2ODIuMDgyIEwgMjE4LjU3NyAxNjc3LjI3OSBMIDI4Ni45NDQgMTYzNy44MDcgQyAyODcuODA3IDE2MzkuNDM3IDI4OC42ODUgMTY0MS4wNjUgMjg5LjU4IDE2NDIuNjg5IFogTSAyOTQuMzU0IDEzMzIuMTY4IEwgMjE4LjU3NiAxMjg4LjQxNyBMIDIyMS4zNSAxMjgzLjYxNCBMIDI5Ny4yMTMgMTMyNy40MTQgQyAyOTYuMjQ2IDEzMjguOTkgMjk1LjI5MyAxMzMwLjU3NSAyOTQuMzU0IDEzMzIuMTY4IFogTSA0MDYuMDc5IDEyMjAuMjU5IEwgMzYxLjY1MyAxMTQzLjMxMSBMIDM2Ni40NTcgMTE0MC41MzcgTCA0MTAuODg5IDEyMTcuNDk1IEMgNDA5LjI3NiAxMjE4LjQwMyA0MDcuNjczIDEyMTkuMzI0IDQwNi4wNzkgMTIyMC4yNTkgWiIgc3R5bGU9ImZpbGw6IHJnYig2NCwgNDAsIDApOyBzdHJva2U6IG5vbmU7IiBieDpvcmlnaW49IjAgMCIvPgogICAgICA8L2c+CiAgICA8L2c+CiAgICA8cmVjdCB4PSI0MjkuMzY3IiB5PSI0MjUuOTc5IiB3aWR0aD0iMjAuMDAzIiBoZWlnaHQ9IjIxLjczNSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTEiIHRyYW5zZm9ybT0ibWF0cml4KDEsIDAsIDAsIDAuOTk5OTk5LCAtMzAxLjM3NzA3NSwgLTIxOS40OTg0MzQpIi8+CiAgICA8cmVjdCB4PSI0NDYuODUzIiB5PSIyNS40NjIiIHdpZHRoPSIyNS45OSIgaGVpZ2h0PSIyNS45OSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTMiLz4KICAgIDxyZWN0IHg9Ii0xMzUuOTU2IiB5PSI2MDUuMTQxIiB3aWR0aD0iMzkuODUxIiBoZWlnaHQ9IjM5Ljg1MSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTQiIHRyYW5zZm9ybT0ibWF0cml4KDAuOTk5OTk5LCAwLCAwLCAxLCA4NDYuMTc2NzU2LCAtMzk0Ljk1NjAyNCkiLz4KICAgIDxyZWN0IHg9IjIyNS43NzkiIHk9IjY4OS44MzYiIHdpZHRoPSIyNS45OSIgaGVpZ2h0PSIyNS45OSIgc3R5bGU9ImRpc3BsYXk6IGlubGluZTsgb3BhY2l0eTogMTsgZmlsbDogcmdiKDgwLCAxNTAsIDIwMCk7IGZpbGwtb3BhY2l0eTogMTsgc3Ryb2tlOiBub25lOyBzdHJva2Utd2lkdGg6IDM0Ljk7IHN0cm9rZS1taXRlcmxpbWl0OiA0OyBzdHJva2UtZGFzaGFycmF5OiBub25lOyBzdHJva2UtZGFzaG9mZnNldDogMDsgc3Ryb2tlLW9wYWNpdHk6IDAuOTk2MDc4OyIgaWQ9InBhdGgtOTUiLz4KICAgIDxyZWN0IHg9IjU5NC4wMTMiIHk9IjczNy4xNDIiIHdpZHRoPSIyMC43OTIiIGhlaWdodD0iMjIuNTI1IiBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBvcGFjaXR5OiAxOyBmaWxsOiByZ2IoODAsIDE1MCwgMjAwKTsgZmlsbC1vcGFjaXR5OiAxOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMzQuOTsgc3Ryb2tlLW1pdGVybGltaXQ6IDQ7IHN0cm9rZS1kYXNoYXJyYXk6IG5vbmU7IHN0cm9rZS1kYXNob2Zmc2V0OiAwOyBzdHJva2Utb3BhY2l0eTogMC45OTYwNzg7IiBpZD0icGF0aC05NiIvPgogICAgPHJlY3QgeD0iMzg3LjIzNSIgeT0iNDc5LjM5NyIgd2lkdGg9IjI1Ljk5IiBoZWlnaHQ9IjI1Ljk5IiBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBvcGFjaXR5OiAxOyBmaWxsOiByZ2IoODAsIDE1MCwgMjAwKTsgZmlsbC1vcGFjaXR5OiAxOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMzQuOTsgc3Ryb2tlLW1pdGVybGltaXQ6IDQ7IHN0cm9rZS1kYXNoYXJyYXk6IG5vbmU7IHN0cm9rZS1kYXNob2Zmc2V0OiAwOyBzdHJva2Utb3BhY2l0eTogMC45OTYwNzg7IiBpZD0icGF0aC05OCIgdHJhbnNmb3JtPSJtYXRyaXgoMS4wMDAwMDEsIDAsIDAsIDAuOTk5OTk5LCAzNjUuMjc0OTY2LCA3OS4yOTUyNDEpIi8+CiAgICA8cmVjdCB4PSI0MjEuOTYyIiB5PSI3NzUuOTM3IiB3aWR0aD0iMzIuOTIiIGhlaWdodD0iMzQuNjUzIiBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBvcGFjaXR5OiAxOyBmaWxsOiByZ2IoODAsIDE1MCwgMjAwKTsgZmlsbC1vcGFjaXR5OiAxOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMzQuOTsgc3Ryb2tlLW1pdGVybGltaXQ6IDQ7IHN0cm9rZS1kYXNoYXJyYXk6IG5vbmU7IHN0cm9rZS1kYXNob2Zmc2V0OiAwOyBzdHJva2Utb3BhY2l0eTogMC45OTYwNzg7IiBpZD0icGF0aC05OSIvPgogICAgPHJlY3QgeD0iLTY2LjkzOSIgeT0iLTU2OC45NzgiIHdpZHRoPSIyMC44NjciIGhlaWdodD0iMjAuNzkyIiBzdHlsZT0iZGlzcGxheTogaW5saW5lOyBvcGFjaXR5OiAxOyBmaWxsOiByZ2IoODAsIDE1MCwgMjAwKTsgZmlsbC1vcGFjaXR5OiAxOyBzdHJva2U6IG5vbmU7IHN0cm9rZS13aWR0aDogMzQuOTsgc3Ryb2tlLW1pdGVybGltaXQ6IDQ7IHN0cm9rZS1kYXNoYXJyYXk6IG5vbmU7IHN0cm9rZS1kYXNob2Zmc2V0OiAwOyBzdHJva2Utb3BhY2l0eTogMC45OTYwNzg7IiBpZD0icGF0aC0xMDEiIHRyYW5zZm9ybT0ibWF0cml4KC0xLCAwLCAwLCAwLjk5OTk5NywgMzEuMzU2OTI4LCA5ODQuNzM2MzA2KSIvPgogIDwvZz4KICA8cGF0aCBkPSJNIDE1NC41MjYgMjMwLjY2IEMgMTQ4LjgwMSAyMzAuNjYgMTQ0LjI3OSAyMzIuNTY4IDE0MC45NjIgMjM2LjM4MyBDIDEzNy42NDggMjQwLjE5NSAxMzUuOTkyIDI0NS40MTcgMTM1Ljk5MiAyNTIuMDQ2IEMgMTM1Ljk5MiAyNTguODYzIDEzNy41OTEgMjY0LjEzIDE0MC43ODUgMjY3Ljg0OSBDIDE0My45ODEgMjcxLjU2NyAxNDguNTM2IDI3My40MjYgMTU0LjQ1MyAyNzMuNDI2IEMgMTU4LjA4NiAyNzMuNDI2IDE2Mi4yMzIgMjcyLjc3MyAxNjYuODkyIDI3MS40NjggTCAxNjYuODkyIDI3Ni43NzggQyAxNjMuMjc5IDI3OC4xMzMgMTU4LjgyNSAyNzguODA4IDE1My41MjggMjc4LjgwOCBDIDE0NS44NTMgMjc4LjgwOCAxMzkuOTI5IDI3Ni40ODIgMTM1Ljc2MSAyNzEuODI3IEMgMTMxLjU5IDI2Ny4xNjcgMTI5LjUwMiAyNjAuNTUgMTI5LjUwMiAyNTEuOTcyIEMgMTI5LjUwMiAyNDYuNjA0IDEzMC41MDcgMjQxLjg5OSAxMzIuNTE5IDIzNy44NjIgQyAxMzQuNTI2IDIzMy44MjMgMTM3LjQyNCAyMzAuNzEgMTQxLjIxIDIyOC41MjQgQyAxNDUuMDAyIDIyNi4zMzkgMTQ5LjQ2MyAyMjUuMjQ3IDE1NC41OTMgMjI1LjI0NyBDIDE2MC4wNiAyMjUuMjQ3IDE2NC44MzUgMjI2LjI0MyAxNjguOTI0IDIyOC4yMzggTCAxNjYuMzU2IDIzMy40MzkgQyAxNjIuNDExIDIzMS41ODYgMTU4LjQ2OSAyMzAuNjYgMTU0LjUyNiAyMzAuNjYgWiBNIDE5My40NjggMjc4LjgwOCBDIDE4Ny42OTIgMjc4LjgwOCAxODMuMTM2IDI3Ny4wNTEgMTc5Ljc5OSAyNzMuNTM1IEMgMTc2LjQ2MiAyNzAuMDIxIDE3NC43OTQgMjY1LjEzNyAxNzQuNzk0IDI1OC44ODkgQyAxNzQuNzk0IDI1Mi41OTIgMTc2LjM0MiAyNDcuNTg4IDE3OS40NCAyNDMuODgzIEMgMTgyLjU0MiAyNDAuMTc2IDE4Ni43MDcgMjM4LjMyMyAxOTEuOTMzIDIzOC4zMjMgQyAxOTYuODI5IDIzOC4zMjMgMjAwLjcwMSAyMzkuOTM0IDIwMy41NTIgMjQzLjE1NCBDIDIwNi40MDMgMjQ2LjM3NCAyMDcuODI4IDI1MC42MjEgMjA3LjgyOCAyNTUuODk1IEwgMjA3LjgyOCAyNTkuNjM2IEwgMTgwLjkyNSAyNTkuNjM2IEMgMTgxLjA0MyAyNjQuMjI0IDE4Mi4yIDI2Ny43MDQgMTg0LjM5OCAyNzAuMDgxIEMgMTg2LjU5NiAyNzIuNDU4IDE4OS42OSAyNzMuNjQ2IDE5My42NzkgMjczLjY0NiBDIDE5Ny44ODUgMjczLjY0NiAyMDIuMDQ0IDI3Mi43NjMgMjA2LjE1NSAyNzEuMDA2IEwgMjA2LjE1NSAyNzYuMjc5IEMgMjA0LjA2MyAyNzcuMTgzIDIwMi4wODQgMjc3LjgzMiAyMDAuMjE5IDI3OC4yMjUgQyAxOTguMzU3IDI3OC42MTUgMTk2LjEwOCAyNzguODA4IDE5My40NjggMjc4LjgwOCBNIDE5MS44NjEgMjQzLjI3NSBDIDE4OC43MjYgMjQzLjI3NSAxODYuMjI3IDI0NC4yOTYgMTg0LjM2MiAyNDYuMzQxIEMgMTgyLjQ5NyAyNDguMzgzIDE4MS4zOTcgMjUxLjIxMiAxODEuMDY1IDI1NC44MjYgTCAyMDEuNDg5IDI1NC44MjYgQyAyMDEuNDg5IDI1MS4wOTYgMjAwLjY1NyAyNDguMjM4IDE5OC45OTEgMjQ2LjI1NiBDIDE5Ny4zMjcgMjQ0LjI2OCAxOTQuOTUyIDI0My4yNzUgMTkxLjg2MSAyNDMuMjc1IFogTSAyNDIuNzU4IDI2Ny40NCBDIDI0Mi43NTggMjcxLjA3OCAyNDEuNDAyIDI3My44ODIgMjM4LjY5NSAyNzUuODUyIEMgMjM1Ljk4NiAyNzcuODI0IDIzMi4xODQgMjc4LjgwOCAyMjcuMjg5IDI3OC44MDggQyAyMjIuMTEgMjc4LjgwOCAyMTguMDcyIDI3Ny45OSAyMTUuMTczIDI3Ni4zNTIgTCAyMTUuMTczIDI3MC44NjYgQyAyMTcuMDUgMjcxLjgxNSAyMTkuMDYzIDI3Mi41NjMgMjIxLjIxMyAyNzMuMTEgQyAyMjMuMzYyIDI3My42NTMgMjI1LjQzNCAyNzMuOTI0IDIyNy40MyAyNzMuOTI0IEMgMjMwLjUxOSAyNzMuOTI0IDIzMi44OTUgMjczLjQzMSAyMzQuNTU4IDI3Mi40NDcgQyAyMzYuMjIxIDI3MS40NjIgMjM3LjA1MiAyNjkuOTU5IDIzNy4wNTIgMjY3LjkzOSBDIDIzNy4wNTIgMjY2LjQxOSAyMzYuMzkyIDI2NS4xMTggMjM1LjA3NiAyNjQuMDM1IEMgMjMzLjc1NyAyNjIuOTU2IDIzMS4xODUgMjYxLjY4IDIyNy4zNjIgMjYwLjIwOSBDIDIyMy43MjUgMjU4Ljg1MyAyMjEuMTQxIDI1Ny42NzMgMjE5LjYwNyAyNTYuNjYyIEMgMjE4LjA3NCAyNTUuNjUyIDIxNi45MzUgMjU0LjUwNiAyMTYuMTg5IDI1My4yMTkgQyAyMTUuNDM3IDI1MS45MzggMjE1LjA2NCAyNTAuNDA3IDIxNS4wNjQgMjQ4LjYyMSBDIDIxNS4wNjQgMjQ1LjQzNyAyMTYuMzYgMjQyLjkyNyAyMTguOTUxIDI0MS4wODUgQyAyMjEuNTM3IDIzOS4yNDUgMjI1LjA4OSAyMzguMzIzIDIyOS42MDcgMjM4LjMyMyBDIDIzMy44MTIgMjM4LjMyMyAyMzcuOTIzIDIzOS4xNzkgMjQxLjkzNyAyNDAuODkyIEwgMjM5LjgzMiAyNDUuNzAxIEMgMjM1LjkxNSAyNDQuMDg0IDIzMi4zNjIgMjQzLjI3NSAyMjkuMTc1IDI0My4yNzUgQyAyMjYuMzczIDI0My4yNzUgMjI0LjI1OCAyNDMuNzE1IDIyMi44MzEgMjQ0LjU5NCBDIDIyMS40MDcgMjQ1LjQ3NSAyMjAuNjk3IDI0Ni42ODggMjIwLjY5NyAyNDguMjMyIEMgMjIwLjY5NyAyNDkuMjc3IDIyMC45NjMgMjUwLjE2OCAyMjEuNDk5IDI1MC45MDIgQyAyMjIuMDM1IDI1MS42MzkgMjIyLjg5NSAyNTIuMzQyIDIyNC4wODUgMjUzLjAwNiBDIDIyNS4yNjcgMjUzLjY3MiAyMjcuNTQ4IDI1NC42MzYgMjMwLjkyNiAyNTUuODk1IEMgMjM1LjU1OCAyNTcuNTgzIDIzOC42ODUgMjU5LjI3OCAyNDAuMzEzIDI2MC45ODYgQyAyNDEuOTQyIDI2Mi42OTkgMjQyLjc1OCAyNjQuODQ5IDI0Mi43NTggMjY3LjQ0IFogTSAyNTcuOTkxIDI3OC4wOTcgTCAyNTIuMDc0IDI3OC4wOTcgTCAyNTIuMDc0IDIzOS4wMzYgTCAyNTcuOTkxIDIzOS4wMzYgTCAyNTcuOTkxIDI3OC4wOTcgTSAyNTEuNTc0IDIyOC40NTEgQyAyNTEuNTc0IDIyNy4wOTcgMjUxLjkwNyAyMjYuMTA2IDI1Mi41NzIgMjI1LjQ3NyBDIDI1My4yMzcgMjI0Ljg0NCAyNTQuMDY3IDIyNC41MjggMjU1LjA2NSAyMjQuNTI4IEMgMjU2LjAxOSAyMjQuNTI4IDI1Ni44NDEgMjI0Ljg1MSAyNTcuNTI5IDIyNS40OTUgQyAyNTguMjE4IDIyNi4xMzUgMjU4LjU2MyAyMjcuMTIxIDI1OC41NjMgMjI4LjQ1MSBDIDI1OC41NjMgMjI5Ljc4MSAyNTguMjE4IDIzMC43NzMgMjU3LjUyOSAyMzEuNDI2IEMgMjU2Ljg0MSAyMzIuMDc5IDI1Ni4wMTkgMjMyLjQwNSAyNTUuMDY1IDIzMi40MDUgQyAyNTQuMDY3IDIzMi40MDUgMjUzLjIzNyAyMzIuMDc5IDI1Mi41NzIgMjMxLjQyNiBDIDI1MS45MDcgMjMwLjc3MyAyNTEuNTc0IDIyOS43ODEgMjUxLjU3NCAyMjguNDUxIFogTSAyNzUuOTMzIDIzOS4wMzYgTCAyNzUuOTMzIDI2NC4zNzUgQyAyNzUuOTMzIDI2Ny41NTggMjc2LjY1OCAyNjkuOTM1IDI3OC4xMDkgMjcxLjUwNCBDIDI3OS41NTggMjczLjA3MyAyODEuODI4IDI3My44NTggMjg0LjkxOCAyNzMuODU4IEMgMjg5LjAwNCAyNzMuODU4IDI5MS45OTEgMjcyLjc0MSAyOTMuODgxIDI3MC41MDcgQyAyOTUuNzY5IDI2OC4yNzMgMjk2LjcxMiAyNjQuNjI1IDI5Ni43MTIgMjU5LjU2NCBMIDI5Ni43MTIgMjM5LjAzNiBMIDMwMi42MyAyMzkuMDM2IEwgMzAyLjYzIDI3OC4wOTcgTCAyOTcuNzQ1IDI3OC4wOTcgTCAyOTYuODkzIDI3Mi44NjEgTCAyOTYuNTcgMjcyLjg2MSBDIDI5NS4zNTkgMjc0Ljc4MiAyOTMuNjc5IDI3Ni4yNTQgMjkxLjUyOSAyNzcuMjc3IEMgMjg5LjM4IDI3OC4yOTggMjg2LjkyNiAyNzguODA4IDI4NC4xNjkgMjc4LjgwOCBDIDI3OS40MTcgMjc4LjgwOCAyNzUuODU4IDI3Ny42ODIgMjczLjQ5MyAyNzUuNDI4IEMgMjcxLjEzIDI3My4xNjkgMjY5Ljk0NyAyNjkuNTU2IDI2OS45NDcgMjY0LjU4OCBMIDI2OS45NDcgMjM5LjAzNiBMIDI3NS45MzMgMjM5LjAzNiBaIE0gMzY0LjU3MyAyNzguMDk3IEwgMzY0LjU3MyAyNTIuNjg1IEMgMzY0LjU3MyAyNDkuNTczIDM2My45MDkgMjQ3LjI0MSAzNjIuNTc4IDI0NS42ODIgQyAzNjEuMjQ4IDI0NC4xMjYgMzU5LjE4IDI0My4zNDcgMzU2LjM3NCAyNDMuMzQ3IEMgMzUyLjY5MiAyNDMuMzQ3IDM0OS45NzEgMjQ0LjQwNiAzNDguMjExIDI0Ni41MjIgQyAzNDYuNDU1IDI0OC42MzYgMzQ1LjU3NyAyNTEuODkxIDM0NS41NzcgMjU2LjI4NiBMIDM0NS41NzcgMjc4LjA5NyBMIDMzOS42NiAyNzguMDk3IEwgMzM5LjY2IDI1Mi42ODUgQyAzMzkuNjYgMjQ5LjU3MyAzMzguOTkzIDI0Ny4yNDEgMzM3LjY2NCAyNDUuNjgyIEMgMzM2LjMzNSAyNDQuMTI2IDMzNC4yNTUgMjQzLjM0NyAzMzEuNDI5IDI0My4zNDcgQyAzMjcuNzIyIDI0My4zNDcgMzI1LjAwOCAyNDQuNDU4IDMyMy4yODQgMjQ2LjY4MSBDIDMyMS41NiAyNDguOTAzIDMyMC43IDI1Mi41NDYgMzIwLjcgMjU3LjYwNiBMIDMyMC43IDI3OC4wOTcgTCAzMTQuNzgxIDI3OC4wOTcgTCAzMTQuNzgxIDIzOS4wMzYgTCAzMTkuNTkzIDIzOS4wMzYgTCAzMjAuNTU0IDI0NC4zODIgTCAzMjAuODQgMjQ0LjM4MiBDIDMyMS45NTggMjQyLjQ3OSAzMjMuNTM1IDI0MC45OTcgMzI1LjU2NiAyMzkuOTMgQyAzMjcuNTk4IDIzOC44NiAzMjkuODcyIDIzOC4zMjMgMzMyLjM5MSAyMzguMzIzIEMgMzM4LjQ5NyAyMzguMzIzIDM0Mi40OSAyNDAuNTMzIDM0NC4zNjcgMjQ0Ljk1NCBMIDM0NC42NDcgMjQ0Ljk1NCBDIDM0NS44MTUgMjQyLjkwOSAzNDcuNTAxIDI0MS4yOTQgMzQ5LjcwNyAyNDAuMTA2IEMgMzUxLjkxOCAyMzguOTE4IDM1NC40MzggMjM4LjMyMyAzNTcuMjY5IDIzOC4zMjMgQyAzNjEuNjg0IDIzOC4zMjMgMzY0Ljk5MSAyMzkuNDYgMzY3LjE4OSAyNDEuNzMxIEMgMzY5LjM4NyAyNDMuOTk3IDM3MC40ODUgMjQ3LjYyNSAzNzAuNDg1IDI1Mi42MTggTCAzNzAuNDg1IDI3OC4wOTcgTCAzNjQuNTczIDI3OC4wOTcgWiIgc3R5bGU9InRleHQtdHJhbnNmb3JtOiBub25lOyBmaWxsOiByZ2IoNjQsIDQwLCAwKTsgaXNvbGF0aW9uOiBhdXRvOyBvcGFjaXR5OiAxOyIgYng6b3JpZ2luPSIwLjUgMC41Ii8+Cjwvc3ZnPg==); - background-size: 200px 200px; } - } ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/event_item.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/event_item.st deleted file mode 100644 index 9fdd774d40b7996695d7dd737d86957b09ba73f7..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/event_item.st +++ /dev/null @@ -1,6 +0,0 @@ -event_item(e) ::= << - <div class="item"> - <h3>$e.description$</h3> - <h4 class="gray">$e.time; format="short"$</h4> - </div> ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html.st deleted file mode 100644 index 443bbb59028ce00347a89e06254956dab3b9b3cc..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html.st +++ /dev/null @@ -1,19 +0,0 @@ -html(content, useCss) ::= << - <!DOCTYPE html> - <head> - <meta charset="UTF-8"> - <title>Cesium+</title> - $if(useCss)$ - <style> - $css()$ - $css_logo()$ - </style> - $endif$ - </head> - <body class="platform-browser platform-linux platform-ready"> - <ion-content> - $content$ - </ion-content> - </body> -</html> ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_email_content.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_email_content.st deleted file mode 100644 index c6fa7ed4cb0dd2c8cb62918f4f474e4ec61fa800..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_email_content.st +++ /dev/null @@ -1,76 +0,0 @@ -html_email_content(issuerPubkey, issuerName, senderPubkey, senderName, events, url) ::= << -<table cellspacing="0" cellpadding="0" width="100%" - style="font-size:12px;font-family:Helvetica Neue,Helvetica,Lucida Grande,tahoma,verdana,arial,sans-serif;border-spacing:0px;border-collapse:collapse;max-width:600px!important;"> - <tr> - <td> - <div style="background:#1a237e;width:100%;text-align:center;border-radius:4px;min-height:35px;"> - - $cesium_logo(url, true)$ - - <p style="margin:0px;padding:8px 0px;text-align:center;color:white;font-size:14px;"> - $i18n_args("duniter4j.es.subscription.email.html.hello", issuerName)$ - </p> - </div> - </td> - </tr> - - <tr> - <td> - <table cellspacing="0" cellpadding="0" width="100%" > - <tr> - <td> - <p style="margin:0px;padding:16px;font-size: 12px;"> - $i18n_args("duniter4j.es.subscription.email.html.unreadCount", {$length(events)$} )$ - $if(issuerPubkey)$ - <br/> - <span style="font-size:12px;color:grey !important;"> - $i18n_args("duniter4j.es.subscription.email.html.pubkey", [{$[url, "/#/app/wot/", issuerPubkey, "/"]; separator=""$}, {$issuerPubkey; format="pubkey"$}])$ - </span> - $endif$ - </p> - - </td> - <td> - <p style="margin:0px;width:100%;text-align:right;min-height: 64px;padding: 16px 0px;"> - <a style="overflow:hidden!important;background-color:#387ef5;border-color:transparent;border-radius:2px;border-shadow: 2px 2px rgba(50,50,50,0.32);box-sizing: border-box;color:white;display:inline-block;font-size:14px;font-weight: 500;height: 47px;letter-spacing: 0.5px;line-height:42px;margin:0;min-height:47px;min-width:52px;padding-bottom:0px;padding-left:24px;padding-right:24px;padding-top:0px;text-align:center;text-decoration:none;text-transform:uppercase;" - href="$url$">$i18n("duniter4j.es.subscription.email.openCesium")$ >></a> - </p> - </td> - </tr> - </table> - </td> - </tr> - - <tr> - <td> - <div style="background-color:#f5f5f5;border: 0;box-sizing: border-box; color: rgba(0, 0, 0, 0.54);font-size: 14px;font-weight: 700;height: 48px;line-height: 48px;min-height: 48px;padding-bottom: 8px;padding-left: 16px;padding-right: 16px;padding-top: 8px;vertical-align: baseline;"> - $i18n("duniter4j.es.subscription.email.notificationsDivider")$ - </div> - </td> - </tr> - - $events:{e|$html_event_item(e)$}$ - - <tr> - <td> - <div style="width:100%;text-align:center;min-height:32px;padding:8px;"> - - </div> - </td> - </tr> - - <tr> - <td> - <div style="background-color: rgb(236, 240, 247) !important;border-color: rgb(221, 223, 226) !important;width:100%;text-align:center;border-radius:4px;"> - <p style="margin:0px;padding:8px 0px;text-align:center;color:grey !important;text-decoration:none !important;"> - $i18n_args("duniter4j.es.subscription.email.html.footer.sendBy", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ - <br/> - <small> - $i18n_args("duniter4j.es.subscription.email.html.footer.disableHelp", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ - </small> - </p> - </div> - </td> - </tr> -</table> ->> diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_event_item.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_event_item.st deleted file mode 100644 index 8515d55e9f785e6e3a42228c23b256fd18ef325e..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/html_event_item.st +++ /dev/null @@ -1,10 +0,0 @@ -html_event_item(e) ::= << - <tr> - <td> - <div style="border-bottom: solid 1px #ccc !important;color: rgb(68, 68, 68);display: block;font-size: 14px;font-weight: 400;line-height: 20px;margin-bottom: -1px;margin-left: -1px;margin-right: -1px;margin-top: -1px;padding-bottom: 16px;padding-left: 16px;padding-right: 16px;padding-top: 16px;white-space: normal;"> - <h3 style="color: rgb(0,0,0);font-family:-apple-system,Helvetica Neue,Helvetica,Roboto,Segoe UI,sans-serif;font-size: 14px;font-synthesis: weight style;font-weight: 500;line-height: 16.8px;margin-bottom: 4px;margin-left: 0px;margin-right: 0px;margin-top: 0px;padding: 0;">$e.description$</h3> - <h4 style="color: grey !important;font-size: 12px;font-weight: 500;line-height: 14.4px;margin: 0;padding: 0;">$e.time; format="short"$</h4> - </div> - </td> - </tr> ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n.st deleted file mode 100644 index d5d042c66c9f715169af75d7d19936237ecc45ce..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n.st +++ /dev/null @@ -1,2 +0,0 @@ -i18n(key) ::= "$key; format=\"i18n\"$" - diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n_args.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n_args.st deleted file mode 100644 index 120d7753ab0370967f43ec23feb1afc2da84e3dd..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/i18n_args.st +++ /dev/null @@ -1,2 +0,0 @@ -i18n_args(key, params) ::= "$key; format={i18n:$params; separator=\",\"$}$" - diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_email.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_email.st deleted file mode 100644 index 004367d99c9518604cc2c57fa1d864fb0bc37690..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_email.st +++ /dev/null @@ -1,15 +0,0 @@ -text_email(issuerPubkey, issuerName, senderPubkey, senderName, events, url) ::= << -$i18n_args("duniter4j.es.subscription.email.hello", issuerName)$ -$i18n_args("duniter4j.es.subscription.email.unreadCount", {$length(events)$} )$ - -$i18n("duniter4j.es.subscription.email.notificationsDivider")$ -$events:{e|$text_event_item(e)$}$ - -$i18n("duniter4j.es.subscription.email.openCesium")$ : $url$ -$if(issuerPubkey)$$i18n_args("duniter4j.es.subscription.email.pubkey", [{$[url, "/#/app/wot/", issuerPubkey, "/"]; separator=""$}, {$issuerPubkey; format="pubkey"$}])$$endif$ - ------------------------------------------------ -$i18n_args("duniter4j.es.subscription.email.footer.sendBy", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ -$i18n_args("duniter4j.es.subscription.email.footer.disableHelp", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ - ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_event_item.st b/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_event_item.st deleted file mode 100644 index df9c16e49beda7258a3423b83ccc2a483d3a6771..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/org/duniter/elasticsearch/subscription/templates/text_event_item.st +++ /dev/null @@ -1,4 +0,0 @@ -text_event_item(e) ::= << - - [$e.time; format="short"$] $e.description$ - ->> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/plugin-security.policy b/duniter4j-es-subscription/src/main/resources/plugin-security.policy deleted file mode 100644 index ea5437f6726c44547a1dce7470385fc69c055e22..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/plugin-security.policy +++ /dev/null @@ -1,5 +0,0 @@ -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"; -}; \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/subscription-categories-bulk-insert.json b/duniter4j-es-subscription/src/main/resources/subscription-categories-bulk-insert.json deleted file mode 100644 index 8ac3e0669dbfb78a531796459237628b3f0ee5f7..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/main/resources/subscription-categories-bulk-insert.json +++ /dev/null @@ -1,4 +0,0 @@ -{ "index": { "_id": "_notification"}} -{ "name": "Notifications" , "parent": null} -{ "index": { "_id": "email"}} -{ "name": "Email", "parent": "_notification"} diff --git a/duniter4j-es-subscription/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-subscription/src/test/es-home/config/elasticsearch.yml deleted file mode 100644 index 41a32f793d417e43795d0b1f278c6448b04015e2..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/es-home/config/elasticsearch.yml +++ /dev/null @@ -1,193 +0,0 @@ -# ======================== Elasticsearch Configuration ========================= -# -# NOTE: Elasticsearch comes with reasonable defaults for most settings. -# Before you set out to tweak and tune the configuration, make sure you -# understand what are you trying to accomplish and the consequences. -# -# The primary way of configuring a node is via this file. This template lists -# the most important settings you may want to configure for a production cluster. -# -# Please see the documentation for further information on configuration options: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/setup-configuration.html> -# -# ---------------------------------- Cluster ----------------------------------- -# -# Use a descriptive name for your cluster: -# -# cluster.name: my-application -cluster.name: duniter4j-es-subscription-TEST -# -# ------------------------------------ Node ------------------------------------ -# -# Use a descriptive name for the node: -# -# node.name: node-1 -# -# Add custom attributes to the node: -# -# node.rack: r1 -# -# ----------------------------------- Paths ------------------------------------ -# -# Path to directory where to store the data (separate multiple locations by comma): -# -# path.data: /path/to/data -# -# Path to log files: -# -# path.logs: /path/to/logs -# -# ----------------------------------- Memory ----------------------------------- -# -# Lock the memory on startup: -# -# bootstrap.mlockall: true -# -# Make sure that the `ES_HEAP_SIZE` environment variable is set to about half the memory -# available on the system and that the owner of the process is allowed to use this limit. -# -# Elasticsearch performs poorly when the system is swapping the memory. -# -# ---------------------------------- Network ----------------------------------- -# -# Set the bind address to a specific IP (IPv4 or IPv6): -# -# network.host: 192.168.233.1 -# -# Set a custom port for HTTP: -# -# http.port: 9200-9300 - -http.cors.allow-origin: "/.*/" -http.cors.enabled: true - -# Internal transport layer -# -# transport.tcp.port: 9210-9220 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-network.html> -# -# --------------------------------- Discovery ---------------------------------- -# -# Pass an initial list of hosts to perform discovery when new node is started: -# The default list of hosts is ["127.0.0.1", "[::1]"] -# -# discovery.zen.ping.unicast.hosts: ["host1", "host2"] -#discovery.zen.ping.unicast.hosts: ["127.0.0.1", ""] -# -# Prevent the "split brain" by configuring the majority of nodes (total number of nodes / 2 + 1): -# -# discovery.zen.minimum_master_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-discovery.html> -# -# ---------------------------------- Gateway ----------------------------------- -# -# Block initial recovery after a full cluster restart until N nodes are started: -# -# gateway.recover_after_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-gateway.html> -# -# ---------------------------------- Various ----------------------------------- -# -# Disable starting multiple nodes on a single system: -# -# node.max_local_storage_nodes: 1 -# -# Require explicit names when deleting indices: -# -# rest.destructive_requires_name: true - -security.manager.enabled: false - -# -# ---------------------------------- Duniter4j --------------------------------- -# -# Disbale duniter4j plugin -# -# duniter.enabled: false -# -# Reset and reload all Duniter4j data at startup - DO SET to true in production -# -# duniter.indices.reload: true -# -# Default string analyzer -# -duniter.string.analyzer: french -# -# Enabling node blockchain synchronization -# -duniter.blockchain.enable: false -# -# Duniter node to synchronize -# -duniter.host: g1-test.duniter.org -duniter.port: 10900 -# -# ---------------------------------- 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 -# -# Security token prefix (default: 'duniter-') -# -# duniter.auth.token.prefix: duniter- -# -# Token validity duration, in seconds (default: 600) -# -# duniter.auth.tokenValidityDuration: 3600 # = 1hour -# -# ---------------------------------- Duniter4j P2P sync ------------------------- -# -# Should synchronize data using P2P -# -duniter.p2p.enable: false -# -# ---------------------------------- Duniter4j SMTP server ---------------------- -# -# SMTP server configuration (host and port) -# -duniter.mail.enable: true -#duniter.mail.smtp.host: localhost -#duniter.mail.smtp.port: 25 -#duniter.mail.smtp.host: smtp.gmail.com -#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 -#duniter.mail.admin: benoit.lavenier@e-is.pro -# -# Mail subject prefix -# -#duniter.mail.subject.prefix: [Duniter4j ES] - -# ---------------------------------- Duniter4j Websocket server ---------------------- -# -# Websocket port (usefull for listen changes) -# -duniter.ws.port: 9400-9410 - -# ---------------------------------- Duniter4j Subscription services ------------ -# -# Enable subscription services ? (default: true) -# -duniter.subscription.enable: true -# -# Time interval (millisecond) to send email ? (default: 3600000 = 1h) -# -duniter.subscription.email.interval: 10000 \ No newline at end of file diff --git a/duniter4j-es-subscription/src/test/es-home/config/logging.yml b/duniter4j-es-subscription/src/test/es-home/config/logging.yml deleted file mode 100644 index e3ec67452e065dee23b0cf42769676c0ac5baa20..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/es-home/config/logging.yml +++ /dev/null @@ -1,105 +0,0 @@ -# you can override this using by setting a system property, for example -Des.logger.level=DEBUG -es.logger.level: INFO -rootLogger: ${es.logger.level}, console, file -logger: - # log rest execution errors for easier debugging - action: DEBUG - - # deprecation logging, turn to DEBUG to see them - deprecation: INFO, deprecation_log_file - - # reduce the logging for aws, too much is logged under the default INFO - com.amazonaws: WARN - # aws will try to do some sketchy JMX stuff, but its not needed. - com.amazonaws.jmx.SdkMBeanRegistrySupport: ERROR - com.amazonaws.metrics.AwsSdkMetrics: ERROR - - org.apache.http: INFO - - org.duniter: INFO - #org.duniter.elasticsearch: DEBUG - - duniter : INFO - #duniter.network.p2p: TRACE - - security: INFO - cluster.metadata: ERROR - cluster.routing.allocation: ERROR - - org.nuiton.i18n: WARN - org.nuiton.config: WARN - org.nuiton.i18n: ERROR - org.nuiton.config: ERROR - org.nuiton.converter: WARN - org.apache.http: WARN - org.apache.http.client: ERROR - org.glassfish.grizzly: WARN - org.glassfish.tyrus: WARN - - # gateway - #gateway: DEBUG - #index.gateway: DEBUG - - # peer shard recovery - #indices.recovery: DEBUG - - # discovery - #discovery: TRACE - - index.search.slowlog: TRACE, index_search_slow_log_file - index.indexing.slowlog: TRACE, index_indexing_slow_log_file - -additivity: - index.search.slowlog: false - index.indexing.slowlog: false - deprecation: false - -appender: - console: - type: console - layout: - type: consolePattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %.10000m%n" - - # Use the following log4j-extras RollingFileAppender to enable gzip compression of log files. - # For more information see https://logging.apache.org/log4j/extras/apidocs/org/apache/log4j/rolling/RollingFileAppender.html - #file: - #type: extrasRollingFile - #file: ${path.logs}/${cluster.name}.log - #rollingPolicy: timeBased - #rollingPolicy.FileNamePattern: ${path.logs}/${cluster.name}.log.%d{yyyy-MM-dd}.gz - #layout: - #type: pattern - #conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - deprecation_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_deprecation.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_search_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_search_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_indexing_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_indexing_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" diff --git a/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties b/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties deleted file mode 100644 index 49bd43a2e29da14384b67a9b51b727354fe90521..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties +++ /dev/null @@ -1,80 +0,0 @@ -# Elasticsearch plugin descriptor file -# This file must exist as 'plugin-descriptor.properties' at -# the root directory of all plugins. -# -# A plugin can be 'site', 'jvm', or both. -# -### example site plugin for "foo": -# -# foo.zip <-- zip file for the plugin, with this structure: -# _site/ <-- the contents that will be served -# plugin-descriptor.properties <-- example contents below: -# -# site=true -# description=My cool plugin -# version=1.0 -# -### example jvm plugin for "foo" -# -# foo.zip <-- zip file for the plugin, with this structure: -# <arbitrary name1>.jar <-- classes, resources, dependencies -# <arbitrary nameN>.jar <-- any number of jars -# plugin-descriptor.properties <-- example contents below: -# -# jvm=true -# classname=foo.bar.BazPlugin -# description=My cool plugin -# version=2.0.0-rc1 -# elasticsearch.version=2.0 -# java.version=1.7 -# -### mandatory elements for all plugins: -# -# 'description': simple summary of the plugin -description=The mapper attachments plugin adds the attachment type to Elasticsearch using Apache Tika. -# -# 'version': plugin's version -version=2.3.3 -# -# 'name': the plugin name -name=mapper-attachments - -### mandatory elements for site plugins: -# -# 'site': set to true to indicate contents of the _site/ -# directory in the root of the plugin should be served. -site=false -# -### mandatory elements for jvm plugins : -# -# 'jvm': true if the 'classname' class should be loaded -# from jar files in the root directory of the plugin. -# Note that only jar files in the root directory are -# added to the classpath for the plugin! If you need -# other resources, package them into a resources jar. -jvm=true -# -# 'classname': the name of the class to load, fully-qualified. -classname=org.elasticsearch.mapper.attachments.MapperAttachmentsPlugin -# -# 'java.version' version of java the code is built against -# use the system property java.specification.version -# version string must be a sequence of nonnegative decimal integers -# separated by "."'s and may have leading zeros -java.version=1.7 -# -# 'elasticsearch.version' version of elasticsearch compiled against -# You will have to release a new version of the plugin for each new -# elasticsearch release. This version is checked when the plugin -# is loaded so Elasticsearch will refuse to start in the presence of -# plugins with the incorrect elasticsearch.version. -elasticsearch.version=2.4.6 -# -### deprecated elements for jvm plugins : -# -# 'isolated': true if the plugin should have its own classloader. -# passing false is deprecated, and only intended to support plugins -# that have hard dependencies against each other. If this is -# not specified, then the plugin is isolated by default. -isolated=true -# diff --git a/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-security.policy b/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-security.policy deleted file mode 100644 index 3264766682269e55baae64ca70d1170101260abf..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-security.policy +++ /dev/null @@ -1,34 +0,0 @@ -/* - * Licensed to Elasticsearch under one or more contributor - * license agreements. See the NOTICE file distributed with - * this work for additional information regarding copyright - * ownership. Elasticsearch licenses this file to you under - * the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - -// NOTE: when modifying this file, look at restrictions in TikaImpl too -grant { - // needed to apply additional sandboxing to tika parsing - permission java.security.SecurityPermission "createAccessControlContext"; - - // TODO: fix PDFBox not to actually install bouncy castle like this - permission java.security.SecurityPermission "putProviderProperty.BC"; - permission java.security.SecurityPermission "insertProvider"; - // needed only on java 7 - permission java.security.SecurityPermission "insertProvider.BC"; - // TODO: fix POI XWPF to not do this: https://bz.apache.org/bugzilla/show_bug.cgi?id=58597 - permission java.lang.reflect.ReflectPermission "suppressAccessChecks"; - // needed by xmlbeans, as part of POI for MS xml docs - permission java.lang.RuntimePermission "getClassLoader"; -}; diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestFixtures.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestFixtures.java deleted file mode 100644 index 5546a2ab6ead810fb80babb6957500f0165589ad..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestFixtures.java +++ /dev/null @@ -1,31 +0,0 @@ -package org.duniter.elasticsearch.subscription; - -/* - * #%L - * Duniter4j :: ElasticSearch Indexer - * %% - * 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% - */ - - -public class TestFixtures extends org.duniter.core.test.TestFixtures { - - public String getEmail() { - return "contact@e-is.pro"; - } -} diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestResource.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestResource.java deleted file mode 100644 index 787de591246f140857d0bb9a22ab7c515191f85a..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestResource.java +++ /dev/null @@ -1,109 +0,0 @@ -package org.duniter.elasticsearch.subscription; - -/* - * #%L - * Duniter4j :: 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 org.apache.commons.io.FileUtils; -import org.elasticsearch.bootstrap.Elasticsearch; -import org.junit.runner.Description; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.File; - -public class TestResource extends org.duniter.core.test.TestResource { - - private static final Logger log = LoggerFactory.getLogger(TestResource.class); - - - public static TestResource create() { - return new TestResource(null, true); - } - - public static TestResource createNotStartEs() { - return new TestResource(null, false); - } - - public static TestResource create(boolean startES) { - return new TestResource(null, startES); - } - - public static TestResource create(String configName) { - return new TestResource(configName, true); - } - - public static TestResource create(String configName, boolean startES) { - return new TestResource(configName, startES); - } - - private TestFixtures fixtures = new TestFixtures(); - private final boolean startESNode; - - protected TestResource(String configName, boolean startESNode) { - super(configName); - this.startESNode = startESNode; - } - - public TestFixtures getFixtures() { - return fixtures; - } - - public PluginSettings getPluginSettings() { - return PluginSettings.instance(); - } - - protected void before(Description description) throws Throwable { - super.before(description); - - // Prepare ES home - File esHomeDir = getResourceDirectory("es-home"); - - System.setProperty("es.path.home", esHomeDir.getCanonicalPath()); - - FileUtils.copyDirectory(new File("src/test/es-home"), esHomeDir); - FileUtils.copyDirectory(new File("target/classes"), new File(esHomeDir, "plugins/duniter4j-es-subscription")); - - // Copy dependencies plugins - FileUtils.copyDirectory(new File("../duniter4j-es-core/target/classes"), new File(esHomeDir, "plugins/duniter4j-es-core")); - FileUtils.copyDirectory(new File("../duniter4j-es-user/target/classes"), new File(esHomeDir, "plugins/duniter4j-es-user")); - - if (startESNode) { - Elasticsearch.main(new String[]{"start"}); - } - - /*while(true) { - Thread.sleep(10000); - }*/ - } - - /** - * Return configuration files prefix (i.e. 'allegro-test') - * Could be override by external project - * - * @return the prefix to use to retrieve configuration files - */ - protected String getConfigFilesPrefix() { - return "duniter4j-es-subscription-test"; - } - -} diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java deleted file mode 100644 index 72aa10fdc6743b77a46cd3265cb1429bf93202bb..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java +++ /dev/null @@ -1,182 +0,0 @@ -package org.duniter.elasticsearch.subscription.service; - -/* - * #%L - * Duniter4j :: ElasticSearch Indexer - * %% - * 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.core.JsonProcessingException; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.google.common.collect.ImmutableList; -import org.duniter.core.client.model.ModelUtils; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.local.Wallet; -import org.duniter.core.client.service.ServiceLocator; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.StringUtils; -import org.duniter.core.util.crypto.CryptoUtils; -import org.duniter.core.util.json.JsonAttributeParser; -import org.duniter.core.util.url.URLs; -import org.duniter.elasticsearch.subscription.TestResource; -import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.model.UserEventCodes; -import org.duniter.elasticsearch.user.service.UserEventService; -import org.junit.*; -import org.nuiton.i18n.I18n; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.stringtemplate.v4.ST; -import org.stringtemplate.v4.STGroupFile; - -import java.util.Date; -import java.util.List; -import java.util.Locale; -import java.util.Map; - -/** - * Created by Benoit on 06/05/2015. - */ -public class SubscriptionServiceTest { - private static final Logger log = LoggerFactory.getLogger(SubscriptionServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private SubscriptionService service; - private UserEventService userEventService; - private CryptoService cryptoService; - - @Before - public void setUp() throws Exception { - service = ServiceLocator.instance().getBean(SubscriptionService.class); - cryptoService = ServiceLocator.instance().getCryptoService(); - userEventService = ServiceLocator.instance().getBean(UserEventService.class); - } - - @Test - public void create() throws JsonProcessingException { - Wallet wallet = createTestWallet(); - - createAndIndexSubscription(wallet); - - } - - @Test - public void executeEmailSubscriptions() throws Exception{ - Wallet wallet = createTestWallet(); - try { - createAndIndexSubscription(wallet); - } catch(Exception e) { - Assume.assumeNoException(e); - } - - userEventService.indexEvent(Locale.getDefault(), - UserEvent.newBuilder( - UserEvent.EventType.INFO, - UserEventCodes.MEMBER_JOIN.name()) - .setRecipient(wallet.getPubKeyHash()) - .build()) - .get(); - - // wait 10s - Thread.sleep(10000); - - service.executeEmailSubscriptions(EmailSubscription.Frequency.daily); - - // wait 10s - Thread.sleep(10000); - } - - @Test - @Ignore - public void startNode() throws Exception { - - while(true) { - Thread.sleep(10000); - } - } - - /* -- internal methods -- */ - - protected Wallet createTestWallet() { - Wallet wallet = new Wallet( - resource.getFixtures().getCurrency(), - resource.getFixtures().getUid(), - CryptoUtils.decodeBase58(resource.getFixtures().getUserPublicKey()), - CryptoUtils.decodeBase58(resource.getFixtures().getUserSecretKey())); - - return wallet; - } - - protected EmailSubscription createAndIndexSubscription(Wallet wallet) throws JsonProcessingException { - - EmailSubscription subscription = createEmailSubscription(wallet); - - // Compute full JSON (with hash + signature) - String json = JacksonUtils.getThreadObjectMapper().writeValueAsString(subscription); - - String id = service.create(json); - Assert.assertNotNull(id); - - subscription.setId(id); - return subscription; - } - - protected EmailSubscription createEmailSubscription(Wallet wallet) throws JsonProcessingException { - - ObjectMapper objectMapper = JacksonUtils.getThreadObjectMapper(); - - EmailSubscription subscription = new EmailSubscription(); - subscription.setVersion(2); - subscription.setIssuer(wallet.getPubKeyHash()); - subscription.setTime(System.currentTimeMillis()/1000); - subscription.setRecipient(resource.getPluginSettings().getNodePubkey()); - subscription.setType(EmailSubscription.TYPE); - - // Encrypt email then fill - String email = resource.getPluginSettings().getMailAdmin(); - byte[] nonce = cryptoService.getBoxRandomNonce(); - - EmailSubscription.Content content = EmailSubscription.newContent(); - content.setEmail(email); - //content.setFrequency(EmailSubscription.Frequency.daily); - String jsonContent = objectMapper.writeValueAsString(content); - - String cypherContent = cryptoService.box(jsonContent, nonce, wallet.getSecKey(), wallet.getPubKey()); - subscription.setRecipientContent(cypherContent); - subscription.setNonce(CryptoUtils.encodeBase58(nonce)); - - // Fill hash + signature - String json = objectMapper.writeValueAsString(subscription); - - json = JsonAttributeParser.newStringParser(Record.PROPERTY_SIGNATURE).removeFromJson(json); - json = JsonAttributeParser.newStringParser(Record.PROPERTY_HASH).removeFromJson(json); - - String hash = cryptoService.hash(json); - subscription.setHash(hash); - subscription.setSignature(cryptoService.sign(hash, wallet.getSecKey())); - - return subscription; - } -} - diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java deleted file mode 100644 index 33932a39e2e83c0da74097cec9210e2d52d8c15d..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java +++ /dev/null @@ -1,109 +0,0 @@ -package org.duniter.elasticsearch.subscription.service; - -/* - * #%L - * Duniter4j :: ElasticSearch Indexer - * %% - * 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.core.client.model.ModelUtils; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.subscription.util.stringtemplate.DateRenderer; -import org.duniter.elasticsearch.subscription.util.stringtemplate.StringRenderer; -import org.junit.Test; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.stringtemplate.v4.ST; -import org.stringtemplate.v4.STGroup; -import org.stringtemplate.v4.STGroupDir; - -import java.util.Date; -import java.util.Locale; - -import static org.junit.Assert.assertNotNull; - -/** - * Created by Benoit on 06/05/2015. - */ -public class SubscriptionTemplateTest { - private static final Logger log = LoggerFactory.getLogger(SubscriptionTemplateTest.class); - - private static final boolean verbose = false; - - @Test - public void testHtmlEmail() throws Exception{ - - try { - STGroup group = new STGroupDir("templates", '$', '$'); - - group.registerRenderer(Date.class, new DateRenderer()); - group.registerRenderer(String.class, new StringRenderer()); - - ST tpl = group.getInstanceOf("html_email_content"); - tpl.add("issuerName", "MyIssuerName"); - tpl.add("issuerPubkey", "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"); - tpl.add("url", "https://g1.duniter.fr"); - tpl.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); - tpl.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); - tpl.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); - tpl.addAggr("events.{description, time}", new Object[]{"My event description 2", new Date()}); - assertNotNull(tpl); - - String email = tpl.render(new Locale("en", "GB")); - - if (verbose) { - System.out.println(email); - } - } - catch (Exception e) { - throw new TechnicalException(e); - } - } - - @Test - public void testTextEmail() throws Exception{ - - try { - STGroup group = new STGroupDir("templates", '$', '$'); - - group.registerRenderer(Date.class, new DateRenderer()); - group.registerRenderer(String.class, new StringRenderer()); - - ST tpl = group.getInstanceOf("text_email"); - tpl.add("issuerPubkey", "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"); - tpl.add("issuerName", "kimamila"); - tpl.add("url", "https://g1.duniter.fr"); - tpl.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); - tpl.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); - tpl.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); - assertNotNull(tpl); - - String text = tpl.render(new Locale("en", "GB")); - - if (verbose) { - System.out.println(text); - } - - } - catch (Exception e) { - throw new TechnicalException(e); - } - } -} - diff --git a/duniter4j-es-subscription/src/test/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-subscription/src/test/resources/META-INF/services/org.duniter.core.beans.Bean deleted file mode 100644 index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..0000000000000000000000000000000000000000 diff --git a/duniter4j-es-subscription/src/test/resources/curl_test.sh b/duniter4j-es-subscription/src/test/resources/curl_test.sh deleted file mode 100755 index 4f62377a7b20ee747e1a99f1c6ab627e9caa8b35..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/resources/curl_test.sh +++ /dev/null @@ -1,16 +0,0 @@ -#!/bin/sh - -curl -XPOST "http://data.duniter.fr/market/comment/_search?pretty" -d' -{ - "query": { - "bool":{ - "filter": [ - {"term":{ - "record":"AVbieTIAup9uzWgKipsC" - } - } - ] - } - } -}' - diff --git a/duniter4j-es-subscription/src/test/resources/duniter4j-es-subscription-test.properties b/duniter4j-es-subscription/src/test/resources/duniter4j-es-subscription-test.properties deleted file mode 100644 index c58166c5f88111dc375775b9077638967d1499e6..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/resources/duniter4j-es-subscription-test.properties +++ /dev/null @@ -1,2 +0,0 @@ -# Empty test file -# (need for inherited TestResource). See files 'src/test/es-home/config' \ No newline at end of file diff --git a/duniter4j-es-subscription/src/test/resources/log4j.properties b/duniter4j-es-subscription/src/test/resources/log4j.properties deleted file mode 100644 index 29c857ca161ac4bb1d91bc0576252fc828c44911..0000000000000000000000000000000000000000 --- a/duniter4j-es-subscription/src/test/resources/log4j.properties +++ /dev/null @@ -1,17 +0,0 @@ -### -# Global logging configuration -log4j.rootLogger=ERROR, stdout - -# Console output -log4j.appender.stdout=org.apache.log4j.ConsoleAppender -log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m%n - -# duniter4j levels -log4j.logger.org.duniter=INFO -log4j.logger.org.duniter.elasticsearch.subscription=DEBUG - -# Other frameworks levels -log4j.logger.org.elasticsearch=WARN -log4j.logger.org.apache=ERROR -log4j.logger.org.nuiton.converter=ERROR \ No newline at end of file diff --git a/duniter4j-es-user/LICENSE.txt b/duniter4j-es-user/LICENSE.txt deleted file mode 100644 index 94a9ed024d3859793618152ea559a168bbcbb5e2..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/LICENSE.txt +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/> - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - <one line to give the program's name and a brief idea of what it does.> - Copyright (C) <year> <name of author> - - 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/>. - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - <program> Copyright (C) <year> <name of author> - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -<http://www.gnu.org/licenses/>. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -<http://www.gnu.org/philosophy/why-not-lgpl.html>. diff --git a/duniter4j-es-user/pom.xml b/duniter4j-es-user/pom.xml deleted file mode 100644 index abf8b1a1990fabcea8d6b5e47a400008ddf6b71a..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/pom.xml +++ /dev/null @@ -1,157 +0,0 @@ -<?xml version="1.0" encoding="UTF-8"?> -<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> - <parent> - <artifactId>duniter4j</artifactId> - <groupId>org.duniter</groupId> - <version>1.0.4-SNAPSHOT</version> - </parent> - <modelVersion>4.0.0</modelVersion> - - <artifactId>duniter4j-es-user</artifactId> - <packaging>jar</packaging> - <name>Duniter4j :: ElasticSearch User plugin</name> - <description>A ElasticSearch plugin used to store user data, such as profiles, private messages and pages</description> - - <dependencies> - <dependency> - <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-core</artifactId> - <version>${project.version}</version> - </dependency> - - <!-- Elastic Search --> - <dependency> - <groupId>org.elasticsearch</groupId> - <artifactId>elasticsearch</artifactId> - <scope>provided</scope> - </dependency> - - <!-- Websocket --> - <dependency> - <groupId>javax.websocket</groupId> - <artifactId>javax.websocket-api</artifactId> - <scope>provided</scope> - </dependency> - - <!-- Unit test --> - <dependency> - <groupId>org.elasticsearch.plugin</groupId> - <artifactId>mapper-attachments</artifactId> - <version>${elasticsearch.version}</version> - <scope>test</scope> - </dependency> - <dependency> - <groupId>log4j</groupId> - <artifactId>log4j</artifactId> - <optional>true</optional> - <scope>test</scope> - </dependency> - <dependency> - <groupId>org.slf4j</groupId> - <artifactId>slf4j-api</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>org.slf4j</groupId> - <artifactId>slf4j-log4j12</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - <scope>test</scope> - </dependency> - <dependency> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna-platform</artifactId> - <scope>test</scope> - <exclusions> - <exclusion> - <groupId>net.java.dev.jna</groupId> - <artifactId>jna</artifactId> - </exclusion> - </exclusions> - </dependency> - <dependency> - <groupId>junit</groupId> - <artifactId>junit</artifactId> - <scope>test</scope> - </dependency> - </dependencies> - - <build> - <resources> - <resource> - <directory>src/main/filtered-resources</directory> - <filtering>true</filtering> - <includes> - <include>*.config</include> - <include>**/*.properties</include> - </includes> - </resource> - <resource> - <directory>src/main/resources</directory> - <filtering>false</filtering> - </resource> - </resources> - - <plugins> - <plugin> - <groupId>org.nuiton.i18n</groupId> - <artifactId>i18n-maven-plugin</artifactId> - - <executions> - <execution> - <id>scan-sources</id> - <configuration> - <entries> - <entry> - <specificGoal>parserValidation</specificGoal> - <basedir>${maven.src.dir}/main/java/</basedir> - <includes> - <param>**/**-validation.xml</param> - </includes> - </entry> - </entries> - </configuration> - <goals> - <goal>parserJava</goal> - <goal>parserValidation</goal> - <goal>gen</goal> - </goals> - </execution> - <execution> - <id>make-bundle</id> - <goals> - <goal>bundle</goal> - </goals> - </execution> - </executions> - </plugin> - - <plugin> - <artifactId>maven-assembly-plugin</artifactId> - <executions> - <execution> - <id>assembly-plugin</id> - <phase>package</phase> - <goals> - <goal>single</goal> - </goals> - <configuration> - <attach>true</attach> - <appendAssemblyId>false</appendAssemblyId> - <finalName>${project.artifactId}-${project.version}</finalName> - <descriptors> - <descriptor> - ${basedir}/src/main/assembly/plugin.xml - </descriptor> - </descriptors> - <skipAssembly>${assembly.skip}</skipAssembly> - </configuration> - </execution> - </executions> - </plugin> - </plugins> - </build> -</project> \ No newline at end of file diff --git a/duniter4j-es-user/src/main/assembly/plugin.xml b/duniter4j-es-user/src/main/assembly/plugin.xml deleted file mode 100644 index 141f9bb56a0c0393ad0f19537ef197805bc5bc18..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/assembly/plugin.xml +++ /dev/null @@ -1,43 +0,0 @@ -<?xml version="1.0"?> -<assembly> - <id>plugin</id> - - - <formats> - <format>zip</format> - </formats> - - <includeBaseDirectory>false</includeBaseDirectory> - - <dependencySets> - <dependencySet> - <outputDirectory>/</outputDirectory> - <useProjectArtifact>true</useProjectArtifact> - <useTransitiveFiltering>true</useTransitiveFiltering> - <excludes> - <exclude>org.duniter:duniter4j-es-core</exclude> - <exclude>org.elasticsearch:elasticsearch</exclude> - <exclude>net.java.dev.jna:jna</exclude> - <exclude>com.fasterxml.jackson.core:jackson-core</exclude> - <exclude>log4j:log4j</exclude> - </excludes> - </dependencySet> - </dependencySets> - - <fileSets> - <fileSet> - <includes> - <include>LICENSE</include> - </includes> - </fileSet> - - <fileSet> - <directory>target/classes</directory> - <outputDirectory/> - <includes> - <include>plugin-descriptor.properties</include> - <include>plugin-security.policy</include> - </includes> - </fileSet> - </fileSets> -</assembly> \ No newline at end of file diff --git a/duniter4j-es-user/src/main/filtered-resources/log4j.properties b/duniter4j-es-user/src/main/filtered-resources/log4j.properties deleted file mode 100644 index 7b6667b1facc361ed8b1993869f728e2c01f1799..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/filtered-resources/log4j.properties +++ /dev/null @@ -1,32 +0,0 @@ - -# Global logging configuration -log4j.rootLogger=ERROR, stdout, file -#log4j.rootLogger=ERROR, stdout - -# Console output -log4j.appender.stdout=org.apache.log4j.ConsoleAppender -log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p %m%n - -# Duniter4j levels -log4j.logger.org.duniter=INFO -#log4j.logger.org.duniter.core.client=DEBUG -#log4j.logger.org.duniter.core.client.service=DEBUG -log4j.logger.org.duniter.elasticsearch=DEBUG - -# Other frameworks levels -log4j.logger.org.nuiton.util=WARN -log4j.logger.org.nuiton.config=WARN -log4j.logger.org.nuiton.converter=WARN -log4j.logger.org.nuiton.i18n=ERROR -log4j.logger.org.elasticsearch=WARN -#log4j.logger.org.elasticsearch=INFO - -log4j.appender.file=org.apache.log4j.RollingFileAppender -log4j.appender.file.file=${duniter4j.log.file} -log4j.appender.file.MaxFileSize=10MB -log4j.appender.file.MaxBackupIndex=4 - -log4j.appender.file.layout=org.apache.log4j.PatternLayout -log4j.appender.file.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m%n - diff --git a/duniter4j-es-user/src/main/filtered-resources/plugin-descriptor.properties b/duniter4j-es-user/src/main/filtered-resources/plugin-descriptor.properties deleted file mode 100644 index f09cec3558235dcb5988a287e6f9d5e0d1551e83..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/filtered-resources/plugin-descriptor.properties +++ /dev/null @@ -1,9 +0,0 @@ -name=duniter4j-es-user -description=Plugin for Duniter User API -version=${project.version} -site=false -jvm=true -classname=org.duniter.elasticsearch.user.Plugin -java.version=1.8 -elasticsearch.version=2.4.6 -isolated=false diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/Plugin.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/Plugin.java deleted file mode 100644 index 987ab8bcc05be1d50f5f0104ef1452f2aec3accb..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/Plugin.java +++ /dev/null @@ -1,93 +0,0 @@ -package org.duniter.elasticsearch.user; - -/* - * #%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 com.google.common.collect.Lists; -import org.duniter.elasticsearch.user.dao.DaoModule; -import org.duniter.elasticsearch.user.rest.RestModule; -import org.duniter.elasticsearch.user.service.ServiceModule; -import org.duniter.elasticsearch.user.synchro.SynchroModule; -import org.duniter.elasticsearch.user.websocket.WebSocketModule; -import org.elasticsearch.common.component.LifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Module; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; - -import java.util.Collection; - -public class Plugin extends org.elasticsearch.plugins.Plugin { - - private ESLogger logger; - - private boolean enable; - - @Inject public Plugin(Settings settings) { - this.enable = settings.getAsBoolean("duniter.user.enabled", true); - this.logger = Loggers.getLogger(Plugin.class.getName(), settings, new String[0]); - } - - @Override - public String name() { - return "duniter4j-es-user"; - } - - @Override - public String description() { - return "Duniter User Plugin"; - } - - @Override - public Collection<Module> nodeModules() { - Collection<Module> modules = Lists.newArrayList(); - if (!enable) { - logger.warn(description() + " has been disabled."); - return modules; - } - - modules.add(new DaoModule()); - modules.add(new ServiceModule()); - modules.add(new WebSocketModule()); - modules.add(new RestModule()); - modules.add(new SynchroModule()); - - return modules; - } - - @Override - public Collection<Class<? extends LifecycleComponent>> nodeServices() { - Collection<Class<? extends LifecycleComponent>> components = Lists.newArrayList(); - if (!enable) { - return components; - } - components.add(PluginSettings.class); - components.add(PluginInit.class); - return components; - } - - /* -- protected methods -- */ - - -} \ No newline at end of file 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 deleted file mode 100644 index 697c1de15174007daf466761a5aea0e76095ba89..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java +++ /dev/null @@ -1,182 +0,0 @@ -package org.duniter.elasticsearch.user; - -/* - * #%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.service.DocStatService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.dao.group.GroupCommentDao; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.group.GroupRecordDao; -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.UserEvent; -import org.duniter.elasticsearch.user.model.UserEventCodes; -import org.duniter.elasticsearch.user.service.*; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.elasticsearch.common.settings.Settings; -import org.nuiton.i18n.I18n; - -/** - * Created by blavenie on 17/06/16. - */ -public class PluginInit extends AbstractLifecycleComponent<PluginInit> { - - private final PluginSettings pluginSettings; - private final ThreadPool threadPool; - private final Injector injector; - private final ESLogger logger; - private final String clusterName; - - @Inject - public PluginInit(Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) { - super(settings); - this.logger = Loggers.getLogger("duniter.user", settings, new String[0]); - this.pluginSettings = pluginSettings; - this.threadPool = threadPool; - this.injector = injector; - this.clusterName = settings.get("cluster.name"); - } - - @Override - protected void doStart() { - threadPool.scheduleOnClusterReady(() -> { - createIndices(); - - // Waiting cluster back to GREEN or YELLOW state, before doAfterStart - threadPool.scheduleOnClusterReady(this::doAfterStart); - - }); - } - - @Override - protected void doStop() { - - } - - @Override - protected void doClose() { - - } - - protected void createIndices() { - - // Reload all indices - if (pluginSettings.reloadAllIndices()) { - if (logger.isInfoEnabled()) { - logger.info("Reloading indices..."); - } - injector.getInstance(HistoryService.class) - .deleteIndex() - .createIndexIfNotExists(); - injector.getInstance(MessageService.class) - .deleteIndex() - .createIndexIfNotExists(); - injector.getInstance(UserService.class) - .deleteIndex() - .createIndexIfNotExists(); - injector.getInstance(GroupService.class) - .deleteIndex() - .createIndexIfNotExists(); - injector.getInstance(UserInvitationService.class) - .deleteIndex() - .createIndexIfNotExists(); - injector.getInstance(PageService.class) - .deleteIndex() - .createIndexIfNotExists(); - - if (logger.isInfoEnabled()) { - logger.info("Reloading indices [OK]"); - } - } - - else { - if (logger.isDebugEnabled()) { - logger.debug("Checking indices..."); - } - - boolean cleanBlockchainUserEvents = injector.getInstance(UserService.class).isIndexExists() && pluginSettings.reloadBlockchainIndices(); - - injector.getInstance(HistoryService.class).createIndexIfNotExists(); - injector.getInstance(UserService.class).createIndexIfNotExists(); - injector.getInstance(MessageService.class).createIndexIfNotExists(); - injector.getInstance(GroupService.class).createIndexIfNotExists(); - injector.getInstance(UserInvitationService.class).createIndexIfNotExists(); - injector.getInstance(PageService.class).createIndexIfNotExists(); - - if (logger.isDebugEnabled()) { - logger.debug("Checking indices [OK]"); - } - - // Clean user events on blockchain - if (cleanBlockchainUserEvents) { - int blockNumber = pluginSettings.reloadBlockchainIndicesFrom(); - if (logger.isInfoEnabled()) { - logger.info(String.format("Deleting user events on blockchain from block #%s (blockchain will be reload)...", blockNumber)); - } - - // Delete events that reference a block - injector.getInstance(UserEventService.class) - .deleteBlockEventsFrom(blockNumber); - if (logger.isInfoEnabled()) { - logger.info("Deleting user events on blockchain [OK]"); - } - } - } - - // Register stats on indices - if (pluginSettings.enableDocStats()) { - injector.getInstance(DocStatService.class) - .registerIndex(UserService.INDEX, UserService.PROFILE_TYPE) - .registerIndex(UserService.INDEX, UserService.SETTINGS_TYPE) - .registerIndex(MessageService.INDEX, MessageService.INBOX_TYPE) - .registerIndex(MessageService.INDEX, MessageService.OUTBOX_TYPE) - .registerIndex(UserInvitationService.INDEX, UserInvitationService.CERTIFICATION_TYPE) - .registerIndex(UserEventService.INDEX, UserEventService.EVENT_TYPE) - .registerIndex(PageIndexDao.INDEX, PageRecordDao.TYPE) - .registerIndex(PageIndexDao.INDEX, PageCommentDao.TYPE) - .registerIndex(GroupIndexDao.INDEX, GroupRecordDao.TYPE) - .registerIndex(GroupIndexDao.INDEX, GroupCommentDao.TYPE) - .registerIndex(HistoryService.INDEX, HistoryService.DELETE_TYPE) - ; - } - - } - - protected void doAfterStart() { - - // Notify admin - injector.getInstance(AdminService.class) - .notifyAdmin(new UserEvent( - UserEvent.EventType.INFO, - UserEventCodes.NODE_STARTED.name(), - I18n.n("duniter.user.event.NODE_STARTED"), - clusterName)); - } - - -} 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 deleted file mode 100644 index 152034a27f4b0a98fd702e965ca11eca076f152a..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java +++ /dev/null @@ -1,255 +0,0 @@ -package org.duniter.elasticsearch.user; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.service.CryptoService; -import org.duniter.core.util.StringUtils; -import org.duniter.core.util.crypto.CryptoUtils; -import org.duniter.core.util.crypto.KeyPair; -import org.elasticsearch.common.component.AbstractLifecycleComponent; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; - -import java.util.Locale; - -/** - * Access to configuration options - * @author Benoit Lavenier <benoit.lavenier@e-is.pro> - * @since 1.0 - */ -public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { - - private org.duniter.elasticsearch.PluginSettings delegate; - private CryptoService cryptoService; - - private KeyPair nodeKeyPair; - private boolean isRandomNodeKeyPair; - private String nodePubkey; - - @Inject - public PluginSettings(Settings settings, - org.duniter.elasticsearch.PluginSettings delegate, - CryptoService cryptoService) { - super(settings); - this.delegate = delegate; - this.cryptoService = cryptoService; - - // Add i18n bundle name - delegate.addI18nBundleName(getI18nBundleName()); - } - - @Override - protected void doStart() { - - } - - @Override - protected void doClose() { - - } - - @Override - protected void doStop() { - - } - - public org.duniter.elasticsearch.PluginSettings getDelegate() { - return delegate; - } - - public String getDefaultStringAnalyzer() { - return delegate.getDefaultStringAnalyzer(); - } - - public boolean reloadAllIndices() { - return delegate.reloadAllIndices(); - } - - public boolean reloadBlockchainIndices() { - return delegate.reloadBlockchainIndices(); - } - - public int reloadBlockchainIndicesFrom() { - return delegate.reloadBlockchainIndicesFrom(); - } - - public boolean enableDocStats() { - return delegate.enableDocStats(); - } - - public boolean enableSynchro() { - return delegate.enableSynchro(); - } - - public int getSynchroTimeOffset() { - return settings.getAsInt("duniter.synchro.timeOffsetInSec", 60*60 /*1 hour*/ ); - } - - - public boolean getMailEnable() { - return settings.getAsBoolean("duniter.mail.enable", Boolean.TRUE); - } - - 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 boolean isMailSmtpStartTLS() { - return settings.getAsBoolean("duniter.mail.smtp.starttls", false); - } - - public boolean isMailSmtpUseSSL() { - return settings.getAsBoolean("duniter.mail.smtp.ssl", false); - } - - 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", "[Cesium+]"); - } - - /* -- delegate methods -- */ - - public String getClusterName() { - return delegate.getClusterName(); - } - - public String getNodeBmaHost() { - return delegate.getNodeBmaHost(); - } - - public int getNodeBmaPort() { - return delegate.getNodeBmaPort(); - } - - public int getIndexBulkSize() { - return delegate.getIndexBulkSize(); - } - - public boolean enableBlockchainSync() { - return delegate.enableBlockchainSync(); - } - - public String getKeyringSalt() { - return delegate.getKeyringSalt(); - } - - public String getKeyringPassword() { - return delegate.getKeyringPassword(); - } - - public String getKeyringPublicKey() { - return delegate.getKeyringPublicKey(); - } - - public String getKeyringSecretKey() { - return delegate.getKeyringSecretKey(); - } - - public boolean allowDocumentDeletionByAdmin() { - return delegate.allowDocumentDeletionByAdmin(); - } - - public void addI18nBundleName(String bundleName) { - delegate.addI18nBundleName(bundleName); - } - - public Locale getI18nLocale() { - return delegate.getI18nLocale(); - } - - public KeyPair getNodeKeypair() { - initNodeKeyring(); - return this.nodeKeyPair; - } - - public boolean isRandomNodeKeypair() { - initNodeKeyring(); - return this.isRandomNodeKeyPair; - } - - public String getNodePubkey() { - initNodeKeyring(); - 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 getShareBaseUrl() { - return settings.get("duniter.share.base.url"); - } - - /* -- protected methods -- */ - - protected String getI18nBundleName() { - return "duniter4j-es-user-i18n"; - } - - protected synchronized void initNodeKeyring() { - if (this.nodeKeyPair != null) return; - if (StringUtils.isNotBlank(getKeyringSalt()) && - StringUtils.isNotBlank(getKeyringPassword())) { - this.nodeKeyPair = cryptoService.getKeyPair(getKeyringSalt(), getKeyringPassword()); - this.nodePubkey = CryptoUtils.encodeBase58(this.nodeKeyPair.getPubKey()); - this.isRandomNodeKeyPair = false; - } - else { - // Use a ramdom keypair - this.nodeKeyPair = cryptoService.getRandomKeypair(); - this.nodePubkey = CryptoUtils.encodeBase58(this.nodeKeyPair.getPubKey()); - this.isRandomNodeKeyPair = true; - - logger.warn(String.format("No keyring in config. salt/password (or keyring) is need to signed user event documents. Will use a generated key [%s]", this.nodePubkey)); - if (logger.isDebugEnabled()) { - logger.debug(String.format(" salt: " + getKeyringSalt().replaceAll(".", "*"))); - logger.debug(String.format("password: " + getKeyringPassword().replaceAll(".", "*"))); - } - } - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/AbstractCommentDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/AbstractCommentDaoImpl.java deleted file mode 100644 index 640921131099ffc3641d9ba8c1a16c3ef8cc04ca..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/AbstractCommentDaoImpl.java +++ /dev/null @@ -1,166 +0,0 @@ -package org.duniter.elasticsearch.user.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.client.model.elasticsearch.Records; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.user.PluginSettings; -import org.elasticsearch.action.search.SearchPhaseExecutionException; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.index.query.TermQueryBuilder; - -import java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class AbstractCommentDaoImpl<T extends AbstractCommentDaoImpl> extends AbstractIndexTypeDao<T> implements CommentDao<T> { - - - protected PluginSettings pluginSettings; - - public AbstractCommentDaoImpl(String index, PluginSettings pluginSettings) { - super(index, CommentDao.TYPE); - this.pluginSettings = pluginSettings; - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - public String create(final String json) { - return super.indexDocumentFromJson(json); - } - - public void update(final String id, final String json) { - super.updateDocumentFromJson(id, json); - } - - @Override - public long countReplies(String id) { - - // Prepare count request - SearchRequestBuilder searchRequest = client - .prepareSearch(getIndex()) - .setTypes(getType()) - .setFetchSource(false) - .setSearchType(SearchType.QUERY_AND_FETCH) - .setSize(0); - - // Query = filter on reference - TermQueryBuilder query = QueryBuilders.termQuery(RecordComment.PROPERTY_REPLY_TO_JSON, id); - searchRequest.setQuery(query); - - // Execute query - try { - SearchResponse response = searchRequest.execute().actionGet(); - return response.getHits().getTotalHits(); - } - catch(SearchPhaseExecutionException e) { - // Failed or no item on index - logger.error(String.format("Error while counting comment replies: %s", e.getMessage()), e); - } - return 1; - } - - - public XContentBuilder createTypeMapping() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // version - .startObject(Record.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // issuer - .startObject(Record.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // creationTime - .startObject(Records.PROPERTY_CREATION_TIME) - .field("type", "integer") - .endObject() - - // time - .startObject(Record.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - // message - .startObject("message") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // record - .startObject("record") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // reply to - .startObject("reply_to") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // aggregations - .startObject("aggregations") - .field("type", "nested") - .field("dynamic", "true") - .startObject("properties") - .startObject("reply_count") - .field("type", "integer") - .field("index", "not_analyzed") - .endObject() - .endObject() - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s]: %s", getType(), ioe.getMessage()), ioe); - } - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/AbstractRecordDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/AbstractRecordDaoImpl.java deleted file mode 100644 index c1428cbe3f1ed466db402340873065cedecd85de..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/AbstractRecordDaoImpl.java +++ /dev/null @@ -1,272 +0,0 @@ -package org.duniter.elasticsearch.user.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.elasticsearch.Records; -import org.duniter.elasticsearch.user.model.UserProfile; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.ObjectUtils; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.user.PluginSettings; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class AbstractRecordDaoImpl<T extends AbstractRecordDaoImpl> extends AbstractIndexTypeDao<T> implements RecordDao<T> { - - protected PluginSettings pluginSettings; - - private boolean isPubkeyFieldEnable = false; - private boolean isNestedPicturesEnable = false; - private boolean isNestedCategoryEnable = false; - - public AbstractRecordDaoImpl(String index, PluginSettings pluginSettings) { - this(index, RecordDao.TYPE, pluginSettings); - } - - public AbstractRecordDaoImpl(String index, String type, PluginSettings pluginSettings) { - super(index, type); - this.pluginSettings = pluginSettings; - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public void checkSameDocumentIssuer(String id, String expectedIssuer) { - String issuer = getMandatoryFieldsById(id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); - if (!ObjectUtils.equals(expectedIssuer, issuer)) { - throw new TechnicalException("Not same issuer"); - } - } - - public XContentBuilder createTypeMapping() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // version - .startObject(Record.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // title - .startObject(Records.PROPERTY_TITLE) - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // description - .startObject(Records.PROPERTY_DESCRIPTION) - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // creationTime - .startObject(Records.PROPERTY_CREATION_TIME) - .field("type", "integer") - .endObject() - - // time - .startObject(Record.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - // issuer - .startObject(Record.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // hash - .startObject(Record.PROPERTY_HASH) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // signature - .startObject(Record.PROPERTY_SIGNATURE) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // address - .startObject(Records.PROPERTY_ADDRESS) - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // city - .startObject(Records.PROPERTY_CITY) - .field("type", "string") - .endObject() - - // geoPoint - .startObject(Records.PROPERTY_GEO_POINT) - .field("type", "geo_point") - .endObject() - - // avatar - .startObject(Records.PROPERTY_AVATAR) - .field("type", "attachment") - .startObject("fields") // fields - .startObject("content") // content - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("author") // author - .field("store", "no") - .endObject() - .startObject("content_type") // content_type - .field("store", "yes") - .endObject() - .endObject() - .endObject() - - // social networks - .startObject(Records.PROPERTY_SOCIALS) - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("type") // type - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("url") // url - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .endObject() - .endObject() - - // tags - .startObject(Records.PROPERTY_TAGS) - .field("type", "completion") - .field("search_analyzer", "simple") - .field("analyzer", "simple") - .field("preserve_separators", "false") - .endObject(); - - // pubkey - if (isPubkeyFieldEnable) { - mapping.startObject("pubkey") - .field("type", "string") - .field("index", "not_analyzed") - .endObject(); - } - - // pictures - if (isNestedPicturesEnable) { - mapping.startObject(Records.PROPERTY_PICTURES) - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("file") // file - .field("type", "attachment") - .startObject("fields") - .startObject("content") // content - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "yes") - .field("analyzer", stringAnalyzer) - .endObject() - .startObject("author") // author - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("content_type") // content_type - .field("store", "yes") - .endObject() - .endObject() - .endObject() - .endObject() - .endObject() - - // picturesCount - .startObject(Records.PROPERTY_PICTURES_COUNT) - .field("type", "integer") - .endObject(); - } - - // category - if (isNestedCategoryEnable) { - mapping.startObject(Records.PROPERTY_CATEGORY) - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("id") // id - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("parent") // parent - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("name") // name - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - .endObject() - .endObject(); - } - - mapping.endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } - - /* -- protected methods -- */ - - protected void setNestedPicturesEnable(boolean isPicturesEnable) { - this.isNestedPicturesEnable = isPicturesEnable; - } - - protected void setNestedCategoryEnable(boolean isNestedCategoryEnable) { - this.isNestedCategoryEnable = isNestedCategoryEnable; - } - - protected void setPubkeyFieldEnable(boolean isPubkeyFieldEnable) { - this.isPubkeyFieldEnable = isPubkeyFieldEnable; - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/CommentDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/CommentDao.java deleted file mode 100644 index 8ac0203aad06a555edf41f5d3c5643194fe152e7..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/CommentDao.java +++ /dev/null @@ -1,41 +0,0 @@ -package org.duniter.elasticsearch.user.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.elasticsearch.dao.IndexTypeDao; - -/** - * Created by Benoit on 30/03/2015. - */ -public interface CommentDao<T extends CommentDao> extends IndexTypeDao<T> { - - String TYPE = "comment"; - - String create(final String json); - - void update(final String id, final String json); - - long countReplies(String id); - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/DaoModule.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/DaoModule.java deleted file mode 100644 index c0431f7dd58f12c071762cd81877961c9450e044..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/DaoModule.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.duniter.elasticsearch.user.dao; - -/* - * #%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.user.dao.group.*; -import org.duniter.elasticsearch.user.dao.page.*; -import org.duniter.elasticsearch.user.dao.profile.*; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class DaoModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // User - bind(UserIndexDao.class).to(UserIndexDaoImpl.class).asEagerSingleton(); - bind(UserProfileDao.class).to(UserProfileDaoImpl.class).asEagerSingleton(); - bind(UserSettingsDao.class).to(UserSettingsDaoImpl.class).asEagerSingleton(); - - // Page - bind(PageIndexDao.class).to(PageIndexDaoImpl.class).asEagerSingleton(); - bind(PageCommentDao.class).to(PageCommentDaoImpl.class).asEagerSingleton(); - bind(PageRecordDao.class).to(PageRecordDaoImpl.class).asEagerSingleton(); - - // Group - bind(GroupIndexDao.class).to(GroupIndexDaoImpl.class).asEagerSingleton(); - bind(GroupCommentDao.class).to(GroupCommentDaoImpl.class).asEagerSingleton(); - bind(GroupRecordDao.class).to(GroupRecordDaoImpl.class).asEagerSingleton(); - - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/RecordDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/RecordDao.java deleted file mode 100644 index 81dd50a205f301d0f1e28c509bf0c7c94a1d196e..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/RecordDao.java +++ /dev/null @@ -1,43 +0,0 @@ -package org.duniter.elasticsearch.user.dao; - -/* - * #%L - * Duniter4j :: 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 org.duniter.elasticsearch.dao.IndexTypeDao; - -/** - * Created by Benoit on 30/03/2015. - */ -public interface RecordDao<T extends RecordDao> extends IndexTypeDao<T> { - - String TYPE = "record"; - - String PROPERTY_AVATAR = "avatar"; - - String create(final String json); - - void update(final String id, final String json); - - void checkSameDocumentIssuer(String id, String expectedIssuer); - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupCommentDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupCommentDao.java deleted file mode 100644 index 6dc51a5a7ade5c83274f4fee6143a855c2f6f528..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupCommentDao.java +++ /dev/null @@ -1,31 +0,0 @@ -package org.duniter.elasticsearch.user.dao.group; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.dao.CommentDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface GroupCommentDao extends CommentDao { -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupCommentDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupCommentDaoImpl.java deleted file mode 100644 index 7a94256941e606d033133f5a151afeaec38fc3a9..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupCommentDaoImpl.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.duniter.elasticsearch.user.dao.group; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.AbstractCommentDaoImpl; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class GroupCommentDaoImpl extends AbstractCommentDaoImpl implements GroupCommentDao { - - - @Inject - public GroupCommentDaoImpl(PluginSettings pluginSettings) { - super(GroupIndexDao.INDEX, pluginSettings); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupIndexDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupIndexDao.java deleted file mode 100644 index 98fb10308ee3abb4a8a7bb387d97ae3765a18b79..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupIndexDao.java +++ /dev/null @@ -1,33 +0,0 @@ -package org.duniter.elasticsearch.user.dao.group; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.dao.IndexDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface GroupIndexDao extends IndexDao<GroupIndexDao> { - String INDEX = "group"; - String CATEGORY_TYPE = "category"; -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupIndexDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupIndexDaoImpl.java deleted file mode 100644 index 662ad6ae27b7083c4dd874892cecb89b0c7e59e0..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupIndexDaoImpl.java +++ /dev/null @@ -1,74 +0,0 @@ -package org.duniter.elasticsearch.user.dao.group; - -/* - * #%L - * Ğchange Pod :: ElasticSearch plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexDao; -import org.duniter.elasticsearch.dao.handler.AddSequenceAttributeHandler; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.CommentDao; -import org.duniter.elasticsearch.user.dao.RecordDao; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class GroupIndexDaoImpl extends AbstractIndexDao<GroupIndexDao> implements GroupIndexDao { - - - private PluginSettings pluginSettings; - private RecordDao recordDao; - private CommentDao commentDao; - - @Inject - public GroupIndexDaoImpl(PluginSettings pluginSettings, GroupRecordDao recordDao, GroupCommentDao commentDao) { - super(INDEX); - - this.pluginSettings = pluginSettings; - this.commentDao = commentDao; - this.recordDao = recordDao; - } - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", getIndex())); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(getIndex()); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(recordDao.getType(), recordDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(commentDao.getType(), commentDao.createTypeMapping()); - createIndexRequestBuilder.execute().actionGet(); - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupRecordDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupRecordDao.java deleted file mode 100644 index e0e2a553f78744c9e68181a24d8fe73b68b16fc0..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupRecordDao.java +++ /dev/null @@ -1,33 +0,0 @@ -package org.duniter.elasticsearch.user.dao.group; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.dao.RecordDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface GroupRecordDao extends RecordDao { - - String create(String id, String json); -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupRecordDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupRecordDaoImpl.java deleted file mode 100644 index 79fa981f55b11ebb6c56269cbd82910dd6cba98a..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/group/GroupRecordDaoImpl.java +++ /dev/null @@ -1,53 +0,0 @@ -package org.duniter.elasticsearch.user.dao.group; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.AbstractRecordDaoImpl; -import org.elasticsearch.action.index.IndexResponse; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class GroupRecordDaoImpl extends AbstractRecordDaoImpl implements GroupRecordDao { - - @Inject - public GroupRecordDaoImpl(PluginSettings pluginSettings) { - super(GroupIndexDao.INDEX, pluginSettings); - - setNestedPicturesEnable(true); - setNestedCategoryEnable(false); // no category - setPubkeyFieldEnable(false); // no pubkey (only issuer) - } - - public String create(String id, String json) { - - IndexResponse response = client.prepareIndex(getIndex(), getType()) - .setSource(json) - .setId(id) - .setRefresh(false) - .execute().actionGet(); - return response.getId(); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageCommentDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageCommentDao.java deleted file mode 100644 index 5ebb33e017508f987667cb6b6bd21c6b67d592f8..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageCommentDao.java +++ /dev/null @@ -1,31 +0,0 @@ -package org.duniter.elasticsearch.user.dao.page; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.dao.CommentDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface PageCommentDao extends CommentDao { -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageCommentDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageCommentDaoImpl.java deleted file mode 100644 index 05ccd5df775256d7f220cae0d204670d2f34b7ed..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageCommentDaoImpl.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.duniter.elasticsearch.user.dao.page; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.AbstractCommentDaoImpl; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class PageCommentDaoImpl extends AbstractCommentDaoImpl implements PageCommentDao { - - - @Inject - public PageCommentDaoImpl(PluginSettings pluginSettings) { - super(PageIndexDao.INDEX, pluginSettings); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageIndexDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageIndexDao.java deleted file mode 100644 index 9cea7936678e616b1c66b9f1de07958517de7e02..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageIndexDao.java +++ /dev/null @@ -1,33 +0,0 @@ -package org.duniter.elasticsearch.user.dao.page; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.dao.IndexDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface PageIndexDao extends IndexDao<PageIndexDao> { - String INDEX = "page"; - String CATEGORY_TYPE = "category"; -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageIndexDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageIndexDaoImpl.java deleted file mode 100644 index 1d69e5b32ccf18f31f4db2a011cee3a836d1f070..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageIndexDaoImpl.java +++ /dev/null @@ -1,122 +0,0 @@ -package org.duniter.elasticsearch.user.dao.page; - -/* - * #%L - * Ğchange Pod :: ElasticSearch plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexDao; -import org.duniter.elasticsearch.dao.handler.AddSequenceAttributeHandler; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.CommentDao; -import org.duniter.elasticsearch.user.dao.RecordDao; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class PageIndexDaoImpl extends AbstractIndexDao<PageIndexDao> implements PageIndexDao { - - - private static final String CATEGORIES_BULK_CLASSPATH_FILE = "page-categories-bulk-insert.json"; - - private PluginSettings pluginSettings; - private RecordDao recordDao; - private CommentDao commentDao; - - @Inject - public PageIndexDaoImpl(PluginSettings pluginSettings, PageRecordDao recordDao, PageCommentDao commentDao) { - super(INDEX); - - this.pluginSettings = pluginSettings; - this.commentDao = commentDao; - this.recordDao = recordDao; - } - - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", getIndex())); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(getIndex()); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(recordDao.getType(), recordDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(commentDao.getType(), commentDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(PageIndexDao.CATEGORY_TYPE, createCategoryTypeMapping()); - createIndexRequestBuilder.execute().actionGet(); - - // Fill categories - fillRecordCategories(); - } - - public void fillRecordCategories() { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s/%s] Fill data", getIndex(), PageIndexDao.CATEGORY_TYPE)); - } - - // Insert categories - client.bulkFromClasspathFile(CATEGORIES_BULK_CLASSPATH_FILE, - getIndex(), - PageIndexDao.CATEGORY_TYPE, - // Add order attribute - new AddSequenceAttributeHandler("order", "\\{.*\"name\".*\\}", 1)); - } - - - protected XContentBuilder createCategoryTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(CATEGORY_TYPE) - .startObject("properties") - - // name - .startObject("name") - .field("type", "string") - .field("analyzer", pluginSettings.getDefaultStringAnalyzer()) - .endObject() - - // parent - .startObject("parent") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), CATEGORY_TYPE, ioe.getMessage()), ioe); - } - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageRecordDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageRecordDao.java deleted file mode 100644 index 6217dfe00a326bc7f3c827a44c398ed196f84328..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageRecordDao.java +++ /dev/null @@ -1,31 +0,0 @@ -package org.duniter.elasticsearch.user.dao.page; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.dao.RecordDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface PageRecordDao extends RecordDao { -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageRecordDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageRecordDaoImpl.java deleted file mode 100644 index 468ceb4f3673710aa2d9993f61919118772c25c8..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/page/PageRecordDaoImpl.java +++ /dev/null @@ -1,42 +0,0 @@ -package org.duniter.elasticsearch.user.dao.page; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.AbstractRecordDaoImpl; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class PageRecordDaoImpl extends AbstractRecordDaoImpl implements PageRecordDao { - - @Inject - public PageRecordDaoImpl(PluginSettings pluginSettings) { - super(PageIndexDao.INDEX, pluginSettings); - - setNestedPicturesEnable(true); - setNestedCategoryEnable(true); - setPubkeyFieldEnable(true); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserIndexDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserIndexDao.java deleted file mode 100644 index 77dd8933f3947c578ab728aebf2fc4d3fcb80954..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserIndexDao.java +++ /dev/null @@ -1,32 +0,0 @@ -package org.duniter.elasticsearch.user.dao.profile; - -/* - * #%L - * Ğchange Pod :: ElasticSearch 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.elasticsearch.dao.IndexDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface UserIndexDao extends IndexDao<UserIndexDao> { - String INDEX = "user"; -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserIndexDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserIndexDaoImpl.java deleted file mode 100644 index 8858df1b958387352fc64bd8688d8ccce49e1042..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserIndexDaoImpl.java +++ /dev/null @@ -1,70 +0,0 @@ -package org.duniter.elasticsearch.user.dao.profile; - -/* - * #%L - * Ğchange Pod :: ElasticSearch plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.elasticsearch.dao.AbstractIndexDao; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.service.UserEventService; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class UserIndexDaoImpl extends AbstractIndexDao<UserIndexDao> implements UserIndexDao { - - - private PluginSettings pluginSettings; - private UserProfileDao profileDao; - private UserSettingsDao settingsDao; - - @Inject - public UserIndexDaoImpl(PluginSettings pluginSettings, UserProfileDao profileDao, UserSettingsDao settingsDao) { - super(INDEX); - - this.pluginSettings = pluginSettings; - this.settingsDao = settingsDao; - this.profileDao = profileDao; - } - - /** - * Create index for mail - * @throws JsonProcessingException - */ - public void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", getIndex())); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(getIndex()); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(profileDao.getType(), profileDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(settingsDao.getType(), settingsDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(UserEventService.EVENT_TYPE, UserEventService.createEventType()); - createIndexRequestBuilder.execute().actionGet(); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserProfileDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserProfileDao.java deleted file mode 100644 index 261569df3be43961cacf7e91020aa73a15b53e68..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserProfileDao.java +++ /dev/null @@ -1,32 +0,0 @@ -package org.duniter.elasticsearch.user.dao.profile; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.elasticsearch.user.dao.RecordDao; - -public interface UserProfileDao<T extends UserProfileDao> extends RecordDao<T> { - - String TYPE = "profile"; - - String create(final String issuer, final String json); -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserProfileDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserProfileDaoImpl.java deleted file mode 100644 index a2af79827f25b2128c0dde1e910dc06a3fc4f2f3..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserProfileDaoImpl.java +++ /dev/null @@ -1,197 +0,0 @@ -package org.duniter.elasticsearch.user.dao.profile; - -/* - * #%L - * Ğchange Pod :: ElasticSearch plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.client.model.elasticsearch.Records; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.ObjectUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.exception.InvalidFormatException; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.model.UserProfile; -import org.elasticsearch.action.index.IndexResponse; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class UserProfileDaoImpl extends AbstractIndexTypeDao<UserProfileDaoImpl> implements UserProfileDao<UserProfileDaoImpl> { - - private PluginSettings pluginSettings; - - @Inject - public UserProfileDaoImpl(PluginSettings pluginSettings) { - super(UserIndexDao.INDEX, UserProfileDao.TYPE); - this.pluginSettings = pluginSettings; - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public void checkSameDocumentIssuer(String id, String expectedIssuer) { - String issuer = getMandatoryFieldsById(id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); - if (!ObjectUtils.equals(expectedIssuer, issuer)) { - throw new TechnicalException("Not same issuer"); - } - } - - @Override - public String create(final String json) { - try { - JsonNode actualObj = getObjectMapper().readTree(json); - String issuer = actualObj.get(Record.PROPERTY_ISSUER).asText(); - - return create(issuer, json); - } - catch(IOException e) { - throw new InvalidFormatException("Invalid record JSON: " + e.getMessage(), e); - } - } - - @Override - public String create(final String issuer, final String json) { - - IndexResponse response = client.prepareIndex(getIndex(), getType()) - .setSource(json) - .setId(issuer) // always use the issuer pubkey as id - .setRefresh(false) - .execute().actionGet(); - return response.getId(); - } - - @Override - public XContentBuilder createTypeMapping() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // version - .startObject(UserProfile.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // title - .startObject(UserProfile.PROPERTY_TITLE) - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // description - .startObject(UserProfile.PROPERTY_DESCRIPTION) - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // time - .startObject(UserProfile.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - // issuer - .startObject(UserProfile.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // city - .startObject(UserProfile.PROPERTY_CITY) - .field("type", "string") - .endObject() - - // address - .startObject(UserProfile.PROPERTY_ADDRESS) - .field("type", "string") - .endObject() - - // geoPoint - .startObject(Records.PROPERTY_GEO_POINT) - .field("type", "geo_point") - .endObject() - - // avatar - .startObject(Records.PROPERTY_AVATAR) - .field("type", "attachment") - .startObject("fields") // fields - .startObject("content") // content - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("author") // author - .field("store", "no") - .endObject() - .startObject("content_type") // content_type - .field("store", "yes") - .endObject() - .endObject() - .endObject() - - // social networks - .startObject(Records.PROPERTY_SOCIALS) - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("type") // type - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("url") // url - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .endObject() - .endObject() - - // tags - .startObject(Records.PROPERTY_TAGS) - .field("type", "completion") - .field("search_analyzer", "simple") - .field("analyzer", "simple") - .field("preserve_separators", "false") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserSettingsDao.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserSettingsDao.java deleted file mode 100644 index b1057437a1f360a98df61e1d5faea5a221d89f47..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserSettingsDao.java +++ /dev/null @@ -1,32 +0,0 @@ -package org.duniter.elasticsearch.user.dao.profile; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.elasticsearch.user.dao.RecordDao; - -public interface UserSettingsDao<T extends UserSettingsDao> extends RecordDao<T> { - - String TYPE = "settings"; - - String create(final String issuer, final String json); -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserSettingsDaoImpl.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserSettingsDaoImpl.java deleted file mode 100644 index 57e3d14b1d82f30bcb74bc8bbb00e753f608c54d..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/dao/profile/UserSettingsDaoImpl.java +++ /dev/null @@ -1,132 +0,0 @@ -package org.duniter.elasticsearch.user.dao.profile; - -/* - * #%L - * Ğchange Pod :: ElasticSearch plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.ObjectUtils; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.exception.InvalidFormatException; -import org.duniter.elasticsearch.user.PluginSettings; -import org.elasticsearch.action.index.IndexResponse; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class UserSettingsDaoImpl extends AbstractIndexTypeDao<UserSettingsDaoImpl> - implements UserSettingsDao<UserSettingsDaoImpl> { - - @Inject - public UserSettingsDaoImpl() { - super(UserIndexDao.INDEX, UserSettingsDao.TYPE); - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public void checkSameDocumentIssuer(String id, String expectedIssuer) { - String issuer = getMandatoryFieldsById(id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); - if (!ObjectUtils.equals(expectedIssuer, issuer)) { - throw new TechnicalException("Not same issuer"); - } - } - - @Override - public String create(final String json) { - try { - JsonNode actualObj = getObjectMapper().readTree(json); - String issuer = actualObj.get(Record.PROPERTY_ISSUER).asText(); - - return create(issuer, json); - } - catch(IOException e) { - throw new InvalidFormatException("Invalid record JSON: " + e.getMessage(), e); - } - } - - @Override - public String create(final String issuer, final String json) { - - IndexResponse response = client.prepareIndex(getIndex(), getType()) - .setSource(json) - .setId(issuer) // always use the issuer pubkey as id - .setRefresh(false) - .execute().actionGet(); - return response.getId(); - } - - @Override - public XContentBuilder createTypeMapping() { - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // version - .startObject(Record.PROPERTY_VERSION) - .field("type", "integer") - .endObject() - - // time - .startObject(Record.PROPERTY_TIME) - .field("type", "integer") - .endObject() - - // issuer - .startObject(Record.PROPERTY_ISSUER) - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // nonce - .startObject("nonce") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // content - .startObject("content") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } -} diff --git a/duniter4j-es-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 deleted file mode 100644 index 3073c805f247a62e2eb6a0c1c8d2f62cda28178b..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/Attachment.java +++ /dev/null @@ -1,63 +0,0 @@ -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/Message.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/Message.java deleted file mode 100644 index 04f2e4c4df7b1100e91cf29dd52556d989b62cd0..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/Message.java +++ /dev/null @@ -1,105 +0,0 @@ -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.JsonIgnore; -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.annotation.JsonSetter; -import com.fasterxml.jackson.databind.ObjectMapper; -import org.duniter.core.util.Preconditions; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.nuiton.i18n.I18n; - -import java.util.Locale; - -/** - * Created by blavenie on 29/11/16. - */ -public class Message extends Record { - - - public static final String PROPERTY_NONCE="nonce"; - public static final String PROPERTY_TITLE="title"; - public static final String PROPERTY_CONTENT="content"; - public static final String PROPERTY_RECIPIENT="recipient"; - public static final String PROPERTY_READ_SIGNATURE="read_signature"; - - private String nonce; - - private String recipient; - - private String content; - - private String readSignature; - - public Message() { - super(); - } - - public String getContent() { - return content; - } - public void setContent(String content) { - this.content = content; - } - - public String getRecipient() { - return recipient; - } - - public void setRecipient(String recipient) { - this.recipient = recipient; - } - - @JsonGetter(PROPERTY_READ_SIGNATURE) - public String getReadSignature() { - return readSignature; - } - - @JsonSetter(PROPERTY_READ_SIGNATURE) - public void setReadSignature(String readSignature) { - this.readSignature = readSignature; - } - - public String getNonce() { - return nonce; - } - - public void setNonce(String nonce) { - this.nonce = nonce; - } - - @JsonIgnore - public String toJson() { - try { - ObjectMapper mapper = new ObjectMapper(); - mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL); - return mapper.writeValueAsString(this); - } catch(Exception e) { - throw new TechnicalException(e); - } - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEvent.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEvent.java deleted file mode 100644 index 70d0d09ef54321e5f62c15d154a6af5ecadec1e8..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEvent.java +++ /dev/null @@ -1,315 +0,0 @@ -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.*; -import com.fasterxml.jackson.databind.ObjectMapper; -import org.duniter.core.util.Preconditions; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.nuiton.i18n.I18n; - -import java.util.Locale; - -/** - * Created by blavenie on 29/11/16. - */ -public class UserEvent extends Record { - - public enum EventType { - INFO, - WARN, - ERROR - } - - public static Builder newBuilder() { - return new Builder(); - } - - public static Builder newBuilder(UserEvent.EventType type, String code) { - return new Builder(type, code, null, null); - } - - public static Builder newBuilder(UserEvent.EventType type, String code, String message, String... params) { - return new Builder(type, code, message, params); - } - - public static final String PROPERTY_ID="id"; - 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"; - public static final String PROPERTY_REFERENCE="reference"; - public static final String PROPERTY_RECIPIENT="recipient"; - - public static final String PROPERTY_READ_SIGNATURE="readSignature"; - - - private String id; - - private EventType type; - - private String recipient; - - private String code; - - private String message; - - private String[] params; - - private Reference reference; - - private String readSignature; - - public UserEvent() { - super(); - } - - public UserEvent(EventType type, String code, String message, String... params) { - super(); - this.type = type; - this.code = code; - this.message = message; - this.params = params; - setTime(getDefaultTime()); - } - - public UserEvent(UserEvent another) { - super(another); - this.type = another.getType(); - this.code = another.getCode(); - this.params = another.getParams(); - this.reference = (another.getReference() != null) ? new Reference(another.getReference()) : null; - this.message = another.getMessage(); - this.recipient = another.getRecipient(); - this.readSignature = another.getReadSignature(); - } - - public EventType getType() { - return type; - } - - public String getCode() { - return code; - } - - public String getMessage() { - return message; - } - - public String[] getParams() { - return params; - } - - public Reference getReference() { - return reference; - } - - public String getLocalizedMessage(Locale locale) { - return I18n.l(locale, this.message, this.params); - } - - public void setType(EventType type) { - this.type = type; - } - - public void setCode(String code) { - this.code = code; - } - - public void setMessage(String message) { - this.message = message; - } - - public void setParams(String[] params) { - this.params = params; - } - - public void setReference(Reference reference) { - this.reference = reference; - } - - public String getRecipient() { - return recipient; - } - - public void setRecipient(String recipient) { - this.recipient = recipient; - } - - @JsonGetter("read_signature") - public String getReadSignature() { - return readSignature; - } - - @JsonSetter("read_signature") - public void setReadSignature(String readSignature) { - this.readSignature = readSignature; - } - - public String getId() { - return id; - } - - public void setId(String id) { - this.id = id; - } - - private static long getDefaultTime() { - return Math.round(1d * System.currentTimeMillis() / 1000); - } - - public static class Builder { - - private UserEvent result; - - private Builder() { - result = new UserEvent(); - } - - public Builder(UserEvent.EventType type, String code, String message, String... params) { - result = new UserEvent(type, code, message, params); - } - - public Builder setMessage(String message, String... params) { - result.setMessage(message); - result.setParams(params); - return this; - } - - public Builder setRecipient(String recipient) { - result.setRecipient(recipient); - return this; - } - - public Builder setIssuer(String issuer) { - result.setIssuer(issuer); - return this; - } - - public Builder setReference(String index, String type, String id) { - result.setReference(new Reference(index, type, id)); - return this; - } - - public Builder setReferenceHash(String hash) { - Preconditions.checkNotNull(result.getReference(), "No reference set. Please call setReference() first"); - result.getReference().setHash(hash); - return this; - } - - public Builder setReference(String index, String type, String id, String anchor) { - result.setReference(new Reference(index, type, id, anchor)); - return this; - } - - public Builder setReferenceAnchor(String anchor) { - Preconditions.checkNotNull(result.getReference(), "No reference set. Please call setReference() first"); - result.getReference().setAnchor(anchor); - return this; - } - - public Builder setTime(long time) { - result.setTime(time); - return this; - } - - public UserEvent build() { - if (result.getTime() == null) { - result.setTime(getDefaultTime()); - } - return new UserEvent(result); - } - } - - - - public static class Reference { - - public static final String PROPERTY_INDEX="index"; - public static final String PROPERTY_TYPE="type"; - public static final String PROPERTY_ID="id"; - public static final String PROPERTY_ANCHOR="anchor"; - public static final String PROPERTY_HASH="hash"; - - private String index; - - private String type; - - private String id; - - private String anchor; - - private String hash; - - public Reference() { - } - - public Reference(String index, String type, String id) { - this(index, type, id, null); - } - - public Reference(String index, String type, String id, String anchor) { - this.index = index; - this.type = type; - this.id = id; - this.anchor = anchor; - } - - public Reference(Reference another) { - this.index = another.getIndex(); - this.type = another.getType(); - this.id = another.getId(); - this.hash = another.getHash(); - this.anchor = another.getAnchor(); - } - - public String getIndex() { - return index; - } - - public String getType() { - return type; - } - - public String getId() { - return id; - } - - public String getAnchor() { - return anchor; - } - - public void setAnchor(String anchor) { - this.anchor = anchor; - } - - public String getHash() { - return hash; - } - - public void setHash(String hash) { - this.hash = hash; - } - } -} 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 deleted file mode 100644 index f56e1474ff3219c7d20e9c7870bc77965aec4f0b..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserEventCodes.java +++ /dev/null @@ -1,54 +0,0 @@ -package org.duniter.elasticsearch.user.model; - -/* - * #%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% - */ - -/** - * Created by blavenie on 29/11/16. - */ -public enum UserEventCodes { - - NODE_STARTED, - NODE_BMA_UP, - NODE_BMA_DOWN, - - // Membership state - MEMBER_JOIN, - MEMBER_LEAVE, - MEMBER_ACTIVE, - MEMBER_REVOKE, - MEMBER_EXCLUDE, - - // TX - TX_SENT, - TX_RECEIVED, - - // CERTIFICATION - CERT_SENT, - CERT_RECEIVED, - - // Message - MESSAGE_RECEIVED, - - // Invitation - INVITATION_TO_CERTIFY -} 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 deleted file mode 100644 index 99c94b55a8ae38bad94d70b3ec0ea852565ec625..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/UserProfile.java +++ /dev/null @@ -1,103 +0,0 @@ -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 org.duniter.core.client.model.elasticsearch.Record; - -/** - * Created by blavenie on 01/03/16. - */ -public class UserProfile extends Record { - - public static final String PROPERTY_TITLE = "title"; - public static final String PROPERTY_DESCRIPTION="description"; - public static final String PROPERTY_EMAIL="email"; - public static final String PROPERTY_LOCALE="locale"; - public static final String PROPERTY_AVATAR="avatar"; - public static final String PROPERTY_ADDRESS="address"; - public static final String PROPERTY_CITY="city"; - - private String title; - private String description; - private String email; - private String locale; - private String address; - private String city; - private Attachment avatar; - - public String getTitle() { - return title; - } - - public void setTitle(String title) { - this.title = title; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public String getEmail() { - return email; - } - - public void setEmail(String email) { - this.email = email; - } - - public String getLocale() { - return locale; - } - - public void setLocale(String locale) { - this.locale = locale; - } - - public String getAddress() { - return address; - } - - public void setAddress(String address) { - this.address = address; - } - - public String getCity() { - return city; - } - - 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/model/page/RegistryRecord.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/page/RegistryRecord.java deleted file mode 100644 index dc9742488334cc0175c6aceefe9e441e3854f2ff..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/model/page/RegistryRecord.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.duniter.elasticsearch.user.model.page; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.core.client.model.elasticsearch.Record; - -import java.util.HashMap; -import java.util.Map; - -/** - * Created by blavenie on 01/12/16. - */ -public class RegistryRecord extends Record { - - public static final String PROPERTY_TITLE="title"; - public static final String PROPERTY_DESCRIPTION="description"; - public static final String PROPERTY_THUMBNAIL="thumbnail"; - - private String title; - private String description; - private Map<String, String> thumbnail = new HashMap<>(); - - public String getTitle() { - return title; - } - - public void setTitle(String title) { - this.title = title; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public Map<String, String> getThumbnail() { - return thumbnail; - } - - public void setThumbnail(Map<String, String> thumbnail) { - this.thumbnail = thumbnail; - } - -} 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 deleted file mode 100644 index fd91b23d0607055b152d9848d1e520c408abadba..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/RestModule.java +++ /dev/null @@ -1,94 +0,0 @@ -package org.duniter.elasticsearch.user.rest; - -/* - * #%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.user.rest.group.*; -import org.duniter.elasticsearch.user.rest.history.RestHistoryDeleteIndexAction; -import org.duniter.elasticsearch.user.rest.invitation.RestInvitationCertificationIndexAction; -import org.duniter.elasticsearch.user.rest.message.RestMessageInboxIndexAction; -import org.duniter.elasticsearch.user.rest.message.RestMessageInboxMarkAsReadAction; -import org.duniter.elasticsearch.user.rest.message.RestMessageOutboxIndexAction; -import org.duniter.elasticsearch.user.rest.message.compat.RestMessageRecordGetAction; -import org.duniter.elasticsearch.user.rest.message.compat.RestMessageRecordIndexAction; -import org.duniter.elasticsearch.user.rest.message.compat.RestMessageRecordMarkAsReadAction; -import org.duniter.elasticsearch.user.rest.message.compat.RestMessageRecordSearchAction; -import org.duniter.elasticsearch.user.rest.mixed.RestMixedSearchAction; -import org.duniter.elasticsearch.user.rest.page.*; -import org.duniter.elasticsearch.user.rest.user.*; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class RestModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // User - bind(RestUserProfileIndexAction.class).asEagerSingleton(); - bind(RestUserProfileUpdateAction.class).asEagerSingleton(); - bind(RestUserSettingsIndexAction.class).asEagerSingleton(); - bind(RestUserSettingsUpdateAction.class).asEagerSingleton(); - bind(RestUserEventMarkAsReadAction.class).asEagerSingleton(); - bind(RestUserEventSearchAction.class).asEagerSingleton(); - bind(RestUserAvatarAction.class).asEagerSingleton(); - bind(RestUserShareLinkAction.class).asEagerSingleton(); - - // Group - bind(RestGroupIndexAction.class).asEagerSingleton(); - bind(RestGroupUpdateAction.class).asEagerSingleton(); - bind(RestGroupCommentIndexAction.class).asEagerSingleton(); - bind(RestGroupCommentUpdateAction.class).asEagerSingleton(); - bind(RestGroupImageAction.class).asEagerSingleton(); - - // History - bind(RestHistoryDeleteIndexAction.class).asEagerSingleton(); - - // Message - bind(RestMessageInboxIndexAction.class).asEagerSingleton(); - bind(RestMessageOutboxIndexAction.class).asEagerSingleton(); - bind(RestMessageInboxMarkAsReadAction.class).asEagerSingleton(); - - // Invitation - bind(RestInvitationCertificationIndexAction.class).asEagerSingleton(); - - // Page - bind(RestPageRecordIndexAction.class).asEagerSingleton(); - bind(RestPageRecordUpdateAction.class).asEagerSingleton(); - bind(RestPageCommentIndexAction.class).asEagerSingleton(); - bind(RestPageCommentUpdateAction.class).asEagerSingleton(); - bind(RestPageCategoryAction.class).asEagerSingleton(); - bind(RestPageImageAction.class).asEagerSingleton(); - bind(RestPageShareLinkAction.class).asEagerSingleton(); - - // Mixed search - bind(RestMixedSearchAction.class).asEagerSingleton(); - - // Backward compatibility - { - // message/record - bind(RestMessageRecordIndexAction.class).asEagerSingleton(); - bind(RestMessageRecordSearchAction.class).asEagerSingleton(); - bind(RestMessageRecordGetAction.class).asEagerSingleton(); - bind(RestMessageRecordMarkAsReadAction.class).asEagerSingleton(); - } - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupCommentIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupCommentIndexAction.java deleted file mode 100644 index cab09d101cad130111d6431a9e67955466b759c3..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupCommentIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.group; - -/* - * #%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.dao.group.GroupCommentDao; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.service.GroupService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestGroupCommentIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestGroupCommentIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - GroupService service) { - super(settings, controller, client, securityController, - GroupIndexDao.INDEX, GroupCommentDao.TYPE, - json -> service.indexCommentFromJson(json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupCommentUpdateAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupCommentUpdateAction.java deleted file mode 100644 index 873c19535d765aa2ba8937ce72c943c247e695b8..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupCommentUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.group; - -/* - * #%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.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.dao.group.GroupCommentDao; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.service.GroupService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestGroupCommentUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestGroupCommentUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - GroupService service) { - super(settings, controller, client, securityController, - GroupIndexDao.INDEX, GroupCommentDao.TYPE, - (id, json) -> service.updateCommentFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupImageAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupImageAction.java deleted file mode 100644 index 7649db196b6ed05808b43d6df208ef71fadb491f..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupImageAction.java +++ /dev/null @@ -1,40 +0,0 @@ -package org.duniter.elasticsearch.user.rest.group; - -/* - * #%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.core.client.model.elasticsearch.UserGroup; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.group.GroupRecordDao; -import org.elasticsearch.common.inject.Inject; - -public class RestGroupImageAction { - - @Inject - public RestGroupImageAction(RestSecurityController securityController) { - - // Allow to get avatar - securityController.allowImageAttachment(GroupIndexDao.INDEX, GroupRecordDao.TYPE, UserGroup.PROPERTY_AVATAR); - - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupIndexAction.java deleted file mode 100644 index ef33347a059e4003777b83c862e67732725840c1..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupIndexAction.java +++ /dev/null @@ -1,46 +0,0 @@ -package org.duniter.elasticsearch.user.rest.group; - -/* - * #%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.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.group.GroupRecordDao; -import org.duniter.elasticsearch.user.service.GroupService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestGroupIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestGroupIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - GroupService service) { - super(settings, controller, client, securityController, - GroupIndexDao.INDEX, - GroupRecordDao.TYPE, - json -> service.indexRecordProfileFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupUpdateAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupUpdateAction.java deleted file mode 100644 index 6aaf9d24c5bd6cba158fad666ef00a684b3bd88c..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/group/RestGroupUpdateAction.java +++ /dev/null @@ -1,47 +0,0 @@ -package org.duniter.elasticsearch.user.rest.group; - -/* - * #%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.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.group.GroupRecordDao; -import org.duniter.elasticsearch.user.service.GroupService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestGroupUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestGroupUpdateAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - GroupService service) { - super(settings, controller, client, securityController, - GroupIndexDao.INDEX, - GroupRecordDao.TYPE, - (id, json) -> service.updateRecordFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/history/RestHistoryDeleteIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/history/RestHistoryDeleteIndexAction.java deleted file mode 100644 index 8c7ddae551366ac77b78701bb70ea003b970ce41..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/history/RestHistoryDeleteIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.history; - -/* - * #%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.HistoryService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestHistoryDeleteIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestHistoryDeleteIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, HistoryService service) { - super(settings, controller, client, securityController, - HistoryService.INDEX, - HistoryService.DELETE_TYPE, - json -> service.indexDeleteFromJson(json)); - } -} \ No newline at end of file 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 deleted file mode 100644 index b33c7af9b68cb8fbffd571af6e66a6ca31c50ccd..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/invitation/RestInvitationCertificationIndexAction.java +++ /dev/null @@ -1,44 +0,0 @@ -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/message/RestMessageInboxIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageInboxIndexAction.java deleted file mode 100644 index 0fbc1e005e34a657bb7b9f630d861ba6f46cc945..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageInboxIndexAction.java +++ /dev/null @@ -1,44 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message; - -/* - * #%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.MessageService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMessageInboxIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestMessageInboxIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - final MessageService service) { - super(settings, controller, client, securityController, - MessageService.INDEX, - MessageService.INBOX_TYPE, - json -> service.indexInboxFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageInboxMarkAsReadAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageInboxMarkAsReadAction.java deleted file mode 100644 index 8aeee2d53b14c3c365e05d312354833c5fe56418..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageInboxMarkAsReadAction.java +++ /dev/null @@ -1,44 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message; - -/* - * #%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.MessageService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMessageInboxMarkAsReadAction extends AbstractRestPostMarkAsReadAction { - - @Inject - public RestMessageInboxMarkAsReadAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - MessageService messageService) { - super(settings, controller, client, securityController, MessageService.INDEX, MessageService.INBOX_TYPE, - (id, signature) -> { - messageService.markMessageAsRead(id, signature); - }); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageOutboxIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageOutboxIndexAction.java deleted file mode 100644 index b3fdb0542741042b30a74327e641a39e54174e5a..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/RestMessageOutboxIndexAction.java +++ /dev/null @@ -1,46 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message; - -/* - * #%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.MessageService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMessageOutboxIndexAction extends AbstractRestPostIndexAction { - - - - @Inject - public RestMessageOutboxIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - final MessageService service) { - super(settings, controller, client, securityController, - MessageService.INDEX, - MessageService.OUTBOX_TYPE, - json -> service.indexOuboxFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordGetAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordGetAction.java deleted file mode 100644 index 1986585e9023cdaf4af84b666036b51440f5ad72..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordGetAction.java +++ /dev/null @@ -1,94 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message.compat; - -/* - * #%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.user.service.MessageService; -import org.elasticsearch.action.get.GetRequest; -import org.elasticsearch.action.get.GetResponse; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.Strings; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.index.VersionType; -import org.elasticsearch.rest.*; -import org.elasticsearch.rest.action.support.RestActions; -import org.elasticsearch.rest.action.support.RestBuilderListener; -import org.elasticsearch.search.fetch.source.FetchSourceContext; - -import static org.elasticsearch.rest.RestRequest.Method.GET; -import static org.elasticsearch.rest.RestStatus.NOT_FOUND; -import static org.elasticsearch.rest.RestStatus.OK; - -/** - * /message/record has been replaced by /message/inbox - * @deprecated - */ -@Deprecated -public class RestMessageRecordGetAction extends BaseRestHandler { - - @Inject - public RestMessageRecordGetAction(Settings settings, RestController controller, Client client) { - super(settings, controller, client); - controller.registerHandler(GET, String.format("%s/%s/{id}", MessageService.INDEX, MessageService.RECORD_TYPE), this); - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel channel, Client client) throws Exception { - GetRequest getRequest = new GetRequest(MessageService.INDEX, MessageService.INBOX_TYPE, request.param("id")); - getRequest.operationThreaded(true); - getRequest.refresh(request.paramAsBoolean("refresh", getRequest.refresh())); - getRequest.routing(request.param("routing")); // order is important, set it after routing, so it will set the routing - getRequest.parent(request.param("parent")); - getRequest.preference(request.param("preference")); - getRequest.realtime(request.paramAsBoolean("realtime", null)); - getRequest.ignoreErrorsOnGeneratedFields(request.paramAsBoolean("ignore_errors_on_generated_fields", false)); - - String sField = request.param("fields"); - if (sField != null) { - String[] sFields = Strings.splitStringByCommaToArray(sField); - if (sFields != null) { - getRequest.fields(sFields); - } - } - - getRequest.version(RestActions.parseVersion(request)); - getRequest.versionType(VersionType.fromString(request.param("version_type"), getRequest.versionType())); - - getRequest.fetchSourceContext(FetchSourceContext.parseFromRestRequest(request)); - - client.get(getRequest, new RestBuilderListener<GetResponse>(channel) { - @Override - public RestResponse buildResponse(GetResponse response, XContentBuilder builder) throws Exception { - builder.startObject(); - response.toXContent(builder, request); - builder.endObject(); - if (!response.isExists()) { - return new BytesRestResponse(NOT_FOUND, builder); - } else { - return new BytesRestResponse(OK, builder); - } - } - }); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordIndexAction.java deleted file mode 100644 index 22858ba1d607c7c964ac3d85db9f41e1e45219e9..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordIndexAction.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message.compat; - -/* - * #%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.MessageService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -/** - * /message/record has been replaced by /message/inbox - * @deprecated - */ -@Deprecated -public class RestMessageRecordIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestMessageRecordIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - final MessageService service) { - super(settings, controller, client, securityController, - MessageService.INDEX, - MessageService.RECORD_TYPE, - json -> service.indexInboxFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordMarkAsReadAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordMarkAsReadAction.java deleted file mode 100644 index 3d242049d9f4f0a74025785b4ebbbb6b8246bb00..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordMarkAsReadAction.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message.compat; - -/* - * #%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.MessageService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -/** - * /message/record has been replaced by /message/inbox - * @deprecated - */ -@Deprecated -public class RestMessageRecordMarkAsReadAction extends AbstractRestPostMarkAsReadAction { - - @Inject - public RestMessageRecordMarkAsReadAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - MessageService messageService) { - super(settings, controller, client, securityController, MessageService.INDEX, MessageService.RECORD_TYPE, - (id, signature) -> { - messageService.markMessageAsRead(id, signature); - }); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordSearchAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordSearchAction.java deleted file mode 100644 index 553435af4692bcc4d2a5849b8ca072393a82152b..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/message/compat/RestMessageRecordSearchAction.java +++ /dev/null @@ -1,65 +0,0 @@ -package org.duniter.elasticsearch.user.rest.message.compat; - -/* - * #%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.user.service.MessageService; -import org.elasticsearch.action.search.SearchRequest; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.bytes.BytesReference; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.BaseRestHandler; -import org.elasticsearch.rest.RestChannel; -import org.elasticsearch.rest.RestController; -import org.elasticsearch.rest.RestRequest; -import org.elasticsearch.rest.action.search.RestSearchAction; -import org.elasticsearch.rest.action.support.RestActions; -import org.elasticsearch.rest.action.support.RestStatusToXContentListener; - -import static org.elasticsearch.rest.RestRequest.Method.GET; -import static org.elasticsearch.rest.RestRequest.Method.POST; - -/** - * /message/record has been replaced by /message/inbox - * @deprecated - */ -@Deprecated -public class RestMessageRecordSearchAction extends BaseRestHandler { - - @Inject - public RestMessageRecordSearchAction(Settings settings, RestController controller, Client client) { - super(settings, controller, client); - controller.registerHandler(GET, String.format("%s/%s/_search", MessageService.INDEX, MessageService.RECORD_TYPE), this); - controller.registerHandler(POST, String.format("%s/%s/_search", MessageService.INDEX, MessageService.RECORD_TYPE), this); - } - - @Override - protected void handleRequest(final RestRequest request, RestChannel channel, Client client) throws Exception { - SearchRequest searchRequest = new SearchRequest(); - BytesReference restContent = RestActions.hasBodyContent(request) ? RestActions.getRestContent(request) : null; - RestSearchAction.parseSearchRequest(searchRequest, request, parseFieldMatcher, restContent); - searchRequest.indices(MessageService.INDEX); // override type - searchRequest.types(MessageService.INBOX_TYPE); // override type - client.search(searchRequest, new RestStatusToXContentListener<>(channel)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/mixed/RestMixedSearchAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/mixed/RestMixedSearchAction.java deleted file mode 100644 index 26b4b4e3ba34e9a3b6b840632463de6c2cb7f9c4..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/mixed/RestMixedSearchAction.java +++ /dev/null @@ -1,59 +0,0 @@ -package org.duniter.elasticsearch.user.rest.mixed; - -/* - * #%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.security.RestSecurityController; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.dao.page.PageRecordDao; -import org.duniter.elasticsearch.user.service.GroupService; -import org.duniter.elasticsearch.user.service.UserService; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -/** - * Created by blavenie on 13/12/16. - */ -public class RestMixedSearchAction { - - @Inject - public RestMixedSearchAction(RestSecurityController securityController) { - - String[] paths = { - // Allow search on profile + page + group - String.format("/%s,%s,%s/%s,%s/_search", - UserService.INDEX, PageIndexDao.INDEX, GroupIndexDao.INDEX, - UserService.PROFILE_TYPE, PageRecordDao.TYPE), - - // Allow search on profile + page - String.format("/%s,%s/%s,%s/_search", - UserService.INDEX, PageIndexDao.INDEX, - UserService.PROFILE_TYPE, PageRecordDao.TYPE) - }; - - for(String path: paths) { - securityController.allow(RestRequest.Method.GET, path); - securityController.allow(RestRequest.Method.POST, path); - } - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCategoryAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCategoryAction.java deleted file mode 100644 index 7e9cd624ad22890ade37011e4f73b5027660786f..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCategoryAction.java +++ /dev/null @@ -1,38 +0,0 @@ -package org.duniter.elasticsearch.user.rest.page; - -/* - * #%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.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -public class RestPageCategoryAction { - - @Inject - public RestPageCategoryAction(RestSecurityController securityController) { - // Add security rule for category - securityController.allowIndexType(RestRequest.Method.GET, PageIndexDao.INDEX, PageIndexDao.CATEGORY_TYPE); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCommentIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCommentIndexAction.java deleted file mode 100644 index e228eca78a8cffdb6d7d984ae30e6ca11fe36fda..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCommentIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.page; - -/* - * #%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.user.dao.page.PageCommentDao; -import org.duniter.elasticsearch.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.service.PageService; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestPageCommentIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestPageCommentIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - PageService service) { - super(settings, controller, client, securityController, - PageIndexDao.INDEX, PageCommentDao.TYPE, - json -> service.indexCommentFromJson(json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCommentUpdateAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCommentUpdateAction.java deleted file mode 100644 index 119605cb242f6d770867660f66992ed998b10356..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageCommentUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.page; - -/* - * #%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.user.dao.page.PageCommentDao; -import org.duniter.elasticsearch.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.service.PageService; -import org.duniter.elasticsearch.rest.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestPageCommentUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestPageCommentUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - PageService service) { - super(settings, controller, client, securityController, - PageIndexDao.INDEX, PageCommentDao.TYPE, - (id, json) -> service.updateCommentFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageImageAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageImageAction.java deleted file mode 100644 index 73fcfa7f75e2248c8bccfb42e5951181625ddfab..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageImageAction.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.duniter.elasticsearch.user.rest.page; - -/* - * #%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.user.dao.RecordDao; -import org.duniter.elasticsearch.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.dao.page.PageRecordDao; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.common.inject.Inject; - -public class RestPageImageAction { - - @Inject - public RestPageImageAction(RestSecurityController securityController) { - - // Allow to get avatar - securityController.allowImageAttachment(PageIndexDao.INDEX, PageRecordDao.TYPE, RecordDao.PROPERTY_AVATAR); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageRecordIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageRecordIndexAction.java deleted file mode 100644 index 89c015bb6bfd0f5880464577a2cf844349569685..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageRecordIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.page; - -/* - * #%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.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.dao.page.PageRecordDao; -import org.duniter.elasticsearch.user.service.PageService; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestPageRecordIndexAction extends AbstractRestPostIndexAction { - - - @Inject - public RestPageRecordIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - PageService service) { - super(settings, controller, client, securityController, - PageIndexDao.INDEX, PageRecordDao.TYPE, - json -> service.indexRecordFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageRecordUpdateAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageRecordUpdateAction.java deleted file mode 100644 index cbc02e493ec047369c62a450b8f6645c2a8caa23..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageRecordUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.page; - -/* - * #%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.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.dao.page.PageRecordDao; -import org.duniter.elasticsearch.user.service.PageService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestPageRecordUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestPageRecordUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - PageService service) { - super(settings, controller, client, securityController, - PageIndexDao.INDEX, PageRecordDao.TYPE, - (id, json) -> service.updateRecordFromJson(id, json)); - } - -} \ No newline at end of file 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 deleted file mode 100644 index 49385cf924df5d1aed059bcf799de15f958b0d7c..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/page/RestPageShareLinkAction.java +++ /dev/null @@ -1,112 +0,0 @@ -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.page.RegistryRecord; -import org.duniter.elasticsearch.user.service.PageService; -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; - -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, - pluginSettings.getShareBaseUrl(), - 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.getShareBaseUrl(); - data.image = StringUtils.isBlank(baseUrl) ? "" : baseUrl; - data.image += RestImageAttachmentAction.computeImageUrl(PageIndexDao.INDEX, PageRecordDao.TYPE, id, RegistryRecord.PROPERTY_THUMBNAIL, record.getThumbnail().get("_content_type")); - - // FIXME : use a greater image ? at least 200px x 200px for FaceBook - data.imageHeight = 100; - data.imageWidth = 100; - } - - // 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_200px.png"; - data.imageType = "image/png"; - data.imageHeight = 200; - data.imageWidth = 200; - } - - return data; - } - catch(UnsupportedEncodingException e) { - throw new TechnicalException(e); - } - }; - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserAvatarAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserAvatarAction.java deleted file mode 100644 index 11a606a7bf9a54081d8a47e85a8a4b00eeb09476..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserAvatarAction.java +++ /dev/null @@ -1,40 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%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.security.RestSecurityController; -import org.duniter.elasticsearch.user.model.UserProfile; -import org.duniter.elasticsearch.user.service.UserService; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -public class RestUserAvatarAction { - - @Inject - public RestUserAvatarAction(RestSecurityController securityController) { - - // Allow to get avatar as image - securityController.allowImageAttachment(UserService.INDEX, UserService.PROFILE_TYPE, UserProfile.PROPERTY_AVATAR); - - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserEventMarkAsReadAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserEventMarkAsReadAction.java deleted file mode 100644 index a2d545d8425815199b2ce4c365370c428d8ce522..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserEventMarkAsReadAction.java +++ /dev/null @@ -1,42 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%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.UserEventService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestUserEventMarkAsReadAction extends AbstractRestPostMarkAsReadAction { - - @Inject - public RestUserEventMarkAsReadAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - UserEventService userEventService) { - super(settings, controller, client, securityController, UserEventService.INDEX, UserEventService.EVENT_TYPE, - (id, signature) -> userEventService.markEventAsRead(id, signature)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserEventSearchAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserEventSearchAction.java deleted file mode 100644 index eccc64263e68d17f88e11d7b596bad8158913f91..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserEventSearchAction.java +++ /dev/null @@ -1,42 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%L - * Duniter4j :: ElasticSearch User 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.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.service.UserEventService; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -/** - * Created by blavenie on 13/12/16. - */ -public class RestUserEventSearchAction { - - @Inject - public RestUserEventSearchAction(RestSecurityController securityController) { - securityController.allow(RestRequest.Method.GET, String.format("/%s/%s/_search", UserEventService.INDEX, UserEventService.EVENT_TYPE)); - securityController.allow(RestRequest.Method.POST, String.format("/%s/%s/_search", UserEventService.INDEX, UserEventService.EVENT_TYPE)); - securityController.allow(RestRequest.Method.GET, String.format("/%s/%s/_count", UserEventService.INDEX, UserEventService.EVENT_TYPE)); - securityController.allow(RestRequest.Method.POST, String.format("/%s/%s/_count", UserEventService.INDEX, UserEventService.EVENT_TYPE)); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserProfileIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserProfileIndexAction.java deleted file mode 100644 index 899db7e74d54db3a845941a58f762ff392496c28..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserProfileIndexAction.java +++ /dev/null @@ -1,44 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%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.UserService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestUserProfileIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestUserProfileIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - UserService service) { - super(settings, controller, client, securityController, - UserService.INDEX, - UserService.PROFILE_TYPE, - json -> service.indexProfileFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserProfileUpdateAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserProfileUpdateAction.java deleted file mode 100644 index 89cf3901c0d7e0eb56f43b93d999165a0ef3b774..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserProfileUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%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.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.service.UserService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestUserProfileUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestUserProfileUpdateAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - UserService service) { - super(settings, controller, client, securityController, - UserService.INDEX, - UserService.PROFILE_TYPE, - (id, json) -> service.updateProfileFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserSettingsIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserSettingsIndexAction.java deleted file mode 100644 index 79371aa009d18686a411833a984847ea75fa0ff7..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserSettingsIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%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.UserService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestUserSettingsIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestUserSettingsIndexAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - final UserService service) { - super(settings, controller, client, securityController, - UserService.INDEX, - UserService.SETTINGS_TYPE, - json -> service.indexSettingsFromJson(json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserSettingsUpdateAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserSettingsUpdateAction.java deleted file mode 100644 index 793c5a3a2a3802f02b92eaf0a890cb3e7d85c260..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserSettingsUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -/* - * #%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.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.service.UserService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestUserSettingsUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestUserSettingsUpdateAction(Settings settings, RestController controller, Client client, - RestSecurityController securityController, - final UserService service) { - super(settings, controller, client, securityController, - UserService.INDEX, - UserService.SETTINGS_TYPE, - (id, json) -> service.updateSettingsFromJson(id, json)); - } - -} \ No newline at end of file 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 deleted file mode 100644 index 7a4775207e80be76f665ed97b3e743c2419a9072..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/rest/user/RestUserShareLinkAction.java +++ /dev/null @@ -1,135 +0,0 @@ -package org.duniter.elasticsearch.user.rest.user; - -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, - pluginSettings.getShareBaseUrl(), - createResolver(pluginSettings, userService)); - - if (StringUtils.isBlank(pluginSettings.getShareBaseUrl())) { - 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(); - } - data.locale = locale.toString(); - - 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.getShareBaseUrl(); - data.image = StringUtils.isBlank(baseUrl) ? "" : baseUrl; - data.image += RestImageAttachmentAction.computeImageUrl(UserService.INDEX, UserService.PROFILE_TYPE, id, UserProfile.PROPERTY_AVATAR, profile.getAvatar().getContentType()); - data.imageHeight = 100; - data.imageWidth = 100; - } - - // 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_200px.png"; - data.imageType = "image/png"; - data.imageHeight = 200; - data.imageWidth = 200; - } - - return data; - } - catch(UnsupportedEncodingException e) { - throw new TechnicalException(e); - } - }; - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AbstractService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AbstractService.java deleted file mode 100644 index 4ae645da796d716430e6c64057bdb6ac3b2a29f0..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AbstractService.java +++ /dev/null @@ -1,53 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: ElasticSearch User 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.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.user.PluginSettings; - -/** - * Created by blavenie on 10/01/17. - */ -public abstract class AbstractService extends org.duniter.elasticsearch.service.AbstractService { - - protected PluginSettings pluginSettings; - - public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings) { - this(loggerName, client, pluginSettings, null); - } - - public AbstractService(Duniter4jClient client, PluginSettings pluginSettings) { - this(client, pluginSettings, null); - } - - public AbstractService(Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - this("duniter.user", client, pluginSettings, cryptoService); - } - - public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - super(loggerName, client, pluginSettings.getDelegate(), cryptoService); - this.pluginSettings = pluginSettings; - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AdminService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AdminService.java deleted file mode 100644 index c262f8884ab9b479306ecf2be988b29e1e61467a..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AdminService.java +++ /dev/null @@ -1,112 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.service.CryptoService; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.model.UserProfile; -import org.elasticsearch.common.inject.Inject; -import org.nuiton.i18n.I18n; - -import java.util.Locale; - -/** - * Created by Benoit on 30/03/2015. - */ -public class AdminService extends AbstractService { - - static { - // Reserve i18n - I18n.n("duniter.admin.event.subject.INFO"); - I18n.n("duniter.admin.event.subject.WARN"); - I18n.n("duniter.admin.event.subject.ERROR"); - } - - private final UserEventService userEventService; - private final MailService mailService; - - @Inject - public AdminService(final Duniter4jClient client, - final PluginSettings pluginSettings, - final CryptoService cryptoService, - final UserEventService userEventService, - final MailService mailService) { - super("duniter.admin", client, pluginSettings, cryptoService); - this.userEventService = userEventService; - this.mailService = mailService; - } - - /** - * Notify cluster admin - */ - public void notifyAdmin(UserEvent event) { - Preconditions.checkNotNull(event); - - String nodePubkey = pluginSettings.getNodePubkey(); - - UserProfile adminProfile; - if (StringUtils.isNotBlank(nodePubkey) && !pluginSettings.isRandomNodeKeypair()) { - adminProfile = getUserProfile(nodePubkey, UserProfile.PROPERTY_EMAIL, UserProfile.PROPERTY_LOCALE); - } - else { - adminProfile = new UserProfile(); - } - - // Add new event to index - Locale locale = StringUtils.isNotBlank(adminProfile.getLocale()) ? - new Locale(adminProfile.getLocale()) : - I18n.getDefaultLocale(); - if (StringUtils.isNotBlank(nodePubkey)) { - event.setRecipient(nodePubkey); - userEventService.indexEvent(locale, event); - } - - // Send email to admin - String adminEmail = StringUtils.isNotBlank(adminProfile.getEmail()) ? - adminProfile.getEmail() : - pluginSettings.getMailAdmin(); - if (StringUtils.isNotBlank(adminEmail)) { - String subjectPrefix = pluginSettings.getMailSubjectPrefix(); - mailService.sendTextEmail( - I18n.l(locale, "duniter.admin.event.subject."+event.getType().name(), subjectPrefix), - event.getLocalizedMessage(locale), - adminEmail); - } - } - - /* -- Internal methods -- */ - - private UserProfile getUserProfile(String pubkey, String... fieldnames) { - UserProfile result = client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); - if (result == null) result = new UserProfile(); - return result; - } - - - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java deleted file mode 100644 index b510b8391bab992cf62b940a21ab5a66bc1d5967..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java +++ /dev/null @@ -1,273 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.google.common.collect.ImmutableSet; -import org.duniter.core.client.model.ModelUtils; -import org.duniter.core.client.model.bma.BlockchainBlock; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.websocket.WebsocketClientEndpoint; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.service.AbstractBlockchainListenerService; -import org.duniter.elasticsearch.service.BlockchainService; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.model.UserEventCodes; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.unit.TimeValue; -import org.nuiton.i18n.I18n; - -import java.util.HashSet; -import java.util.Set; -import java.util.concurrent.TimeUnit; - -/** - * Created by Benoit on 30/03/2015. - */ -public class BlockchainUserEventService extends AbstractBlockchainListenerService { - - public static final String DEFAULT_PUBKEYS_SEPARATOR = ", "; - - private final UserService userService; - private final UserEventService userEventService; - private final AdminService adminService; - - @Inject - public BlockchainUserEventService(Duniter4jClient client, PluginSettings settings, CryptoService cryptoService, - ThreadPool threadPool, - BlockchainService blockchainService, - UserService userService, - AdminService adminService, - UserEventService userEventService) { - super("duniter.user.event.blockchain", client, settings.getDelegate(), cryptoService, threadPool, - new TimeValue(500, TimeUnit.MILLISECONDS)); - this.userService = userService; - this.adminService = adminService; - this.userEventService = userEventService; - - if (this.enable) { - blockchainService.registerConnectionListener(createConnectionListeners()); - } - } - - - @Override - protected void processBlockIndex(ChangeEvent change) { - - BlockchainBlock block = readBlock(change); - - // First: Delete old events on same block - { - UserEvent.Reference reference = new UserEvent.Reference(change.getIndex(), BlockchainService.BLOCK_TYPE, change.getId()); - this.bulkRequest = userEventService.addDeleteEventsByReferenceToBulk(reference, this.bulkRequest, this.bulkSize, false); - flushBulkRequestOrSchedule(); - } - - // Joiners - if (CollectionUtils.isNotEmpty(block.getJoiners())) { - for (BlockchainBlock.Joiner joiner: block.getJoiners()) { - notifyUserEvent(block, joiner.getPublicKey(), UserEventCodes.MEMBER_JOIN, I18n.n("duniter.user.event.MEMBER_JOIN"), block.getCurrency()); - } - } - - // Actives - if (CollectionUtils.isNotEmpty(block.getActives())) { - for (BlockchainBlock.Joiner active: block.getActives()) { - notifyUserEvent(block, active.getPublicKey(), UserEventCodes.MEMBER_ACTIVE, I18n.n("duniter.user.event.MEMBER_ACTIVE"), block.getCurrency()); - } - } - - // Leavers - if (CollectionUtils.isNotEmpty(block.getLeavers())) { - for (BlockchainBlock.Joiner leaver: block.getJoiners()) { - notifyUserEvent(block, leaver.getPublicKey(), UserEventCodes.MEMBER_LEAVE, I18n.n("duniter.user.event.MEMBER_LEAVE"), block.getCurrency()); - } - } - - // Revoked - if (CollectionUtils.isNotEmpty(block.getRevoked())) { - for (BlockchainBlock.Revoked revoked: block.getRevoked()) { - notifyUserEvent(block, revoked.getPubkey(), UserEventCodes.MEMBER_REVOKE, I18n.n("duniter.user.event.MEMBER_REVOKE"), block.getCurrency()); - } - } - - // Excluded - if (CollectionUtils.isNotEmpty(block.getExcluded())) { - for (String excluded: block.getExcluded()) { - notifyUserEvent(block, excluded, UserEventCodes.MEMBER_EXCLUDE, I18n.n("duniter.user.event.MEMBER_EXCLUDE"), block.getCurrency()); - } - } - - // Tx - if (CollectionUtils.isNotEmpty(block.getTransactions())) { - for (BlockchainBlock.Transaction tx: block.getTransactions()) { - processTx(block, tx); - } - } - - // Certifications - if (CollectionUtils.isNotEmpty(block.getCertifications())) { - for (BlockchainBlock.Certification cert: block.getCertifications()) { - processCertification(block, cert); - } - } - } - - @Override - protected void processBlockDelete(ChangeEvent change) { - - UserEvent.Reference reference = new UserEvent.Reference(change.getIndex(), BlockchainService.BLOCK_TYPE, change.getId()); - - if (change.getSource() != null) { - BlockchainBlock block = readBlock(change); - reference.setHash(block.getHash()); - } - - this.bulkRequest = userEventService.addDeleteEventsByReferenceToBulk(reference, this.bulkRequest, this.bulkSize, false); - flushBulkRequestOrSchedule(); - } - - /* -- internal method -- */ - - /** - * Create a listener that notify admin when the Duniter node connection is lost or retrieve - */ - private WebsocketClientEndpoint.ConnectionListener createConnectionListeners() { - return new WebsocketClientEndpoint.ConnectionListener() { - private boolean errorNotified = false; - - @Override - public void onSuccess() { - // Send notify on reconnection - if (errorNotified) { - errorNotified = false; - adminService.notifyAdmin(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.NODE_BMA_UP.name()) - .setMessage(I18n.n("duniter.user.event.NODE_BMA_UP"), - pluginSettings.getNodeBmaHost(), - String.valueOf(pluginSettings.getNodeBmaPort()), - pluginSettings.getClusterName()) - .build()); - } - } - - @Override - public void onError(Exception e, long lastTimeUp) { - if (errorNotified) return; // already notify - - // Wait 1 min, then notify admin (once) - long now = System.currentTimeMillis() / 1000; - boolean wait = now - lastTimeUp < 60; - if (!wait) { - errorNotified = true; - adminService.notifyAdmin(UserEvent.newBuilder(UserEvent.EventType.ERROR, UserEventCodes.NODE_BMA_DOWN.name()) - .setMessage(I18n.n("duniter.user.event.NODE_BMA_DOWN"), - pluginSettings.getNodeBmaHost(), - String.valueOf(pluginSettings.getNodeBmaPort()), - pluginSettings.getClusterName(), - String.valueOf(lastTimeUp)) - .build()); - } - } - }; - } - - - - - private void processTx(BlockchainBlock block, BlockchainBlock.Transaction tx) { - Set<String> senders = ImmutableSet.copyOf(tx.getIssuers()); - - // Received - String senderNames = userService.joinNamesFromPubkeys(senders, DEFAULT_PUBKEYS_SEPARATOR, true); - String sendersPubkeys = ModelUtils.joinPubkeys(senders, DEFAULT_PUBKEYS_SEPARATOR, false); - Set<String> receivers = new HashSet<>(); - for (String output : tx.getOutputs()) { - String[] parts = output.split(":"); - if (parts.length >= 3 && parts[2].startsWith("SIG(")) { - String receiver = parts[2].substring(4, parts[2].length() - 1); - if (!senders.contains(receiver) && !receivers.contains(receiver)) { - notifyUserEvent(block, receiver, UserEventCodes.TX_RECEIVED, I18n.n("duniter.user.event.TX_RECEIVED"), sendersPubkeys, senderNames); - receivers.add(receiver); - } - } - } - - // Sent - if (CollectionUtils.isNotEmpty(receivers)) { - String receiverNames = userService.joinNamesFromPubkeys(receivers, DEFAULT_PUBKEYS_SEPARATOR, true); - String receiverPubkeys = ModelUtils.joinPubkeys(receivers, DEFAULT_PUBKEYS_SEPARATOR, false); - for (String sender : senders) { - notifyUserEvent(block, sender, UserEventCodes.TX_SENT, I18n.n("duniter.user.event.TX_SENT"), receiverPubkeys, receiverNames); - } - } - - } - - private void processCertification(BlockchainBlock block, BlockchainBlock.Certification certification) { - String sender = certification.getFromPubkey(); - String receiver = certification.getToPubkey(); - - // Received - String senderName = userService.getProfileTitle(sender); - if (senderName == null) { - senderName = ModelUtils.minifyPubkey(sender); - } - notifyUserEvent(block, receiver, UserEventCodes.CERT_RECEIVED, I18n.n("duniter.user.event.CERT_RECEIVED"), sender, senderName); - - // Sent - String receiverName = userService.getProfileTitle(receiver); - if (receiverName == null) { - receiverName = ModelUtils.minifyPubkey(receiver); - } - notifyUserEvent(block, sender, UserEventCodes.CERT_SENT, I18n.n("duniter.user.event.CERT_SENT"), receiver, receiverName); - } - - private void notifyUserEvent(BlockchainBlock block, String pubkey, UserEventCodes code, String message, String... params) { - UserEvent event = UserEvent.newBuilder(UserEvent.EventType.INFO, code.name()) - .setRecipient(pubkey) - .setMessage(message, params) - .setTime(block.getMedianTime()) - .setReference(block.getCurrency(), BlockchainService.BLOCK_TYPE, String.valueOf(block.getNumber())) - .setReferenceHash(block.getHash()) - .build(); - - event = userEventService.fillUserEvent(event); - - try { - bulkRequest.add(client.prepareIndex(UserEventService.INDEX, UserEventService.EVENT_TYPE) - .setSource(getObjectMapper().writeValueAsBytes(event)) - .setRefresh(false)); - flushBulkRequestOrSchedule(); - } - catch(JsonProcessingException e) { - logger.error("Could not serialize UserEvent into JSON: " + e.getMessage(), e); - } - } - - -} 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 deleted file mode 100644 index ec2335e3e32a11cb51c061e4a66c240b2bb556bd..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java +++ /dev/null @@ -1,217 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.databind.JsonNode; -import org.apache.commons.collections4.MapUtils; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.client.model.elasticsearch.UserGroup; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.user.dao.group.GroupCommentDao; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.group.GroupRecordDao; -import org.duniter.elasticsearch.user.dao.page.PageIndexDao; -import org.duniter.elasticsearch.user.PluginSettings; -import org.elasticsearch.common.inject.Inject; - -import java.util.HashMap; -import java.util.Map; -import java.util.Set; - -/** - * Created by Benoit on 30/03/2015. - */ -public class GroupService extends AbstractService { - - private GroupIndexDao indexDao; - private GroupCommentDao commentDao; - private GroupRecordDao recordDao; - private HistoryService historyService; - - @Inject - public GroupService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - GroupIndexDao indexDao, - GroupCommentDao commentDao, - GroupRecordDao recordDao, - HistoryService historyService) { - super("duniter.group", client, settings, cryptoService); - this.indexDao = indexDao; - this.commentDao = commentDao; - this.recordDao = recordDao; - this.historyService = historyService; - } - - /** - * Create index need for blockchain registry, if need - */ - public GroupService createIndexIfNotExists() { - indexDao.createIndexIfNotExists(); - return this; - } - - public GroupService deleteIndex() { - indexDao.deleteIndex(); - return this; - } - - /** - * - * Index an record - * @param json - * @return the record id - */ - public String indexRecordProfileFromJson(String json) { - - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String title = getTitle(actualObj); - String id = computeIdFromTitle(title); - String issuer = getIssuer(actualObj); - - // Check time is valid - fix #27 - verifyTimeForInsert(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing group [%s] from issuer [%s]", id, issuer.substring(0, 8))); - } - - return recordDao.create(id, json); - } - - /** - * Update a record - * @param json - */ - public void updateRecordFromJson(String id, String json) { - - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check same document issuer - recordDao.checkSameDocumentIssuer(id, issuer); - - // Check time is valid - fix #27 - verifyTimeForUpdate(recordDao.getIndex(), recordDao.getType(), id, actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Updating %s [%s] from issuer [%s]", recordDao.getType(), id, issuer.substring(0, 8))); - } - - recordDao.update(id, json); - } - - public String indexCommentFromJson(String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - // Check time is valid - fix #27 - verifyTimeForInsert(commentObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] Indexing new %s, issuer {%s}", PageIndexDao.INDEX, commentDao.getType(), issuer.substring(0, 8))); - } - return commentDao.create(json); - } - - public void updateCommentFromJson(String id, String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - // Check time is valid - fix #27 - verifyTimeForUpdate(commentDao.getIndex(), commentDao.getType(), id, commentObj); - - if (logger.isDebugEnabled()) { - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - logger.debug(String.format("[%s] Updating existing %s {%s}, issuer {%s}", PageIndexDao.INDEX, commentDao.getType(), id, issuer.substring(0, 8))); - } - - commentDao.update(id, json); - } - - public String getTitleById(String id) { - - Object title = client.getFieldById(recordDao.getIndex(), recordDao.getType(), id, UserGroup.PROPERTY_TITLE); - if (title == null) return null; - return title.toString(); - } - - public Map<String, String> getTitlesByNames(Set<String> ids) { - - Map<String, Object> titles = client.getFieldByIds(recordDao.getIndex(), recordDao.getType(), ids, UserGroup.PROPERTY_TITLE); - if (MapUtils.isEmpty(titles)) return null; - Map<String, String> result = new HashMap<>(); - titles.entrySet().forEach((entry) -> result.put(entry.getKey(), entry.getValue().toString())); - return result; - } - - /* -- 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−Z0-9_-]+", ""); - if (counter > 0) { - id += "_" + counter; - } - - if (!recordDao.isExists(id)) { - return id; - } - - return computeIdFromTitle(title, counter+1); - } - - // Check the record document exists (or has been deleted) - private void checkRecordExistsOrDeleted(String id) { - boolean recordExists; - try { - recordExists = recordDao.isExists(id); - } catch (NotFoundException e) { - // Check if exists in delete history - recordExists = historyService.existsInDeleteHistory(recordDao.getIndex(), recordDao.getType(), id); - } - if (!recordExists) { - throw new NotFoundException(String.format("Comment refers a non-existent document [%s/%s/%s].", recordDao.getIndex(), recordDao.getType(), id)); - } - } -} 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 deleted file mode 100644 index e94a3532241b8a7a5c570387def1a534743f3eb1..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java +++ /dev/null @@ -1,302 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.elasticsearch.DeleteRecord; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.AccessDeniedException; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.model.Message; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.action.index.IndexResponse; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -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.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; - -import java.io.IOException; -import java.util.Objects; - -/** - * Created by Benoit on 30/03/2015. - */ -public class HistoryService extends AbstractService { - - public static final String INDEX = "history"; - public static final String DELETE_TYPE = "delete"; - - @Inject - public HistoryService(Duniter4jClient client, PluginSettings settings, CryptoService cryptoService) { - super("subscription." + INDEX, client, settings, cryptoService); - } - - /** - * Delete blockchain index, and all data - * @throws JsonProcessingException - */ - public HistoryService deleteIndex() { - client.deleteIndexIfExists(INDEX); - return this; - } - - - public boolean existsIndex() { - return client.existsIndex(INDEX); - } - - /** - * Create index need for blockchain mail, if need - */ - public HistoryService createIndexIfNotExists() { - try { - if (!client.existsIndex(INDEX)) { - createIndex(); - } - } - catch(JsonProcessingException e) { - throw new TechnicalException(String.format("Error while creating index [%s]", INDEX)); - } - - return this; - } - - /** - * Create index need for category mail - * @throws JsonProcessingException - */ - public HistoryService createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s/%s]", INDEX, DELETE_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(DELETE_TYPE, createDeleteType()); - createIndexRequestBuilder.execute().actionGet(); - - return this; - } - - - public String indexDeleteFromJson(String recordJson) { - JsonNode source = readAndVerifyIssuerSignature(recordJson); - - // Check if valid deletion - checkIsValidDeletion(source); - - if (logger.isDebugEnabled()) { - String issuer = source.get(DeleteRecord.PROPERTY_ISSUER).asText(); - String index = getMandatoryField(source, DeleteRecord.PROPERTY_INDEX).asText(); - String type = getMandatoryField(source, DeleteRecord.PROPERTY_TYPE).asText(); - String id = getMandatoryField(source, DeleteRecord.PROPERTY_ID).asText(); - logger.debug(String.format("Deleting document [%s/%s/%s] - issuer [%s]", index, type, id, issuer.substring(0, 8))); - } - - // Add deletion to history - IndexResponse response = client.prepareIndex(INDEX, DELETE_TYPE) - .setSource(recordJson) - .setRefresh(false) - .execute().actionGet(); - - // Delete the document - applyDocDelete(source); - - return response.getId(); - } - - public void checkIsValidDeletion(JsonNode actualObj) { - String issuer = actualObj.get(DeleteRecord.PROPERTY_ISSUER).asText(); - String index = getMandatoryField(actualObj, DeleteRecord.PROPERTY_INDEX).asText(); - String type = getMandatoryField(actualObj,DeleteRecord.PROPERTY_TYPE).asText(); - String id = getMandatoryField(actualObj,DeleteRecord.PROPERTY_ID).asText(); - - if (!client.existsIndex(index)) { - throw new NotFoundException(String.format("Index [%s] not exists.", index)); - } - - try { - // Message: check if deletion issuer is the message recipient - if (MessageService.INDEX.equals(index) && MessageService.INBOX_TYPE.equals(type)) { - client.checkSameDocumentField(index, type, id, Message.PROPERTY_RECIPIENT, issuer); - } - // Invitation: check if deletion issuer is the invitation recipient - else if (UserInvitationService.INDEX.equals(index)) { - - client.checkSameDocumentField(index, type, id, Message.PROPERTY_RECIPIENT, issuer); - - } - else { - // Check same document issuer - client.checkSameDocumentIssuer(index, type, id, issuer); - } - } - catch(AccessDeniedException e) { - // Check if admin ask the deletion - // If deletion done by admin: continue if allow in settings - if (!pluginSettings.isRandomNodeKeypair() - && pluginSettings.allowDocumentDeletionByAdmin() - && Objects.equals(issuer, pluginSettings.getNodePubkey())) { - logger.warn(String.format("[%s/%s] Deletion forced by admin, on doc [%s]", index, type, id)); - } - else { - throw e; - } - } - - // Check time is valid - fix #27 - verifyTimeForInsert(actualObj); - } - - public void applyDocDelete(JsonNode actualObj) { - String index = getMandatoryField(actualObj, DeleteRecord.PROPERTY_INDEX).asText(); - String type = getMandatoryField(actualObj,DeleteRecord.PROPERTY_TYPE).asText(); - String id = getMandatoryField(actualObj,DeleteRecord.PROPERTY_ID).asText(); - - // Delete the document - client.prepareDelete(index, type, id).execute().actionGet(); - } - - public boolean existsInDeleteHistory(final String index, final String type, final String id) { - // Prepare search request - SearchRequestBuilder searchRequest = client - .prepareSearch(INDEX) - .setTypes(DELETE_TYPE) - .setFetchSource(false) - .setSearchType(SearchType.QUERY_AND_FETCH); - - // Query = filter on index/type/id - BoolQueryBuilder boolQuery = QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(DeleteRecord.PROPERTY_INDEX, index)) - .filter(QueryBuilders.termQuery(DeleteRecord.PROPERTY_TYPE, type)) - .filter(QueryBuilders.termQuery(DeleteRecord.PROPERTY_ID, id)); - - searchRequest.setQuery(QueryBuilders.nestedQuery(UserEvent.PROPERTY_REFERENCE, QueryBuilders.constantScoreQuery(boolQuery))); - - // Execute query - SearchResponse response = searchRequest.execute().actionGet(); - return response.getHits().getTotalHits() > 0; - } - - /* -- Internal methods -- */ - - - protected XContentBuilder createDeleteType() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(DELETE_TYPE) - .startObject("properties") - - // index - .startObject("index") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // type - .startObject("type") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // id - .startObject("id") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // time - .startObject("time") - .field("type", "integer") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", INDEX, DELETE_TYPE, ioe.getMessage()), ioe); - } - } - - public XContentBuilder createRecordCommentType() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(DELETE_TYPE) - .startObject("properties") - - // issuer - .startObject("issuer") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // time - .startObject("time") - .field("type", "integer") - .endObject() - - // message - .startObject("message") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // record - .startObject("record") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // reply to - .startObject("reply_to") - .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, DELETE_TYPE, ioe.getMessage()), ioe); - } - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MailService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MailService.java deleted file mode 100644 index 201b907593806b69e5b1c0b63ede843031554fa5..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MailService.java +++ /dev/null @@ -1,127 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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 org.duniter.core.exception.TechnicalException; -import org.duniter.core.model.SmtpConfig; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.user.PluginSettings; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by Benoit on 30/03/2015. - */ -public class MailService extends AbstractService { - - private final org.duniter.core.service.MailService delegate; - - private final boolean enable; - - @Inject - public MailService(final Duniter4jClient client, - final PluginSettings pluginSettings, - final CryptoService cryptoService, - final org.duniter.core.service.MailService delegate) { - super("duniter.mail", client, pluginSettings, cryptoService); - this.delegate = delegate; - this.enable = pluginSettings.getMailEnable(); - // Init delegated service - if (this.enable) { - delegate.setSmtpConfig(createConfig(pluginSettings)); - } - } - - /** - * Send email - */ - public void sendTextEmail(String subject, String textContent, String... recipients) { - if (!this.enable) return; - - try { - delegate.sendTextEmail(subject, textContent, recipients); - } - catch(TechnicalException e) { - if (logger.isDebugEnabled()) { - logger.error(e.getMessage(), e); - } - else { - logger.error(e.getMessage()); - } - } - } - - /** - * Send email - */ - public void sendHtmlEmail(String subject, String htmlContent, String... recipients) { - if (!this.enable) return; - - try { - delegate.sendHtmlEmail(subject, htmlContent, recipients); - } - catch(TechnicalException e) { - if (logger.isDebugEnabled()) { - logger.error(e.getMessage(), e); - } - else { - logger.error(e.getMessage()); - } - } - } - - /** - * Send email - */ - public void sendHtmlEmailWithText(String subject, String textContent, String htmlContent, String... recipients) { - if (!this.enable) return; - - try { - delegate.sendHtmlEmailWithText(subject, textContent, htmlContent, recipients); - } - catch(TechnicalException e) { - if (logger.isDebugEnabled()) { - logger.error(e.getMessage(), e); - } - else { - logger.error(e.getMessage()); - } - } - } - - /* -- internal methods -- */ - - protected SmtpConfig createConfig(PluginSettings pluginSettings) { - SmtpConfig config = new SmtpConfig(); - config.setSmtpHost(pluginSettings.getMailSmtpHost()); - config.setSmtpPort(pluginSettings.getMailSmtpPort()); - config.setSmtpUsername(pluginSettings.getMailSmtpUsername()); - config.setSmtpPassword(pluginSettings.getMailSmtpPassword()); - config.setSenderAddress(pluginSettings.getMailFrom()); - config.setStartTLS(pluginSettings.isMailSmtpStartTLS()); - config.setUseSsl(pluginSettings.isMailSmtpUseSSL()); - return config; - } - -} 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 deleted file mode 100644 index a3964f2570fe533aad3690fe58568efb1638eb32..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java +++ /dev/null @@ -1,266 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.client.Duniter4jClient; -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.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 MessageService extends AbstractService { - - public static final String INDEX = "message"; - public static final String INBOX_TYPE = "inbox"; - public static final String OUTBOX_TYPE = "outbox"; - - @Deprecated - public static final String RECORD_TYPE = "record"; - - - private final UserEventService userEventService; - - @Inject - public MessageService(Duniter4jClient 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 MessageService deleteIndex() { - client.deleteIndexIfExists(INDEX); - return this; - } - - public boolean existsIndex() { - return client.existsIndex(INDEX); - } - - /** - * Create index need for blockchain mail, if need - */ - public MessageService createIndexIfNotExists() { - try { - if (!client.existsIndex(INDEX)) { - createIndex(); - } - } - catch(JsonProcessingException e) { - throw new TechnicalException(String.format("Error while creating index [%s]", INDEX)); - } - - return this; - } - - /** - * Create index need for category mail - * @throws JsonProcessingException - */ - public MessageService createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s/%s]", INDEX, INBOX_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(INBOX_TYPE, createInboxType()); - createIndexRequestBuilder.addMapping(OUTBOX_TYPE, createOutboxType()); - createIndexRequestBuilder.execute().actionGet(); - - return this; - } - - public String indexInboxFromJson(String recordJson) { - - JsonNode actualObj = readAndVerifyIssuerSignature(recordJson); - String issuer = getIssuer(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a message from issuer [%s]", issuer.substring(0, 8))); - } - - IndexResponse response = client.prepareIndex(INDEX, INBOX_TYPE) - .setSource(recordJson) - .setRefresh(false) - .execute().actionGet(); - - String messageId = response.getId(); - - // Notify new message - notifyUser(messageId, actualObj); - - return messageId; - } - - public String indexOuboxFromJson(String recordJson) { - - JsonNode source = readAndVerifyIssuerSignature(recordJson); - - if (logger.isDebugEnabled()) { - String issuer = getMandatoryField(source, Message.PROPERTY_ISSUER).asText(); - logger.debug(String.format("Indexing a message from issuer [%s]", issuer.substring(0, 8))); - } - - IndexResponse response = client.prepareIndex(INDEX, OUTBOX_TYPE) - .setSource(recordJson) - .setRefresh(false) - .execute().actionGet(); - - return response.getId(); - } - - public void notifyUser(String messageId, JsonNode actualObj) { - String issuer = getMandatoryField(actualObj, Message.PROPERTY_ISSUER).asText(); - String recipient = getMandatoryField(actualObj, Message.PROPERTY_RECIPIENT).asText(); - Long time = getMandatoryField(actualObj, Message.PROPERTY_TIME).asLong(); - - // Notify recipient - userEventService.notifyUser(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.MESSAGE_RECEIVED.name()) - .setRecipient(recipient) - .setMessage(I18n.n("duniter.user.event.MESSAGE_RECEIVED"), issuer, ModelUtils.minifyPubkey(issuer)) - .setTime(time) - .setReference(INDEX, INBOX_TYPE, messageId) - .build()); - } - - public void markMessageAsRead(String id, String signature) { - Map<String, Object> fields = client.getMandatoryFieldsById(INDEX, INBOX_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, INBOX_TYPE, id) - .setDoc("read_signature", signature); - request.execute(); - } - - /* -- Internal methods -- */ - - public XContentBuilder createInboxType() { - return createMapping(INBOX_TYPE); - } - - public XContentBuilder createOutboxType() { - return createMapping(OUTBOX_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() - - // title (encrypted) - .startObject("title") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // content (encrypted) - .startObject("content") - .field("type", "string") - .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() - - // 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, INBOX_TYPE, ioe.getMessage()), ioe); - } - } -} 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 deleted file mode 100644 index f4b4429e5af3cca16913fdff7eaf0fc9742d8c80..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/PageService.java +++ /dev/null @@ -1,167 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.databind.JsonNode; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.NotFoundException; -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; - -/** - * Created by Benoit on 30/03/2015. - */ -public class PageService extends AbstractService { - - private PageIndexDao indexDao; - private PageRecordDao recordDao; - private PageCommentDao commentDao; - private HistoryService historyService; - - @Inject - public PageService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - HistoryService historyService, - PageIndexDao indexDao, - PageCommentDao commentDao, - PageRecordDao recordDao) { - super("duniter.page", client, settings, cryptoService); - this.indexDao = indexDao; - this.commentDao = commentDao; - this.recordDao = recordDao; - this.historyService = historyService; - } - - /** - * Create index need for blockchain registry, if need - */ - public PageService createIndexIfNotExists() { - indexDao.createIndexIfNotExists(); - return this; - } - - public PageService deleteIndex() { - indexDao.deleteIndex(); - return this; - } - - public String indexRecordFromJson(String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check time is valid - fix #27 - verifyTimeForInsert(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a %s from issuer [%s]", recordDao.getType(), issuer.substring(0, 8))); - } - - return recordDao.create(json); - } - - public void updateRecordFromJson(String id, String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check same document issuer - recordDao.checkSameDocumentIssuer(id, issuer); - - // Check time is valid - fix #27 - verifyTimeForUpdate(recordDao.getIndex(), recordDao.getType(), id, actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Updating %s [%s] from issuer [%s]", recordDao.getType(), id, issuer.substring(0, 8))); - } - - recordDao.update(id, json); - } - - public String indexCommentFromJson(String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - // Check time is valid - fix #27 - verifyTimeForInsert(commentObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s] Indexing new %s, issuer {%s}", PageIndexDao.INDEX, commentDao.getType(), issuer.substring(0, 8))); - } - return commentDao.create(json); - } - - public void updateCommentFromJson(String id, String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - // Check time is valid - fix #27 - verifyTimeForUpdate(commentDao.getIndex(), commentDao.getType(), id, commentObj); - - if (logger.isDebugEnabled()) { - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - logger.debug(String.format("[%s] Updating existing %s {%s}, issuer {%s}", PageIndexDao.INDEX, commentDao.getType(), id, issuer.substring(0, 8))); - } - - 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 -- */ - - // Check the record document exists (or has been deleted) - private void checkRecordExistsOrDeleted(String id) { - boolean recordExists; - try { - recordExists = recordDao.isExists(id); - } catch (NotFoundException e) { - // Check if exists in delete history - recordExists = historyService.existsInDeleteHistory(recordDao.getIndex(), recordDao.getType(), id); - } - if (!recordExists) { - throw new NotFoundException(String.format("Comment refers a non-existent document [%s/%s/%s].", recordDao.getIndex(), recordDao.getType(), id)); - } - } - -} 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 deleted file mode 100644 index 86a95b9af5024ef301ed001ca7fcf356c1268570..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%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.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class ServiceModule extends AbstractModule implements Module { - - @Override protected void configure() { - bind(HistoryService.class).asEagerSingleton(); - - bind(MessageService.class).asEagerSingleton(); - - bind(UserService.class).asEagerSingleton(); - bind(GroupService.class).asEagerSingleton(); - bind(PageService.class).asEagerSingleton(); - - bind(AdminService.class).asEagerSingleton(); - bind(MailService.class).asEagerSingleton(); - bind(UserEventService.class).asEagerSingleton(); - - bind(UserInvitationService.class).asEagerSingleton(); - - bind(BlockchainUserEventService.class).asEagerSingleton(); - - //bind(SynchroService.class).asEagerSingleton(); - } - - /* protected methods */ - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java deleted file mode 100644 index 71d059d631678e3eca510cf220b2f10e4730ef3b..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java +++ /dev/null @@ -1,538 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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 com.google.common.collect.ImmutableList; -import org.duniter.core.client.model.bma.jackson.JacksonUtils; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.CollectionUtils; -import org.duniter.core.util.Preconditions; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.InvalidSignatureException; -import org.duniter.elasticsearch.service.BlockchainService; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeService; -import org.duniter.elasticsearch.service.changes.ChangeSource; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.model.UserProfile; -import org.elasticsearch.action.ActionFuture; -import org.elasticsearch.action.bulk.BulkRequestBuilder; -import org.elasticsearch.action.index.IndexResponse; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.action.update.UpdateResponse; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.BoolQueryBuilder; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.search.sort.SortOrder; - -import java.io.IOException; -import java.util.*; -import java.util.stream.Collectors; - -/** - * Created by Benoit on 30/03/2015. - */ -public class UserEventService extends AbstractService implements ChangeService.ChangeListener { - - - public interface UserEventListener { - String getId(); - String getPubkey(); - void onEvent(UserEvent event); - } - - public static final String INDEX = "user"; - public static final String EVENT_TYPE = "event"; - - private static final Map<String, UserEventListener> LISTENERS = new HashMap<>(); - - private static final List<ChangeSource> CHANGE_LISTEN_SOURCES = ImmutableList.of(new ChangeSource(INDEX, EVENT_TYPE)); - - public static void registerListener(UserEventListener listener) { - synchronized (LISTENERS) { - LISTENERS.put(listener.getId(), listener); - } - } - - public static synchronized void unregisterListener(UserEventListener listener) { - synchronized (LISTENERS) { - LISTENERS.remove(listener.getId()); - } - } - - private final ThreadPool threadPool; - private final boolean trace; - - @Inject - public UserEventService(final Duniter4jClient client, - final PluginSettings pluginSettings, - final CryptoService cryptoService, - final ThreadPool threadPool) { - super("duniter.user.event", client, pluginSettings, cryptoService); - this.threadPool = threadPool; - this.trace = logger.isTraceEnabled(); - - ChangeService.registerListener(this); - - } - - /** - * Notify a user - */ - public ActionFuture<IndexResponse> notifyUser(UserEvent event) { - Preconditions.checkNotNull(event); - Preconditions.checkNotNull(event.getRecipient()); - - // Get user profile locale - UserProfile userProfile = getUserProfile(event.getRecipient(), - UserProfile.PROPERTY_EMAIL, UserProfile.PROPERTY_TITLE, UserProfile.PROPERTY_LOCALE); - - Locale locale = userProfile.getLocale() != null ? new Locale(userProfile.getLocale()) : null; - - // Add new event to index - return indexEvent(locale, event); - } - - /** - * Notify a user - */ - public UserEvent fillUserEvent(UserEvent event) { - Preconditions.checkNotNull(event); - Preconditions.checkNotNull(event.getRecipient()); - - // Get user profile locale - UserProfile userProfile = getUserProfile(event.getRecipient(), - UserProfile.PROPERTY_EMAIL, UserProfile.PROPERTY_TITLE, UserProfile.PROPERTY_LOCALE); - - Locale locale = userProfile.getLocale() != null ? new Locale(userProfile.getLocale()) : null; - - // Add new event to index - return fillUserEvent(locale, event); - } - - public ActionFuture<IndexResponse> indexEvent(Locale locale, UserEvent event) { - - UserEvent completeUserEvent = fillUserEvent(locale, event); - - // Generate json - String eventJson = toJson(completeUserEvent); - - if (logger.isTraceEnabled()) { - logger.trace(String.format("Indexing a event to recipient [%s]", event.getRecipient().substring(0, 8))); - } - - // do indexation - return indexEvent(eventJson, false /*checkSignature*/); - - } - - public ActionFuture<IndexResponse> indexEvent(String eventJson) { - return indexEvent(eventJson, true); - } - - public ActionFuture<IndexResponse> indexEvent(String eventJson, boolean checkSignature) { - - if (checkSignature) { - JsonNode jsonNode = readAndVerifyIssuerSignature(eventJson); - String recipient = getMandatoryField(jsonNode, UserEvent.PROPERTY_ISSUER).asText(); - if (logger.isTraceEnabled()) { - logger.trace(String.format("Indexing a event to recipient [%s]", recipient.substring(0, 8))); - } - } - if (logger.isTraceEnabled()) { - logger.trace(eventJson); - } - - return client.prepareIndex(INDEX, EVENT_TYPE) - .setSource(eventJson) - .setRefresh(false) - .execute(); - } - - - public void deleteEventsByReference(final UserEvent.Reference reference) { - Preconditions.checkNotNull(reference); - - final int bulkSize = pluginSettings.getIndexBulkSize(); - - BulkRequestBuilder bulkRequest = client.prepareBulk(); - addDeleteEventsByReferenceToBulk(reference, bulkRequest, bulkSize, true); - } - - public void deleteBlockEventsFrom(final int fromBlockNumber) { - final int bulkSize = pluginSettings.getIndexBulkSize(); - - BulkRequestBuilder bulkRequest = client.prepareBulk(); - - // Prepare search request - SearchRequestBuilder searchRequest = client - .prepareSearch(INDEX) - .setTypes(EVENT_TYPE) - .setFetchSource(false) - .setSearchType(SearchType.QUERY_AND_FETCH); - - // Query = filter on reference - BoolQueryBuilder boolQuery = QueryBuilders.boolQuery(); - boolQuery.filter(QueryBuilders.termQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_TYPE, BlockchainService.BLOCK_TYPE)); - boolQuery.filter(QueryBuilders.rangeQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_ID).gte(fromBlockNumber)); - - searchRequest.setQuery(QueryBuilders.nestedQuery(UserEvent.PROPERTY_REFERENCE, QueryBuilders.constantScoreQuery(boolQuery))); - - client.bulkDeleteFromSearch(INDEX, EVENT_TYPE, searchRequest, bulkRequest, bulkSize, true); - } - - public ActionFuture<UpdateResponse> markEventAsRead(String id, String signature) { - - Map<String, Object> fields = client.getMandatoryFieldsById(INDEX, EVENT_TYPE, id, UserEvent.PROPERTY_HASH, UserEvent.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 event as read."); - } - - return client.prepareUpdate(INDEX, EVENT_TYPE, id) - .setDoc("read_signature", signature) - .execute(); - } - - - - public List<UserEvent> getUserEvents(String pubkey, Long lastTime, String[] includesCodes, String[] excludesCodes) { - - BoolQueryBuilder query = QueryBuilders.boolQuery() - .must(QueryBuilders.termQuery(UserEvent.PROPERTY_RECIPIENT, pubkey)); - if (lastTime != null) { - query.must(QueryBuilders.rangeQuery(UserEvent.PROPERTY_TIME).gt(lastTime)); - } - - if (CollectionUtils.isNotEmpty(includesCodes)) { - query.must(QueryBuilders.termsQuery(UserEvent.PROPERTY_CODE, includesCodes)); - } - if (CollectionUtils.isNotEmpty(excludesCodes)) { - query.mustNot(QueryBuilders.termsQuery(UserEvent.PROPERTY_CODE, excludesCodes)); - } - - SearchResponse response = client.prepareSearch(INDEX) - .setTypes(EVENT_TYPE) - .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) - .setFetchSource(true) - .setQuery(query) - .addSort(UserEvent.PROPERTY_TIME, SortOrder.DESC) - .get(); - - - return Arrays.asList(response.getHits().getHits()).stream() - .map(searchHit -> client.readSourceOrNull(searchHit, UserEvent.class)) - .filter(Objects::nonNull) - .collect(Collectors.toList()); - } - - public String toJson(UserEvent userEvent) { - return toJson(userEvent, false); - } - - /* -- Internal methods -- */ - - - protected UserEvent fillUserEvent(Locale locale, UserEvent event) { - Preconditions.checkNotNull(event.getRecipient()); - Preconditions.checkNotNull(event.getType()); - Preconditions.checkNotNull(event.getCode()); - - String nodePubkey = pluginSettings.getNodePubkey(); - - // Generate json - if (StringUtils.isNotBlank(nodePubkey)) { - UserEvent signedEvent = new UserEvent(event); - signedEvent.setMessage(event.getLocalizedMessage(locale)); - // set issuer, hash, signature - signedEvent.setIssuer(nodePubkey); - - // Add hash - String hash = cryptoService.hash(toJson(signedEvent, true)); - signedEvent.setHash(hash); - - // Add signature - String signature = cryptoService.sign(toJson(signedEvent, true), pluginSettings.getNodeKeypair().getSecKey()); - signedEvent.setSignature(signature); - - return signedEvent; - } else { - logger.warn("Could not generate hash for new user event (no keyring)"); - return event; - } - } - - public static 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() - - // reference - .startObject("reference") - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("index") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("type") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("id") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("hash") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("anchor") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .endObject() - .endObject() - - // message - .startObject("message") - .field("type", "string") - .endObject() - - // params - .startObject("params") - .field("type", "string") - .endObject() - - // hash - .startObject("hash") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // signature - .startObject("signature") - .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, EVENT_TYPE, ioe.getMessage()), ioe); - } - } - - private UserProfile getUserProfile(String pubkey, String... fieldnames) { - UserProfile result = client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); - if (result == null) result = new UserProfile(); - return result; - } - - public BulkRequestBuilder addDeleteEventsByReferenceToBulk(final UserEvent.Reference reference, - BulkRequestBuilder bulkRequest, - final int bulkSize, - final boolean flushAll) { - - // Prepare search request - SearchRequestBuilder searchRequest = client - .prepareSearch(INDEX) - .setTypes(EVENT_TYPE) - .setFetchSource(false) - .setSearchType(SearchType.QUERY_AND_FETCH); - - // Query = filter on reference - BoolQueryBuilder boolQuery = QueryBuilders.boolQuery(); - if (StringUtils.isNotBlank(reference.getIndex())) { - boolQuery.filter(QueryBuilders.termQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_INDEX, reference.getIndex())); - } - if (StringUtils.isNotBlank(reference.getType())) { - boolQuery.filter(QueryBuilders.termQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_TYPE, reference.getType())); - } - if (StringUtils.isNotBlank(reference.getId())) { - boolQuery.filter(QueryBuilders.termQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_ID, reference.getId())); - } - if (StringUtils.isNotBlank(reference.getHash())) { - boolQuery.filter(QueryBuilders.termQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_HASH, reference.getHash())); - } - if (StringUtils.isNotBlank(reference.getAnchor())) { - boolQuery.filter(QueryBuilders.termQuery(UserEvent.PROPERTY_REFERENCE + "." + UserEvent.Reference.PROPERTY_ANCHOR, reference.getAnchor())); - } - - searchRequest.setQuery(QueryBuilders.nestedQuery(UserEvent.PROPERTY_REFERENCE, QueryBuilders.constantScoreQuery(boolQuery))); - - // Execute query, while there is some data - return client.bulkDeleteFromSearch(INDEX, EVENT_TYPE, searchRequest, bulkRequest, bulkSize, flushAll); - } - - - private UserProfile getUserProfileOrNull(String pubkey, String... fieldnames) { - return client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); - } - - - private String toJson(UserEvent userEvent, boolean cleanHashAndSignature) { - try { - String json = getObjectMapper().writeValueAsString(userEvent); - if (cleanHashAndSignature) { - json = PARSER_SIGNATURE.removeFromJson(json); - json = PARSER_HASH.removeFromJson(json); - } - return json; - } catch(JsonProcessingException e) { - throw new TechnicalException("Unable to serialize UserEvent object", e); - } - } - - @Override - public String getId() { - return "duniter.user.event"; - } - - @Override - public Collection<ChangeSource> getChangeSources() { - return CHANGE_LISTEN_SOURCES; - } - - @Override - public void onChange(ChangeEvent change) { - - try { - - switch (change.getOperation()) { - // on create - case CREATE: - if (change.getSource() != null) { - UserEvent event = getObjectMapper().readValue(change.getSource().streamInput(), UserEvent.class); - processEventCreate(change.getId(), event); - } - break; - - // on update - case INDEX: - if (change.getSource() != null) { - UserEvent event = getObjectMapper().readValue(change.getSource().streamInput(), UserEvent.class); - processEventUpdate(change.getId(), event); - } - break; - - // on delete - case DELETE: - // Do not propagate deletion - - break; - } - - } - catch(IOException e) { - throw new TechnicalException(String.format("Unable to parse received block %s", change.getId()), e); - } - - } - - private void processEventCreate(final String eventId, final UserEvent event) { - - event.setId(eventId); - - if (LISTENERS.size() > 0) { - // Notify listeners - threadPool.schedule(() -> { - synchronized (LISTENERS) { - LISTENERS.values().forEach(listener -> { - if (event.getRecipient().equals(listener.getPubkey())) { - listener.onEvent(event); - } - }); - } - }); - } - - } - - private void processEventUpdate(final String eventId, final UserEvent event) { - - // Skip if user has already read the event - if (StringUtils.isNotBlank(event.getReadSignature())) { - if (this.trace) logger.trace("Updated event already read: Skip propagation to listeners"); - return; - } - - processEventCreate(eventId, event); - } - -} 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 deleted file mode 100644 index e31f1531ba0581f88e2f92384df88466b41a13ac..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java +++ /dev/null @@ -1,204 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.client.Duniter4jClient; -import org.duniter.elasticsearch.model.SynchroResult; -import org.duniter.elasticsearch.synchro.SynchroActionResult; -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.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; - -/** - * 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(Duniter4jClient 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() { - client.deleteIndexIfExists(INDEX); - return this; - } - - /** - * Create index need for blockchain mail, if need - */ - public UserInvitationService createIndexIfNotExists() { - try { - if (!client.existsIndex(INDEX)) { - createIndex(); - } - } - catch(JsonProcessingException e) { - throw new TechnicalException(String.format("Error while creating index [%s]", INDEX)); - } - - return this; - } - - /** - * Create index need for category mail - * @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 source = readAndVerifyIssuerSignature(recordJson); - - // Check time is valid - fix #27 - verifyTimeForInsert(source); - - if (logger.isDebugEnabled()) { - String issuer = getMandatoryField(source, Message.PROPERTY_ISSUER).asText(); - 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 user - notifyUser(invitationId, source); - - return invitationId; - } - - public void notifyUser(String id, JsonNode source) { - String issuer = getMandatoryField(source, Message.PROPERTY_ISSUER).asText(); - String recipient = getMandatoryField(source, Message.PROPERTY_RECIPIENT).asText(); - Long time = getMandatoryField(source, Message.PROPERTY_TIME).asLong(); - - // Notify recipient - userEventService.notifyUser(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.INVITATION_TO_CERTIFY.name()) - .setRecipient(recipient) - .setMessage(I18n.n("duniter.user.event.INVITATION_TO_CERTIFY"), issuer, ModelUtils.minifyPubkey(issuer)) - .setTime(time) - .setReference(INDEX, CERTIFICATION_TYPE, id) - .build()); - } - - /* -- 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() - - // comment (encrypted) - .startObject("comment") - .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/java/org/duniter/elasticsearch/user/service/UserService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java deleted file mode 100644 index 2d080a88ba9d1309ec607cb8aa9a5345c5103003..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java +++ /dev/null @@ -1,247 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/* - * #%L - * Duniter4j :: 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.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.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; -import java.util.Objects; -import java.util.Set; - -/** - * Created by Benoit on 30/03/2015. - */ -public class UserService extends AbstractService { - - - private UserIndexDao indexDao; - private UserProfileDao profileDao; - private UserSettingsDao settingsDao; - - public static final String INDEX = "user"; - public static final String PROFILE_TYPE = "profile"; - public static final String SETTINGS_TYPE = "settings"; - - @Inject - public UserService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - UserIndexDao indexDao, - UserProfileDao profileDao, - UserSettingsDao settingsDao) { - super("duniter." + INDEX, client, settings.getDelegate(), cryptoService); - this.indexDao = indexDao; - this.profileDao = profileDao; - this.settingsDao = settingsDao; - } - - /** - * Create index need for blockchain mail, if need - */ - public UserService createIndexIfNotExists() { - indexDao.createIndexIfNotExists(); - return this; - } - - /** - * Create index need for blockchain mail, if need - */ - public boolean isIndexExists() { - return indexDao.existsIndex(); - } - - public UserService deleteIndex() { - indexDao.deleteIndex(); - return this; - } - - /** - * - * Index an user profile - * @param profileJson - * @return the profile id - */ - public String indexProfileFromJson(String json) { - Preconditions.checkNotNull(json); - - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check time is valid - fix #27 - verifyTimeForInsert(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a %s from issuer [%s]", profileDao.getType(), issuer.substring(0, 8))); - } - - return profileDao.create(issuer, json); - } - - /** - * Update an user profile - * @param id - * @param json - */ - public void updateProfileFromJson(String id, String json) { - Preconditions.checkNotNull(id); - Preconditions.checkNotNull(json); - - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - if (!Objects.equals(issuer, id)) { - throw new AccessDeniedException(String.format("Could not update this document: only the issuer can update.")); - } - - // Check same document issuer - profileDao.checkSameDocumentIssuer(id, issuer); - - // Check time is valid - fix #27 - verifyTimeForUpdate(profileDao.getIndex(), profileDao.getType(), id, actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Updating a user profile from issuer [%s]", issuer.substring(0, 8))); - } - - profileDao.update(id, json); - } - - /** - * - * Index an user settings - * @param json settings, as JSON string - * @return the settings id (=the issuer pubkey) - */ - public String indexSettingsFromJson(String json) { - - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check time is valid - fix #27 - verifyTimeForInsert(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a user settings from issuer [%s]", issuer.substring(0, 8))); - } - - return settingsDao.create(issuer, json); - } - - /** - * Update user settings - * @param id the doc id (should be =issuer) - * @param json settings, as JSON string - */ - public void updateSettingsFromJson(String id, String json) { - - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - if (!Objects.equals(issuer, id)) { - throw new AccessDeniedException(String.format("Could not update this document: not issuer.")); - } - - // Check time is valid - fix #27 - verifyTimeForUpdate(INDEX, SETTINGS_TYPE, id, actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a user settings from issuer [%s]", issuer.substring(0, 8))); - } - - settingsDao.update(issuer, json); - } - - - public String getProfileTitle(String issuer) { - - Object title = client.getFieldById(INDEX, PROFILE_TYPE, issuer, UserProfile.PROPERTY_TITLE); - if (title == null) return null; - return title.toString(); - } - - public Map<String, String> getProfileTitles(Set<String> issuers) { - - Map<String, Object> titles = client.getFieldByIds(INDEX, PROFILE_TYPE, issuers, UserProfile.PROPERTY_TITLE); - if (MapUtils.isEmpty(titles)) return null; - Map<String, String> result = new HashMap<>(); - titles.entrySet().forEach((entry) -> result.put(entry.getKey(), entry.getValue().toString())); - return result; - } - - public String joinNamesFromPubkeys(Set<String> pubkeys, String separator, boolean minify) { - Preconditions.checkNotNull(pubkeys); - Preconditions.checkNotNull(separator); - Preconditions.checkArgument(pubkeys.size()>0); - if (pubkeys.size() == 1) { - String pubkey = pubkeys.iterator().next(); - String title = getProfileTitle(pubkey); - return title != null ? title : - (minify ? ModelUtils.minifyPubkey(pubkey) : pubkey); - } - - Map<String, String> profileTitles = getProfileTitles(pubkeys); - StringBuilder sb = new StringBuilder(); - pubkeys.forEach((pubkey)-> { - String title = profileTitles != null ? profileTitles.get(pubkey) : null; - sb.append(separator); - sb.append(title != null ? title : - (minify ? ModelUtils.minifyPubkey(pubkey) : pubkey)); - }); - - 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/java/org/duniter/elasticsearch/user/synchro/SynchroModule.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/SynchroModule.java deleted file mode 100644 index 981f1fd835b0d388636de55954fcde19b80b9a30..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/SynchroModule.java +++ /dev/null @@ -1,84 +0,0 @@ -package org.duniter.elasticsearch.user.synchro; - -/* - * #%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.core.client.model.bma.EndpointApi; -import org.duniter.elasticsearch.service.PeerService; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.synchro.group.SynchroGroupCommentAction; -import org.duniter.elasticsearch.user.synchro.group.SynchroGroupRecordAction; -import org.duniter.elasticsearch.user.synchro.history.SynchroHistoryIndexAction; -import org.duniter.elasticsearch.user.synchro.invitation.SynchroInvitationCertificationIndexAction; -import org.duniter.elasticsearch.user.synchro.message.SynchroMessageInboxIndexAction; -import org.duniter.elasticsearch.user.synchro.message.SynchroMessageOutboxIndexAction; -import org.duniter.elasticsearch.user.synchro.page.SynchroPageCommentAction; -import org.duniter.elasticsearch.user.synchro.page.SynchroPageRecordAction; -import org.duniter.elasticsearch.user.synchro.user.SynchroUserProfileAction; -import org.duniter.elasticsearch.user.synchro.user.SynchroUserSettingsAction; -import org.duniter.elasticsearch.user.websocket.WebsocketUserEventEndPoint; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Module; - -public class SynchroModule extends AbstractModule implements Module { - - public static class Init { - - @Inject - public Init(PeerService peerService, PluginSettings pluginSettings) { - if (pluginSettings.enableSynchro()) { - // Make sure PeerService will index ES_USER_API peers - peerService.addIncludeEndpointApi(EndpointApi.ES_USER_API); - } - } - } - - @Override protected void configure() { - - bind(Init.class).asEagerSingleton(); - - // History - bind(SynchroHistoryIndexAction.class).asEagerSingleton(); - - // User - bind(SynchroUserProfileAction.class).asEagerSingleton(); - bind(SynchroUserSettingsAction.class).asEagerSingleton(); - - // Message - bind(SynchroMessageInboxIndexAction.class).asEagerSingleton(); - bind(SynchroMessageOutboxIndexAction.class).asEagerSingleton(); - - // Page - bind(SynchroPageRecordAction.class).asEagerSingleton(); - bind(SynchroPageCommentAction.class).asEagerSingleton(); - - // Group - bind(SynchroGroupRecordAction.class).asEagerSingleton(); - bind(SynchroGroupCommentAction.class).asEagerSingleton(); - - // Invitation - bind(SynchroInvitationCertificationIndexAction.class).asEagerSingleton(); - - } - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/group/SynchroGroupCommentAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/group/SynchroGroupCommentAction.java deleted file mode 100644 index ea618a1fcf6ce2580d0eded4253a46e3d33cd4c5..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/group/SynchroGroupCommentAction.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.group; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.group.GroupCommentDao; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.elasticsearch.common.inject.Inject; - -public class SynchroGroupCommentAction extends AbstractSynchroAction { - - @Inject - public SynchroGroupCommentAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(GroupIndexDao.INDEX, GroupCommentDao.TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - synchroService.register(this); - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/group/SynchroGroupRecordAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/group/SynchroGroupRecordAction.java deleted file mode 100644 index 2bdd76e9ef246e3b4d48a5e36978cf43b1afc8f9..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/group/SynchroGroupRecordAction.java +++ /dev/null @@ -1,51 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.group; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.group.GroupIndexDao; -import org.duniter.elasticsearch.user.dao.group.GroupRecordDao; -import org.duniter.elasticsearch.user.service.GroupService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroGroupRecordAction extends AbstractSynchroAction { - - @Inject - public SynchroGroupRecordAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(GroupIndexDao.INDEX, GroupRecordDao.TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - synchroService.register(this); - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/history/SynchroHistoryIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/history/SynchroHistoryIndexAction.java deleted file mode 100644 index 0b09ccc91ee5b90e0af85030b325866a8d4ee0d6..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/history/SynchroHistoryIndexAction.java +++ /dev/null @@ -1,80 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.history; - -/*- - * #%L - * Duniter4j :: ElasticSearch User plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.synchro.SynchroActionResult; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.service.HistoryService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroHistoryIndexAction extends AbstractSynchroAction { - - private HistoryService service; - @Inject - public SynchroHistoryIndexAction(final Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService, - HistoryService service) { - super(service.INDEX, service.DELETE_TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - this.service = service; - - addValidationListener(this::onValidate); - addInsertionListener(this::onInsert); - - synchroService.register(this); - } - - /* -- protected method -- */ - - protected void onValidate(String deleteId, JsonNode source, SynchroActionResult result) { - try { - // Check if valid document - service.checkIsValidDeletion(source); - - } catch(NotFoundException e) { - // doc not exists: continue - } - } - - protected void onInsert(String deleteId, JsonNode source, SynchroActionResult result) { - try { - // Delete the document - service.applyDocDelete(source); - - result.addDelete(); - - } catch(NotFoundException e) { - // doc not exists: continue - logger.debug("Doc to delete could not be found. Skipping deletion"); - } - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/invitation/SynchroInvitationCertificationIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/invitation/SynchroInvitationCertificationIndexAction.java deleted file mode 100644 index 4a99fd15f866324dedaed0e33d8a1cf2e2ce7f36..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/invitation/SynchroInvitationCertificationIndexAction.java +++ /dev/null @@ -1,60 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.invitation; - -/*- - * #%L - * Duniter4j :: ElasticSearch User plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroActionResult; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.service.UserInvitationService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroInvitationCertificationIndexAction extends AbstractSynchroAction { - - private UserInvitationService service; - - @Inject - public SynchroInvitationCertificationIndexAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService, - UserInvitationService service) { - super(UserInvitationService.INDEX, UserInvitationService.CERTIFICATION_TYPE, client, - pluginSettings.getDelegate(), cryptoService, threadPool); - - this.service = service; - - addInsertionListener(this::onInsert); - - synchroService.register(this); - } - - protected void onInsert(String id, JsonNode source, SynchroActionResult result) { - service.notifyUser(id, source); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/message/SynchroMessageInboxIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/message/SynchroMessageInboxIndexAction.java deleted file mode 100644 index e5a2fadf15aa346f45b570e987b2666d89e00979..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/message/SynchroMessageInboxIndexAction.java +++ /dev/null @@ -1,58 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.message; - -/*- - * #%L - * Duniter4j :: ElasticSearch User plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroActionResult; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.service.MessageService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroMessageInboxIndexAction extends AbstractSynchroAction { - - private MessageService service; - @Inject - public SynchroMessageInboxIndexAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService, - MessageService service) { - super(MessageService.INDEX, MessageService.INBOX_TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - this.service = service; - - addInsertionListener(this::onInsert); - - synchroService.register(this); - } - - protected void onInsert(String id, JsonNode source, SynchroActionResult result) { - service.notifyUser(id, source); - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/message/SynchroMessageOutboxIndexAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/message/SynchroMessageOutboxIndexAction.java deleted file mode 100644 index 70ad3d50f066971fc2f44a4f67ee9fab83e504ee..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/message/SynchroMessageOutboxIndexAction.java +++ /dev/null @@ -1,49 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.message; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.service.MessageService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroMessageOutboxIndexAction extends AbstractSynchroAction { - - @Inject - public SynchroMessageOutboxIndexAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(MessageService.INDEX, MessageService.OUTBOX_TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(false); // no update - - synchroService.register(this); - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/page/SynchroPageCommentAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/page/SynchroPageCommentAction.java deleted file mode 100644 index be42592e463180abb71837bb8155380c350097e0..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/page/SynchroPageCommentAction.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.page; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -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.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroPageCommentAction extends AbstractSynchroAction { - - @Inject - public SynchroPageCommentAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(PageIndexDao.INDEX, PageCommentDao.TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - synchroService.register(this); - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/page/SynchroPageRecordAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/page/SynchroPageRecordAction.java deleted file mode 100644 index 99bc81be59bfa8303766f74b56e34049595bb113..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/page/SynchroPageRecordAction.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.page; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -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.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -public class SynchroPageRecordAction extends AbstractSynchroAction { - - @Inject - public SynchroPageRecordAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(PageIndexDao.INDEX, PageRecordDao.TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - synchroService.register(this); - } - -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/user/SynchroUserProfileAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/user/SynchroUserProfileAction.java deleted file mode 100644 index f11a606b0fffcfe2ad89574def0437a3150250f8..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/user/SynchroUserProfileAction.java +++ /dev/null @@ -1,64 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.user; - -/*- - * #%L - * Duniter4j :: ElasticSearch User plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.AccessDeniedException; -import org.duniter.elasticsearch.synchro.SynchroActionResult; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.service.UserService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -import java.util.Objects; - -public class SynchroUserProfileAction extends AbstractSynchroAction { - - @Inject - public SynchroUserProfileAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(UserService.INDEX, UserService.PROFILE_TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - addValidationListener(this::onValidate); - - synchroService.register(this); - } - - - protected void onValidate(String id, JsonNode source, SynchroActionResult result) { - - String issuer = getIssuer(source); - if (!Objects.equals(issuer, id)) { - throw new AccessDeniedException(String.format("Could not save this document: id must be equals to issuer.")); - } - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/user/SynchroUserSettingsAction.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/user/SynchroUserSettingsAction.java deleted file mode 100644 index 2b238c0b646d10d79edf0219f936c71b5b7fab47..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/synchro/user/SynchroUserSettingsAction.java +++ /dev/null @@ -1,65 +0,0 @@ -package org.duniter.elasticsearch.user.synchro.user; - -/*- - * #%L - * Duniter4j :: ElasticSearch User plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.AccessDeniedException; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.synchro.SynchroActionResult; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.dao.profile.UserSettingsDao; -import org.duniter.elasticsearch.user.service.UserService; -import org.duniter.elasticsearch.synchro.AbstractSynchroAction; -import org.elasticsearch.common.inject.Inject; - -import java.util.Objects; - -public class SynchroUserSettingsAction extends AbstractSynchroAction { - - @Inject - public SynchroUserSettingsAction(Duniter4jClient client, - PluginSettings pluginSettings, - CryptoService cryptoService, - ThreadPool threadPool, - SynchroService synchroService) { - super(UserService.INDEX, UserService.SETTINGS_TYPE, client, pluginSettings.getDelegate(), cryptoService, threadPool); - - setEnableUpdate(true); // with update - - addValidationListener(this::onValidate); - - synchroService.register(this); - } - - protected void onValidate(String id, JsonNode source, SynchroActionResult result) { - - String issuer = getIssuer(source); - if (!Objects.equals(issuer, id)) { - throw new AccessDeniedException(String.format("Could not save this document: id must be equals to issuer.")); - } - } -} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebSocketModule.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebSocketModule.java deleted file mode 100644 index 3aaf2d8ed86ccae793a1f76a785cea8d0c7500eb..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebSocketModule.java +++ /dev/null @@ -1,37 +0,0 @@ -package org.duniter.elasticsearch.user.websocket; - -/* - * #%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.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class WebSocketModule extends AbstractModule implements Module { - - @Override protected void configure() { - bind(WebsocketUserEventEndPoint.Init.class).asEagerSingleton(); - } - - - /* protected methods */ - -} \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java deleted file mode 100644 index 6f2c898394748fc3dad5ee8d37f44c65591ef929..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java +++ /dev/null @@ -1,151 +0,0 @@ -package org.duniter.elasticsearch.user.websocket; - -/* - * #%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% - */ - -/* - Copyright 2015 ForgeRock AS - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -import com.fasterxml.jackson.annotation.JsonInclude; -import com.fasterxml.jackson.databind.ObjectMapper; -import org.duniter.core.client.model.bma.Constants; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.user.PluginSettings; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.service.UserEventService; -import org.duniter.elasticsearch.websocket.WebSocketServer; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.Loggers; -import org.nuiton.i18n.I18n; - -import javax.websocket.*; -import javax.websocket.server.ServerEndpoint; -import java.io.IOException; -import java.util.Locale; -import java.util.regex.Pattern; - -@ServerEndpoint(value = "/event/user/{pubkey}/{locale}") -public class WebsocketUserEventEndPoint implements UserEventService.UserEventListener { - - public static Locale defaultLocale; - public static ObjectMapper mapper; - - public static class Init { - - @Inject - public Init(WebSocketServer webSocketServer, PluginSettings pluginSettings) { - webSocketServer.addEndPoint(WebsocketUserEventEndPoint.class); - defaultLocale = pluginSettings.getI18nLocale(); - if (defaultLocale == null) defaultLocale = new Locale("en", "GB"); - - // Define a static mapper - mapper = new ObjectMapper(); - mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL); - } - } - - private static final String PATH_PARAM_PUBKEY = "pubkey"; - private static final String PATH_PARAM_LOCALE = "locale"; - private final static Pattern PUBKEY_PATTERN = Pattern.compile(Constants.Regex.PUBKEY); - - private final ESLogger log = Loggers.getLogger("duniter.ws.user.event"); - private Session session; - private String pubkey; - private Locale locale; - - @OnOpen - public void onOpen(Session session) { - this.session = session; - this.pubkey = session.getPathParameters() != null ? session.getPathParameters().get(PATH_PARAM_PUBKEY) : null; - this.locale = session.getPathParameters() != null ? new Locale(session.getPathParameters().get(PATH_PARAM_LOCALE)) : defaultLocale; - - if (StringUtils.isBlank(pubkey) || !PUBKEY_PATTERN.matcher(pubkey).matches()) { - try { - this.session.close(new CloseReason(CloseReason.CloseCodes.CANNOT_ACCEPT, "Invalid pubkey")); - } catch (IOException e) { - // silent - } - return; - } - - log.debug(I18n.t("duniter4j.ws.user.open", pubkey, session.getId(), locale.toString())); - UserEventService.registerListener(this); - } - - @Override - public void onEvent(UserEvent event) { - try { - UserEvent copy = new UserEvent(event); - copy.setMessage(copy.getLocalizedMessage(locale)); - String json = mapper.writeValueAsString(copy); - - // Force to serialized 'id' (skip @JsonIgnore) - fix #12 - json = "{\"id\":\""+event.getId()+"\"," + json.substring(1); - - session.getAsyncRemote().sendText(json); - } catch(IOException e) { - // silent - } - } - - @Override - public String getId() { - return session == null ? null : session.getId(); - } - - @Override - public String getPubkey() { - return pubkey; - } - - @OnMessage - public void onMessage(String message) { - log.debug("Received message: "+message); - } - - @OnClose - public void onClose(CloseReason reason) { - log.debug("Closing websocket: "+reason); - UserEventService.unregisterListener(this); - this.session = null; - } - - @OnError - public void onError(Throwable t) { - log.error("Error on websocket "+(session == null ? null : session.getId()), t); - } - -} diff --git a/duniter4j-es-user/src/main/misc/curl_test.sh b/duniter4j-es-user/src/main/misc/curl_test.sh deleted file mode 100755 index be2865e117420f1ee1608f68d6226b9db2f5e76f..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/misc/curl_test.sh +++ /dev/null @@ -1,121 +0,0 @@ -#!/bin/sh - -echo "--- COUNT query --- " -curl -XPOST "http://127.0.0.1:9200/user/event/_count?pretty" -d' -{ - query: { - bool: { - must: [ - {term: {recipient: "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"}}, - {range: {time: {gt: 0}}} - ], - must_not: {terms: { "code": ["TX_SENT"]}} - } - }, - sort : [ - { "time" : {"order" : "desc"}} - ], - from: 0, - size: 100, - _source: false -}' - -echo "--- IDS query --- " -curl -XPOST "http://127.0.0.1:9200/market/comment/_search?pretty" -d' -{ - query: { - terms: { - _id: [ "AVlA2v8sW3j-KIPA7pm8" ] - } - }, - from: 0, - size: 100 -}' - -echo "--- COUNT query --- " -curl -XPOST "http://127.0.0.1:9200/market/comment/_search?pretty" -d' -{ - query: { - terms: { - reply_to: ["AVlEmFhF1r62y3TgqdyR"] - } - }, - "aggs" : { - "reply_tos" : { - "terms" : { "field" : "reply_to" } - } - }, - size: 0 -}' - -echo "--- COUNT + GET query --- " -curl -XPOST "http://127.0.0.1:9200/market/comment/_search?pretty" -d' -{ - query: { - terms: { - record: [ "AVk_pr_49ItF-SEayNy1" ] - } - }, - "aggs" : { - "reply_tos" : { - "terms" : { "field" : "reply_to" } - } - }, - sort : [ - { "time" : {"order" : "desc"}} - ], - from: 0, - size: 5 -}' - -echo "--- GET user event --- " -curl -XPOST "http://127.0.0.1:9200/user/event/_search?pretty" -d' -{ - query: { - bool: { - filter: [ - {term: { recipient: "5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"}}, - {term: { code: "TX_RECEIVED"}} - ] - } - }, - sort : [ - { "time" : {"order" : "desc"}} - ], - from: 0, - size: 3 -}' - - -echo "--- GET market pictures content_type--- " -curl -XPOST "http://data.gtest.duniter.fr/user/profile/_search?pretty" -d' -{ - query: { - constant_score: { - filter: [ - {terms: { issuer: ["5ocqzyDMMWf1V8bsoNhWb1iNwax1e9M7VTUN6navs8of"]}} - ] - } - }, - from: 0, - 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/index.sh b/duniter4j-es-user/src/main/misc/index.sh deleted file mode 100755 index 02b66934dd45e338506a7adc1e25500fabac9710..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/misc/index.sh +++ /dev/null @@ -1,14 +0,0 @@ -#!/bin/sh - -curl -X POST '192.168.0.28:9200/places' -d '{ - "mappings": { - "place": { - "properties": { - "id": {"type": "double"}, - "name": {"type": "string"}, - "type": {"type": "string"}, - "location": {"type": "geo_point"} - } - } - } -}' \ No newline at end of file diff --git a/duniter4j-es-user/src/main/misc/page-categories-naf2008_liste_n5.ods b/duniter4j-es-user/src/main/misc/page-categories-naf2008_liste_n5.ods deleted file mode 100644 index acc6ea091339d82dacf7533c99e3cf9876599984..0000000000000000000000000000000000000000 Binary files a/duniter4j-es-user/src/main/misc/page-categories-naf2008_liste_n5.ods and /dev/null differ diff --git a/duniter4j-es-user/src/main/misc/query_many_indices.sh b/duniter4j-es-user/src/main/misc/query_many_indices.sh deleted file mode 100755 index 22a30bdbb72556b7fd04ce51f5cdf9bd07b25c1c..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/misc/query_many_indices.sh +++ /dev/null @@ -1,17 +0,0 @@ -#!/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 deleted file mode 100644 index 3c47a593117794059ed2628a702289ef4d3798b0..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties +++ /dev/null @@ -1,21 +0,0 @@ -duniter.admin.event.subject.ERROR=[%s] Error message -duniter.admin.event.subject.INFO=[%s] Information message -duniter.admin.event.subject.WARN=[%s] Warning message -duniter.user.event.CERT_RECEIVED=You have received a certification from %2$s. -duniter.user.event.CERT_SENT=Your certification to %2$s was executed. -duniter.user.event.INVITATION_TO_CERTIFY=%2$s invites you to certify an identity. -duniter.user.event.MEMBER_ACTIVE=Your membership to %1$s has been renewed successfully. -duniter.user.event.MEMBER_EXCLUDE= -duniter.user.event.MEMBER_JOIN=You are now a member of currency %1$s\! -duniter.user.event.MEMBER_LEAVE=You are not a member anymore of currency %1$s\! -duniter.user.event.MEMBER_REVOKE= -duniter.user.event.MESSAGE_RECEIVED=You received a message from %2$s. -duniter.user.event.NODE_BMA_DOWN=Duniter node [%1$s\:%2$s] is DOWN\: no access from ES node [%3$s]. Last connexion at %4$s. Blockchain indexation waiting. -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 deleted file mode 100644 index ec9ace345c37859ec4eebeece40f906bd7d5233d..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties +++ /dev/null @@ -1,21 +0,0 @@ -duniter.admin.event.subject.ERROR=%s Message d'erreur -duniter.admin.event.subject.INFO=%s Message d'information -duniter.admin.event.subject.WARN=%s Message d'avertissement -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.INVITATION_TO_CERTIFY=%2$s vous invite à certifier une identité. -duniter.user.event.MEMBER_ACTIVE=Votre adhésion comme membre a bien été renouvellée. -duniter.user.event.MEMBER_EXCLUDE=Votre adhésion comme membre n'est plus valide, faute de non renouvellement ou par manque de certifications. -duniter.user.event.MEMBER_JOIN=Vous êtes maintenant membre de la monnaie %1$s \! -duniter.user.event.MEMBER_LEAVE=Votre adhésion a pris fin, suite à votre demande. -duniter.user.event.MEMBER_REVOKE=Votre compte membre a été révoquée. Il ne pourra plus devenir membre. -duniter.user.event.MESSAGE_RECEIVED=Vous avez reçu un message de %2$s. -duniter.user.event.NODE_BMA_DOWN=Noeud Duniter [%1$s\:%2$s] non joignable, depuis le noeud ES API [%3$s]. Dernière connexion à %4$s. Indexation de blockchain en attente. -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/duniter4j-es-user/src/main/resources/page-categories-bulk-insert.json b/duniter4j-es-user/src/main/resources/page-categories-bulk-insert.json deleted file mode 100644 index 3ee70de3d3aa58dfd3ff01f1a578a2f917508d66..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/resources/page-categories-bulk-insert.json +++ /dev/null @@ -1,1506 +0,0 @@ -{"index": {"_id": "A"}} -{"name": "Agriculture, sylviculture et pêche", "parent": null} -{"index": {"_id": "01.11Z"}} -{"name": "Culture de céréales (à l'exception du riz), de légumineuses et de graines oléagineuses", "parent": "A"} -{"index": {"_id": "01.12Z"}} -{"name": "Culture du riz", "parent": "A"} -{"index": {"_id": "01.13Z"}} -{"name": "Culture de légumes, de melons, de racines et de tubercules", "parent": "A"} -{"index": {"_id": "01.14Z"}} -{"name": "Culture de la canne à sucre", "parent": "A"} -{"index": {"_id": "01.15Z"}} -{"name": "Culture du tabac", "parent": "A"} -{"index": {"_id": "01.16Z"}} -{"name": "Culture de plantes à fibres", "parent": "A"} -{"index": {"_id": "01.19Z"}} -{"name": "Autres cultures non permanentes", "parent": "A"} -{"index": {"_id": "01.21Z"}} -{"name": "Culture de la vigne", "parent": "A"} -{"index": {"_id": "01.22Z"}} -{"name": "Culture de fruits tropicaux et subtropicaux", "parent": "A"} -{"index": {"_id": "01.23Z"}} -{"name": "Culture d'agrumes", "parent": "A"} -{"index": {"_id": "01.24Z"}} -{"name": "Culture de fruits à pépins et à noyau", "parent": "A"} -{"index": {"_id": "01.25Z"}} -{"name": "Culture d'autres fruits d'arbres ou d'arbustes et de fruits à coque", "parent": "A"} -{"index": {"_id": "01.26Z"}} -{"name": "Culture de fruits oléagineux", "parent": "A"} -{"index": {"_id": "01.27Z"}} -{"name": "Culture de plantes à boissons", "parent": "A"} -{"index": {"_id": "01.28Z"}} -{"name": "Culture de plantes à épices, aromatiques, médicinales et pharmaceutiques", "parent": "A"} -{"index": {"_id": "01.29Z"}} -{"name": "Autres cultures permanentes", "parent": "A"} -{"index": {"_id": "01.30Z"}} -{"name": "Reproduction de plantes", "parent": "A"} -{"index": {"_id": "01.41Z"}} -{"name": "Élevage de vaches laitières", "parent": "A"} -{"index": {"_id": "01.42Z"}} -{"name": "Élevage d'autres bovins et de buffles", "parent": "A"} -{"index": {"_id": "01.43Z"}} -{"name": "Élevage de chevaux et d'autres équidés", "parent": "A"} -{"index": {"_id": "01.44Z"}} -{"name": "Élevage de chameaux et d'autres camélidés", "parent": "A"} -{"index": {"_id": "01.45Z"}} -{"name": "Élevage d'ovins et de caprins", "parent": "A"} -{"index": {"_id": "01.46Z"}} -{"name": "Élevage de porcins", "parent": "A"} -{"index": {"_id": "01.47Z"}} -{"name": "Élevage de volailles", "parent": "A"} -{"index": {"_id": "01.49Z"}} -{"name": "Élevage d'autres animaux", "parent": "A"} -{"index": {"_id": "01.50Z"}} -{"name": "Culture et élevage associés", "parent": "A"} -{"index": {"_id": "01.61Z"}} -{"name": "Activités de soutien aux cultures", "parent": "A"} -{"index": {"_id": "01.62Z"}} -{"name": "Activités de soutien à la production animale", "parent": "A"} -{"index": {"_id": "01.63Z"}} -{"name": "Traitement primaire des récoltes", "parent": "A"} -{"index": {"_id": "01.64Z"}} -{"name": "Traitement des semences", "parent": "A"} -{"index": {"_id": "01.70Z"}} -{"name": "Chasse, piégeage et services annexes", "parent": "A"} -{"index": {"_id": "02.10Z"}} -{"name": "Sylviculture et autres activités forestières", "parent": "A"} -{"index": {"_id": "02.20Z"}} -{"name": "Exploitation forestière", "parent": "A"} -{"index": {"_id": "02.30Z"}} -{"name": "Récolte de produits forestiers non ligneux poussant à l'état sauvage", "parent": "A"} -{"index": {"_id": "02.40Z"}} -{"name": "Services de soutien à l'exploitation forestière", "parent": "A"} -{"index": {"_id": "03.11Z"}} -{"name": "Pêche en mer", "parent": "A"} -{"index": {"_id": "03.12Z"}} -{"name": "Pêche en eau douce", "parent": "A"} -{"index": {"_id": "03.21Z"}} -{"name": "Aquaculture en mer", "parent": "A"} -{"index": {"_id": "03.22Z"}} -{"name": "Aquaculture en eau douce", "parent": "A"} -{"index": {"_id": "B"}} -{"name": "Industries extractives", "parent": null} -{"index": {"_id": "05.10Z"}} -{"name": "Extraction de houille", "parent": "B"} -{"index": {"_id": "05.20Z"}} -{"name": "Extraction de lignite", "parent": "B"} -{"index": {"_id": "06.10Z"}} -{"name": "Extraction de pétrole brut", "parent": "B"} -{"index": {"_id": "06.20Z"}} -{"name": "Extraction de gaz naturel", "parent": "B"} -{"index": {"_id": "07.10Z"}} -{"name": "Extraction de minerais de fer", "parent": "B"} -{"index": {"_id": "07.21Z"}} -{"name": "Extraction de minerais d'uranium et de thorium", "parent": "B"} -{"index": {"_id": "07.29Z"}} -{"name": "Extraction d'autres minerais de métaux non ferreux", "parent": "B"} -{"index": {"_id": "08.11Z"}} -{"name": "Extraction de pierres ornementales et de construction, de calcaire industriel, de gypse, de craie et d'ardoise", "parent": "B"} -{"index": {"_id": "08.12Z"}} -{"name": "Exploitation de gravières et sablières, extraction d'argiles et de kaolin", "parent": "B"} -{"index": {"_id": "08.91Z"}} -{"name": "Extraction des minéraux chimiques et d'engrais minéraux", "parent": "B"} -{"index": {"_id": "08.92Z"}} -{"name": "Extraction de tourbe", "parent": "B"} -{"index": {"_id": "08.93Z"}} -{"name": "Production de sel", "parent": "B"} -{"index": {"_id": "08.99Z"}} -{"name": "Autres activités extractives n.c.a.", "parent": "B"} -{"index": {"_id": "09.10Z"}} -{"name": "Activités de soutien à l'extraction d'hydrocarbures", "parent": "B"} -{"index": {"_id": "09.90Z"}} -{"name": "Activités de soutien aux autres industries extractives", "parent": "B"} -{"index": {"_id": "C"}} -{"name": "Industrie manufacturière", "parent": null} -{"index": {"_id": "10.11Z"}} -{"name": "Transformation et conservation de la viande de boucherie", "parent": "C"} -{"index": {"_id": "10.12Z"}} -{"name": "Transformation et conservation de la viande de volaille", "parent": "C"} -{"index": {"_id": "10.13A"}} -{"name": "Préparation industrielle de produits à base de viande", "parent": "C"} -{"index": {"_id": "10.13B"}} -{"name": "Charcuterie", "parent": "C"} -{"index": {"_id": "10.20Z"}} -{"name": "Transformation et conservation de poisson, de crustacés et de mollusques", "parent": "C"} -{"index": {"_id": "10.31Z"}} -{"name": "Transformation et conservation de pommes de terre", "parent": "C"} -{"index": {"_id": "10.32Z"}} -{"name": "Préparation de jus de fruits et légumes", "parent": "C"} -{"index": {"_id": "10.39A"}} -{"name": "Autre transformation et conservation de légumes", "parent": "C"} -{"index": {"_id": "10.39B"}} -{"name": "Transformation et conservation de fruits", "parent": "C"} -{"index": {"_id": "10.41A"}} -{"name": "Fabrication d'huiles et graisses brutes", "parent": "C"} -{"index": {"_id": "10.41B"}} -{"name": "Fabrication d'huiles et graisses raffinées", "parent": "C"} -{"index": {"_id": "10.42Z"}} -{"name": "Fabrication de margarine et graisses comestibles similaires", "parent": "C"} -{"index": {"_id": "10.51A"}} -{"name": "Fabrication de lait liquide et de produits frais", "parent": "C"} -{"index": {"_id": "10.51B"}} -{"name": "Fabrication de beurre", "parent": "C"} -{"index": {"_id": "10.51C"}} -{"name": "Fabrication de fromage", "parent": "C"} -{"index": {"_id": "10.51D"}} -{"name": "Fabrication d'autres produits laitiers", "parent": "C"} -{"index": {"_id": "10.52Z"}} -{"name": "Fabrication de glaces et sorbets", "parent": "C"} -{"index": {"_id": "10.61A"}} -{"name": "Meunerie", "parent": "C"} -{"index": {"_id": "10.61B"}} -{"name": "Autres activités du travail des grains", "parent": "C"} -{"index": {"_id": "10.62Z"}} -{"name": "Fabrication de produits amylacés", "parent": "C"} -{"index": {"_id": "10.71A"}} -{"name": "Fabrication industrielle de pain et de pâtisserie fraîche", "parent": "C"} -{"index": {"_id": "10.71B"}} -{"name": "Cuisson de produits de boulangerie", "parent": "C"} -{"index": {"_id": "10.71C"}} -{"name": "Boulangerie et boulangerie-pâtisserie", "parent": "C"} -{"index": {"_id": "10.71D"}} -{"name": "Pâtisserie", "parent": "C"} -{"index": {"_id": "10.72Z"}} -{"name": "Fabrication de biscuits, biscottes et pâtisseries de conservation", "parent": "C"} -{"index": {"_id": "10.73Z"}} -{"name": "Fabrication de pâtes alimentaires", "parent": "C"} -{"index": {"_id": "10.81Z"}} -{"name": "Fabrication de sucre", "parent": "C"} -{"index": {"_id": "10.82Z"}} -{"name": "Fabrication de cacao, chocolat et de produits de confiserie", "parent": "C"} -{"index": {"_id": "10.83Z"}} -{"name": "Transformation du thé et du café", "parent": "C"} -{"index": {"_id": "10.84Z"}} -{"name": "Fabrication de condiments et assaisonnements", "parent": "C"} -{"index": {"_id": "10.85Z"}} -{"name": "Fabrication de plats préparés", "parent": "C"} -{"index": {"_id": "10.86Z"}} -{"name": "Fabrication d'aliments homogénéisés et diététiques", "parent": "C"} -{"index": {"_id": "10.89Z"}} -{"name": "Fabrication d'autres produits alimentaires n.c.a.", "parent": "C"} -{"index": {"_id": "10.91Z"}} -{"name": "Fabrication d'aliments pour animaux de ferme", "parent": "C"} -{"index": {"_id": "10.92Z"}} -{"name": "Fabrication d'aliments pour animaux de compagnie", "parent": "C"} -{"index": {"_id": "11.01Z"}} -{"name": "Production de boissons alcooliques distillées", "parent": "C"} -{"index": {"_id": "11.02A"}} -{"name": "Fabrication de vins effervescents", "parent": "C"} -{"index": {"_id": "11.02B"}} -{"name": "Vinification", "parent": "C"} -{"index": {"_id": "11.03Z"}} -{"name": "Fabrication de cidre et de vins de fruits", "parent": "C"} -{"index": {"_id": "11.04Z"}} -{"name": "Production d'autres boissons fermentées non distillées", "parent": "C"} -{"index": {"_id": "11.05Z"}} -{"name": "Fabrication de bière", "parent": "C"} -{"index": {"_id": "11.06Z"}} -{"name": "Fabrication de malt", "parent": "C"} -{"index": {"_id": "11.07A"}} -{"name": "Industrie des eaux de table", "parent": "C"} -{"index": {"_id": "11.07B"}} -{"name": "Production de boissons rafraîchissantes", "parent": "C"} -{"index": {"_id": "12.00Z"}} -{"name": "Fabrication de produits à base de tabac", "parent": "C"} -{"index": {"_id": "13.10Z"}} -{"name": "Préparation de fibres textiles et filature", "parent": "C"} -{"index": {"_id": "13.20Z"}} -{"name": "Tissage", "parent": "C"} -{"index": {"_id": "13.30Z"}} -{"name": "Ennoblissement textile", "parent": "C"} -{"index": {"_id": "13.91Z"}} -{"name": "Fabrication d'étoffes à mailles", "parent": "C"} -{"index": {"_id": "13.92Z"}} -{"name": "Fabrication d'articles textiles, sauf habillement", "parent": "C"} -{"index": {"_id": "13.93Z"}} -{"name": "Fabrication de tapis et moquettes", "parent": "C"} -{"index": {"_id": "13.94Z"}} -{"name": "Fabrication de ficelles, cordes et filets", "parent": "C"} -{"index": {"_id": "13.95Z"}} -{"name": "Fabrication de non-tissés, sauf habillement", "parent": "C"} -{"index": {"_id": "13.96Z"}} -{"name": "Fabrication d'autres textiles techniques et industriels", "parent": "C"} -{"index": {"_id": "13.99Z"}} -{"name": "Fabrication d'autres textiles n.c.a.", "parent": "C"} -{"index": {"_id": "14.11Z"}} -{"name": "Fabrication de vêtements en cuir", "parent": "C"} -{"index": {"_id": "14.12Z"}} -{"name": "Fabrication de vêtements de travail", "parent": "C"} -{"index": {"_id": "14.13Z"}} -{"name": "Fabrication de vêtements de dessus", "parent": "C"} -{"index": {"_id": "14.14Z"}} -{"name": "Fabrication de vêtements de dessous", "parent": "C"} -{"index": {"_id": "14.19Z"}} -{"name": "Fabrication d'autres vêtements et accessoires", "parent": "C"} -{"index": {"_id": "14.20Z"}} -{"name": "Fabrication d'articles en fourrure", "parent": "C"} -{"index": {"_id": "14.31Z"}} -{"name": "Fabrication d'articles chaussants à mailles", "parent": "C"} -{"index": {"_id": "14.39Z"}} -{"name": "Fabrication d'autres articles à mailles", "parent": "C"} -{"index": {"_id": "15.11Z"}} -{"name": "Apprêt et tannage des cuirs ; préparation et teinture des fourrures", "parent": "C"} -{"index": {"_id": "15.12Z"}} -{"name": "Fabrication d'articles de voyage, de maroquinerie et de sellerie", "parent": "C"} -{"index": {"_id": "15.20Z"}} -{"name": "Fabrication de chaussures", "parent": "C"} -{"index": {"_id": "16.10A"}} -{"name": "Sciage et rabotage du bois, hors imprégnation", "parent": "C"} -{"index": {"_id": "16.10B"}} -{"name": "Imprégnation du bois", "parent": "C"} -{"index": {"_id": "16.21Z"}} -{"name": "Fabrication de placage et de panneaux de bois", "parent": "C"} -{"index": {"_id": "16.22Z"}} -{"name": "Fabrication de parquets assemblés", "parent": "C"} -{"index": {"_id": "16.23Z"}} -{"name": "Fabrication de charpentes et d'autres menuiseries", "parent": "C"} -{"index": {"_id": "16.24Z"}} -{"name": "Fabrication d'emballages en bois", "parent": "C"} -{"index": {"_id": "16.29Z"}} -{"name": "Fabrication d'objets divers en bois ; fabrication d'objets en liège, vannerie et sparterie", "parent": "C"} -{"index": {"_id": "17.11Z"}} -{"name": "Fabrication de pâte à papier", "parent": "C"} -{"index": {"_id": "17.12Z"}} -{"name": "Fabrication de papier et de carton", "parent": "C"} -{"index": {"_id": "17.21A"}} -{"name": "Fabrication de carton ondulé", "parent": "C"} -{"index": {"_id": "17.21B"}} -{"name": "Fabrication de cartonnages", "parent": "C"} -{"index": {"_id": "17.21C"}} -{"name": "Fabrication d'emballages en papier", "parent": "C"} -{"index": {"_id": "17.22Z"}} -{"name": "Fabrication d'articles en papier à usage sanitaire ou domestique", "parent": "C"} -{"index": {"_id": "17.23Z"}} -{"name": "Fabrication d'articles de papeterie", "parent": "C"} -{"index": {"_id": "17.24Z"}} -{"name": "Fabrication de papiers peints", "parent": "C"} -{"index": {"_id": "17.29Z"}} -{"name": "Fabrication d'autres articles en papier ou en carton", "parent": "C"} -{"index": {"_id": "18.11Z"}} -{"name": "Imprimerie de journaux", "parent": "C"} -{"index": {"_id": "18.12Z"}} -{"name": "Autre imprimerie (labeur)", "parent": "C"} -{"index": {"_id": "18.13Z"}} -{"name": "Activités de pré-presse", "parent": "C"} -{"index": {"_id": "18.14Z"}} -{"name": "Reliure et activités connexes", "parent": "C"} -{"index": {"_id": "18.20Z"}} -{"name": "Reproduction d'enregistrements", "parent": "C"} -{"index": {"_id": "19.10Z"}} -{"name": "Cokéfaction", "parent": "C"} -{"index": {"_id": "19.20Z"}} -{"name": "Raffinage du pétrole", "parent": "C"} -{"index": {"_id": "20.11Z"}} -{"name": "Fabrication de gaz industriels", "parent": "C"} -{"index": {"_id": "20.12Z"}} -{"name": "Fabrication de colorants et de pigments", "parent": "C"} -{"index": {"_id": "20.13A"}} -{"name": "Enrichissement et retraitement de matières nucléaires", "parent": "C"} -{"index": {"_id": "20.13B"}} -{"name": "Fabrication d'autres produits chimiques inorganiques de base n.c.a.", "parent": "C"} -{"index": {"_id": "20.14Z"}} -{"name": "Fabrication d'autres produits chimiques organiques de base", "parent": "C"} -{"index": {"_id": "20.15Z"}} -{"name": "Fabrication de produits azotés et d'engrais", "parent": "C"} -{"index": {"_id": "20.16Z"}} -{"name": "Fabrication de matières plastiques de base", "parent": "C"} -{"index": {"_id": "20.17Z"}} -{"name": "Fabrication de caoutchouc synthétique", "parent": "C"} -{"index": {"_id": "20.20Z"}} -{"name": "Fabrication de pesticides et d'autres produits agrochimiques", "parent": "C"} -{"index": {"_id": "20.30Z"}} -{"name": "Fabrication de peintures, vernis, encres et mastics", "parent": "C"} -{"index": {"_id": "20.41Z"}} -{"name": "Fabrication de savons, détergents et produits d'entretien", "parent": "C"} -{"index": {"_id": "20.42Z"}} -{"name": "Fabrication de parfums et de produits pour la toilette", "parent": "C"} -{"index": {"_id": "20.51Z"}} -{"name": "Fabrication de produits explosifs", "parent": "C"} -{"index": {"_id": "20.52Z"}} -{"name": "Fabrication de colles", "parent": "C"} -{"index": {"_id": "20.53Z"}} -{"name": "Fabrication d'huiles essentielles", "parent": "C"} -{"index": {"_id": "20.59Z"}} -{"name": "Fabrication d'autres produits chimiques n.c.a.", "parent": "C"} -{"index": {"_id": "20.60Z"}} -{"name": "Fabrication de fibres artificielles ou synthétiques", "parent": "C"} -{"index": {"_id": "21.10Z"}} -{"name": "Fabrication de produits pharmaceutiques de base", "parent": "C"} -{"index": {"_id": "21.20Z"}} -{"name": "Fabrication de préparations pharmaceutiques", "parent": "C"} -{"index": {"_id": "22.11Z"}} -{"name": "Fabrication et rechapage de pneumatiques", "parent": "C"} -{"index": {"_id": "22.19Z"}} -{"name": "Fabrication d'autres articles en caoutchouc", "parent": "C"} -{"index": {"_id": "22.21Z"}} -{"name": "Fabrication de plaques, feuilles, tubes et profilés en matières plastiques", "parent": "C"} -{"index": {"_id": "22.22Z"}} -{"name": "Fabrication d'emballages en matières plastiques", "parent": "C"} -{"index": {"_id": "22.23Z"}} -{"name": "Fabrication d'éléments en matières plastiques pour la construction", "parent": "C"} -{"index": {"_id": "22.29A"}} -{"name": "Fabrication de pièces techniques à base de matières plastiques", "parent": "C"} -{"index": {"_id": "22.29B"}} -{"name": "Fabrication de produits de consommation courante en matières plastiques", "parent": "C"} -{"index": {"_id": "23.11Z"}} -{"name": "Fabrication de verre plat", "parent": "C"} -{"index": {"_id": "23.12Z"}} -{"name": "Façonnage et transformation du verre plat", "parent": "C"} -{"index": {"_id": "23.13Z"}} -{"name": "Fabrication de verre creux", "parent": "C"} -{"index": {"_id": "23.14Z"}} -{"name": "Fabrication de fibres de verre", "parent": "C"} -{"index": {"_id": "23.19Z"}} -{"name": "Fabrication et façonnage d'autres articles en verre, y compris verre technique", "parent": "C"} -{"index": {"_id": "23.20Z"}} -{"name": "Fabrication de produits réfractaires", "parent": "C"} -{"index": {"_id": "23.31Z"}} -{"name": "Fabrication de carreaux en céramique", "parent": "C"} -{"index": {"_id": "23.32Z"}} -{"name": "Fabrication de briques, tuiles et produits de construction, en terre cuite", "parent": "C"} -{"index": {"_id": "23.41Z"}} -{"name": "Fabrication d'articles céramiques à usage domestique ou ornemental", "parent": "C"} -{"index": {"_id": "23.42Z"}} -{"name": "Fabrication d'appareils sanitaires en céramique", "parent": "C"} -{"index": {"_id": "23.43Z"}} -{"name": "Fabrication d'isolateurs et pièces isolantes en céramique", "parent": "C"} -{"index": {"_id": "23.44Z"}} -{"name": "Fabrication d'autres produits céramiques à usage technique", "parent": "C"} -{"index": {"_id": "23.49Z"}} -{"name": "Fabrication d'autres produits céramiques", "parent": "C"} -{"index": {"_id": "23.51Z"}} -{"name": "Fabrication de ciment", "parent": "C"} -{"index": {"_id": "23.52Z"}} -{"name": "Fabrication de chaux et plâtre", "parent": "C"} -{"index": {"_id": "23.61Z"}} -{"name": "Fabrication d'éléments en béton pour la construction", "parent": "C"} -{"index": {"_id": "23.62Z"}} -{"name": "Fabrication d'éléments en plâtre pour la construction", "parent": "C"} -{"index": {"_id": "23.63Z"}} -{"name": "Fabrication de béton prêt à l'emploi", "parent": "C"} -{"index": {"_id": "23.64Z"}} -{"name": "Fabrication de mortiers et bétons secs", "parent": "C"} -{"index": {"_id": "23.65Z"}} -{"name": "Fabrication d'ouvrages en fibre-ciment", "parent": "C"} -{"index": {"_id": "23.69Z"}} -{"name": "Fabrication d'autres ouvrages en béton, en ciment ou en plâtre", "parent": "C"} -{"index": {"_id": "23.70Z"}} -{"name": "Taille, façonnage et finissage de pierres", "parent": "C"} -{"index": {"_id": "23.91Z"}} -{"name": "Fabrication de produits abrasifs", "parent": "C"} -{"index": {"_id": "23.99Z"}} -{"name": "Fabrication d'autres produits minéraux non métalliques n.c.a.", "parent": "C"} -{"index": {"_id": "24.10Z"}} -{"name": "Sidérurgie", "parent": "C"} -{"index": {"_id": "24.20Z"}} -{"name": "Fabrication de tubes, tuyaux, profilés creux et accessoires correspondants en acier", "parent": "C"} -{"index": {"_id": "24.31Z"}} -{"name": "Étirage à froid de barres", "parent": "C"} -{"index": {"_id": "24.32Z"}} -{"name": "Laminage à froid de feuillards", "parent": "C"} -{"index": {"_id": "24.33Z"}} -{"name": "Profilage à froid par formage ou pliage", "parent": "C"} -{"index": {"_id": "24.34Z"}} -{"name": "Tréfilage à froid", "parent": "C"} -{"index": {"_id": "24.41Z"}} -{"name": "Production de métaux précieux", "parent": "C"} -{"index": {"_id": "24.42Z"}} -{"name": "Métallurgie de l'aluminium", "parent": "C"} -{"index": {"_id": "24.43Z"}} -{"name": "Métallurgie du plomb, du zinc ou de l'étain", "parent": "C"} -{"index": {"_id": "24.44Z"}} -{"name": "Métallurgie du cuivre", "parent": "C"} -{"index": {"_id": "24.45Z"}} -{"name": "Métallurgie des autres métaux non ferreux", "parent": "C"} -{"index": {"_id": "24.46Z"}} -{"name": "Élaboration et transformation de matières nucléaires", "parent": "C"} -{"index": {"_id": "24.51Z"}} -{"name": "Fonderie de fonte", "parent": "C"} -{"index": {"_id": "24.52Z"}} -{"name": "Fonderie d'acier", "parent": "C"} -{"index": {"_id": "24.53Z"}} -{"name": "Fonderie de métaux légers", "parent": "C"} -{"index": {"_id": "24.54Z"}} -{"name": "Fonderie d'autres métaux non ferreux", "parent": "C"} -{"index": {"_id": "25.11Z"}} -{"name": "Fabrication de structures métalliques et de parties de structures", "parent": "C"} -{"index": {"_id": "25.12Z"}} -{"name": "Fabrication de portes et fenêtres en métal", "parent": "C"} -{"index": {"_id": "25.21Z"}} -{"name": "Fabrication de radiateurs et de chaudières pour le chauffage central", "parent": "C"} -{"index": {"_id": "25.29Z"}} -{"name": "Fabrication d'autres réservoirs, citernes et conteneurs métalliques", "parent": "C"} -{"index": {"_id": "25.30Z"}} -{"name": "Fabrication de générateurs de vapeur, à l'exception des chaudières pour le chauffage central", "parent": "C"} -{"index": {"_id": "25.40Z"}} -{"name": "Fabrication d'armes et de munitions", "parent": "C"} -{"index": {"_id": "25.50A"}} -{"name": "Forge, estampage, matriçage ; métallurgie des poudres", "parent": "C"} -{"index": {"_id": "25.50B"}} -{"name": "Découpage, emboutissage", "parent": "C"} -{"index": {"_id": "25.61Z"}} -{"name": "Traitement et revêtement des métaux", "parent": "C"} -{"index": {"_id": "25.62A"}} -{"name": "Décolletage", "parent": "C"} -{"index": {"_id": "25.62B"}} -{"name": "Mécanique industrielle", "parent": "C"} -{"index": {"_id": "25.71Z"}} -{"name": "Fabrication de coutellerie", "parent": "C"} -{"index": {"_id": "25.72Z"}} -{"name": "Fabrication de serrures et de ferrures", "parent": "C"} -{"index": {"_id": "25.73A"}} -{"name": "Fabrication de moules et modèles", "parent": "C"} -{"index": {"_id": "25.73B"}} -{"name": "Fabrication d'autres outillages", "parent": "C"} -{"index": {"_id": "25.91Z"}} -{"name": "Fabrication de fûts et emballages métalliques similaires", "parent": "C"} -{"index": {"_id": "25.92Z"}} -{"name": "Fabrication d'emballages métalliques légers", "parent": "C"} -{"index": {"_id": "25.93Z"}} -{"name": "Fabrication d'articles en fils métalliques, de chaînes et de ressorts", "parent": "C"} -{"index": {"_id": "25.94Z"}} -{"name": "Fabrication de vis et de boulons", "parent": "C"} -{"index": {"_id": "25.99A"}} -{"name": "Fabrication d'articles métalliques ménagers", "parent": "C"} -{"index": {"_id": "25.99B"}} -{"name": "Fabrication d'autres articles métalliques", "parent": "C"} -{"index": {"_id": "26.11Z"}} -{"name": "Fabrication de composants électroniques", "parent": "C"} -{"index": {"_id": "26.12Z"}} -{"name": "Fabrication de cartes électroniques assemblées", "parent": "C"} -{"index": {"_id": "26.20Z"}} -{"name": "Fabrication d'ordinateurs et d'équipements périphériques", "parent": "C"} -{"index": {"_id": "26.30Z"}} -{"name": "Fabrication d'équipements de communication", "parent": "C"} -{"index": {"_id": "26.40Z"}} -{"name": "Fabrication de produits électroniques grand public", "parent": "C"} -{"index": {"_id": "26.51A"}} -{"name": "Fabrication d'équipements d'aide à la navigation", "parent": "C"} -{"index": {"_id": "26.51B"}} -{"name": "Fabrication d'instrumentation scientifique et technique", "parent": "C"} -{"index": {"_id": "26.52Z"}} -{"name": "Horlogerie", "parent": "C"} -{"index": {"_id": "26.60Z"}} -{"name": "Fabrication d'équipements d'irradiation médicale, d'équipements électromédicaux et électrothérapeutiques", "parent": "C"} -{"index": {"_id": "26.70Z"}} -{"name": "Fabrication de matériels optique et photographique", "parent": "C"} -{"index": {"_id": "26.80Z"}} -{"name": "Fabrication de supports magnétiques et optiques", "parent": "C"} -{"index": {"_id": "27.11Z"}} -{"name": "Fabrication de moteurs, génératrices et transformateurs électriques", "parent": "C"} -{"index": {"_id": "27.12Z"}} -{"name": "Fabrication de matériel de distribution et de commande électrique", "parent": "C"} -{"index": {"_id": "27.20Z"}} -{"name": "Fabrication de piles et d'accumulateurs électriques", "parent": "C"} -{"index": {"_id": "27.31Z"}} -{"name": "Fabrication de câbles de fibres optiques", "parent": "C"} -{"index": {"_id": "27.32Z"}} -{"name": "Fabrication d'autres fils et câbles électroniques ou électriques", "parent": "C"} -{"index": {"_id": "27.33Z"}} -{"name": "Fabrication de matériel d'installation électrique", "parent": "C"} -{"index": {"_id": "27.40Z"}} -{"name": "Fabrication d'appareils d'éclairage électrique", "parent": "C"} -{"index": {"_id": "27.51Z"}} -{"name": "Fabrication d'appareils électroménagers", "parent": "C"} -{"index": {"_id": "27.52Z"}} -{"name": "Fabrication d'appareils ménagers non électriques", "parent": "C"} -{"index": {"_id": "27.90Z"}} -{"name": "Fabrication d'autres matériels électriques", "parent": "C"} -{"index": {"_id": "28.11Z"}} -{"name": "Fabrication de moteurs et turbines, à l'exception des moteurs d'avions et de véhicules", "parent": "C"} -{"index": {"_id": "28.12Z"}} -{"name": "Fabrication d'équipements hydrauliques et pneumatiques", "parent": "C"} -{"index": {"_id": "28.13Z"}} -{"name": "Fabrication d'autres pompes et compresseurs", "parent": "C"} -{"index": {"_id": "28.14Z"}} -{"name": "Fabrication d'autres articles de robinetterie", "parent": "C"} -{"index": {"_id": "28.15Z"}} -{"name": "Fabrication d'engrenages et d'organes mécaniques de transmission", "parent": "C"} -{"index": {"_id": "28.21Z"}} -{"name": "Fabrication de fours et brûleurs", "parent": "C"} -{"index": {"_id": "28.22Z"}} -{"name": "Fabrication de matériel de levage et de manutention", "parent": "C"} -{"index": {"_id": "28.23Z"}} -{"name": "Fabrication de machines et d'équipements de bureau (à l'exception des ordinateurs et équipements périphériques)", "parent": "C"} -{"index": {"_id": "28.24Z"}} -{"name": "Fabrication d'outillage portatif à moteur incorporé", "parent": "C"} -{"index": {"_id": "28.25Z"}} -{"name": "Fabrication d'équipements aérauliques et frigorifiques industriels", "parent": "C"} -{"index": {"_id": "28.29A"}} -{"name": "Fabrication d'équipements d'emballage, de conditionnement et de pesage", "parent": "C"} -{"index": {"_id": "28.29B"}} -{"name": "Fabrication d'autres machines d'usage général", "parent": "C"} -{"index": {"_id": "28.30Z"}} -{"name": "Fabrication de machines agricoles et forestières", "parent": "C"} -{"index": {"_id": "28.41Z"}} -{"name": "Fabrication de machines-outils pour le travail des métaux", "parent": "C"} -{"index": {"_id": "28.49Z"}} -{"name": "Fabrication d'autres machines-outils", "parent": "C"} -{"index": {"_id": "28.91Z"}} -{"name": "Fabrication de machines pour la métallurgie", "parent": "C"} -{"index": {"_id": "28.92Z"}} -{"name": "Fabrication de machines pour l'extraction ou la construction", "parent": "C"} -{"index": {"_id": "28.93Z"}} -{"name": "Fabrication de machines pour l'industrie agro-alimentaire", "parent": "C"} -{"index": {"_id": "28.94Z"}} -{"name": "Fabrication de machines pour les industries textiles", "parent": "C"} -{"index": {"_id": "28.95Z"}} -{"name": "Fabrication de machines pour les industries du papier et du carton", "parent": "C"} -{"index": {"_id": "28.96Z"}} -{"name": "Fabrication de machines pour le travail du caoutchouc ou des plastiques", "parent": "C"} -{"index": {"_id": "28.99A"}} -{"name": "Fabrication de machines d'imprimerie", "parent": "C"} -{"index": {"_id": "28.99B"}} -{"name": "Fabrication d'autres machines spécialisées", "parent": "C"} -{"index": {"_id": "29.10Z"}} -{"name": "Construction de véhicules automobiles", "parent": "C"} -{"index": {"_id": "29.20Z"}} -{"name": "Fabrication de carrosseries et remorques", "parent": "C"} -{"index": {"_id": "29.31Z"}} -{"name": "Fabrication d'équipements électriques et électroniques automobiles", "parent": "C"} -{"index": {"_id": "29.32Z"}} -{"name": "Fabrication d'autres équipements automobiles", "parent": "C"} -{"index": {"_id": "30.11Z"}} -{"name": "Construction de navires et de structures flottantes", "parent": "C"} -{"index": {"_id": "30.12Z"}} -{"name": "Construction de bateaux de plaisance", "parent": "C"} -{"index": {"_id": "30.20Z"}} -{"name": "Construction de locomotives et d'autre matériel ferroviaire roulant", "parent": "C"} -{"index": {"_id": "30.30Z"}} -{"name": "Construction aéronautique et spatiale", "parent": "C"} -{"index": {"_id": "30.40Z"}} -{"name": "Construction de véhicules militaires de combat", "parent": "C"} -{"index": {"_id": "30.91Z"}} -{"name": "Fabrication de motocycles", "parent": "C"} -{"index": {"_id": "30.92Z"}} -{"name": "Fabrication de bicyclettes et de véhicules pour invalides", "parent": "C"} -{"index": {"_id": "30.99Z"}} -{"name": "Fabrication d'autres équipements de transport n.c.a.", "parent": "C"} -{"index": {"_id": "31.01Z"}} -{"name": "Fabrication de meubles de bureau et de magasin", "parent": "C"} -{"index": {"_id": "31.02Z"}} -{"name": "Fabrication de meubles de cuisine", "parent": "C"} -{"index": {"_id": "31.03Z"}} -{"name": "Fabrication de matelas", "parent": "C"} -{"index": {"_id": "31.09A"}} -{"name": "Fabrication de sièges d'ameublement d'intérieur", "parent": "C"} -{"index": {"_id": "31.09B"}} -{"name": "Fabrication d'autres meubles et industries connexes de l'ameublement", "parent": "C"} -{"index": {"_id": "32.11Z"}} -{"name": "Frappe de monnaie", "parent": "C"} -{"index": {"_id": "32.12Z"}} -{"name": "Fabrication d'articles de joaillerie et bijouterie", "parent": "C"} -{"index": {"_id": "32.13Z"}} -{"name": "Fabrication d'articles de bijouterie fantaisie et articles similaires", "parent": "C"} -{"index": {"_id": "32.20Z"}} -{"name": "Fabrication d'instruments de musique", "parent": "C"} -{"index": {"_id": "32.30Z"}} -{"name": "Fabrication d'articles de sport", "parent": "C"} -{"index": {"_id": "32.40Z"}} -{"name": "Fabrication de jeux et jouets", "parent": "C"} -{"index": {"_id": "32.50A"}} -{"name": "Fabrication de matériel médico-chirurgical et dentaire", "parent": "C"} -{"index": {"_id": "32.50B"}} -{"name": "Fabrication de lunettes", "parent": "C"} -{"index": {"_id": "32.91Z"}} -{"name": "Fabrication d'articles de brosserie", "parent": "C"} -{"index": {"_id": "32.99Z"}} -{"name": "Autres activités manufacturières n.c.a.", "parent": "C"} -{"index": {"_id": "33.11Z"}} -{"name": "Réparation d'ouvrages en métaux", "parent": "C"} -{"index": {"_id": "33.12Z"}} -{"name": "Réparation de machines et équipements mécaniques", "parent": "C"} -{"index": {"_id": "33.13Z"}} -{"name": "Réparation de matériels électroniques et optiques", "parent": "C"} -{"index": {"_id": "33.14Z"}} -{"name": "Réparation d'équipements électriques", "parent": "C"} -{"index": {"_id": "33.15Z"}} -{"name": "Réparation et maintenance navale", "parent": "C"} -{"index": {"_id": "33.16Z"}} -{"name": "Réparation et maintenance d'aéronefs et d'engins spatiaux", "parent": "C"} -{"index": {"_id": "33.17Z"}} -{"name": "Réparation et maintenance d'autres équipements de transport", "parent": "C"} -{"index": {"_id": "33.19Z"}} -{"name": "Réparation d'autres équipements", "parent": "C"} -{"index": {"_id": "33.20A"}} -{"name": "Installation de structures métalliques, chaudronnées et de tuyauterie", "parent": "C"} -{"index": {"_id": "33.20B"}} -{"name": "Installation de machines et équipements mécaniques", "parent": "C"} -{"index": {"_id": "33.20C"}} -{"name": "Conception d'ensemble et assemblage sur site industriel d'équipements de contrôle des processus industriels", "parent": "C"} -{"index": {"_id": "33.20D"}} -{"name": "Installation d'équipements électriques, de matériels électroniques et optiques ou d'autres matériels", "parent": "C"} -{"index": {"_id": "D"}} -{"name": "Production et distribution d'électricité, de gaz, de vapeur et d'air conditionné", "parent": null} -{"index": {"_id": "35.11Z"}} -{"name": "Production d'électricité", "parent": "D"} -{"index": {"_id": "35.12Z"}} -{"name": "Transport d'électricité", "parent": "D"} -{"index": {"_id": "35.13Z"}} -{"name": "Distribution d'électricité", "parent": "D"} -{"index": {"_id": "35.14Z"}} -{"name": "Commerce d'électricité", "parent": "D"} -{"index": {"_id": "35.21Z"}} -{"name": "Production de combustibles gazeux", "parent": "D"} -{"index": {"_id": "35.22Z"}} -{"name": "Distribution de combustibles gazeux par conduites", "parent": "D"} -{"index": {"_id": "35.23Z"}} -{"name": "Commerce de combustibles gazeux par conduites", "parent": "D"} -{"index": {"_id": "35.30Z"}} -{"name": "Production et distribution de vapeur et d'air conditionné", "parent": "D"} -{"index": {"_id": "E"}} -{"name": "Production et distribution d'eau ; assainissement, gestion des déchets et dépollution", "parent": null} -{"index": {"_id": "36.00Z"}} -{"name": "Captage, traitement et distribution d'eau", "parent": "E"} -{"index": {"_id": "37.00Z"}} -{"name": "Collecte et traitement des eaux usées", "parent": "E"} -{"index": {"_id": "38.11Z"}} -{"name": "Collecte des déchets non dangereux", "parent": "E"} -{"index": {"_id": "38.12Z"}} -{"name": "Collecte des déchets dangereux", "parent": "E"} -{"index": {"_id": "38.21Z"}} -{"name": "Traitement et élimination des déchets non dangereux", "parent": "E"} -{"index": {"_id": "38.22Z"}} -{"name": "Traitement et élimination des déchets dangereux", "parent": "E"} -{"index": {"_id": "38.31Z"}} -{"name": "Démantèlement d'épaves", "parent": "E"} -{"index": {"_id": "38.32Z"}} -{"name": "Récupération de déchets triés", "parent": "E"} -{"index": {"_id": "39.00Z"}} -{"name": "Dépollution et autres services de gestion des déchets", "parent": "E"} -{"index": {"_id": "F"}} -{"name": "Construction", "parent": null} -{"index": {"_id": "41.10A"}} -{"name": "Promotion immobilière de logements", "parent": "F"} -{"index": {"_id": "41.10B"}} -{"name": "Promotion immobilière de bureaux", "parent": "F"} -{"index": {"_id": "41.10C"}} -{"name": "Promotion immobilière d'autres bâtiments", "parent": "F"} -{"index": {"_id": "41.10D"}} -{"name": "Supports juridiques de programmes", "parent": "F"} -{"index": {"_id": "41.20A"}} -{"name": "Construction de maisons individuelles", "parent": "F"} -{"index": {"_id": "41.20B"}} -{"name": "Construction d'autres bâtiments", "parent": "F"} -{"index": {"_id": "42.11Z"}} -{"name": "Construction de routes et autoroutes", "parent": "F"} -{"index": {"_id": "42.12Z"}} -{"name": "Construction de voies ferrées de surface et souterraines", "parent": "F"} -{"index": {"_id": "42.13A"}} -{"name": "Construction d'ouvrages d'art", "parent": "F"} -{"index": {"_id": "42.13B"}} -{"name": "Construction et entretien de tunnels", "parent": "F"} -{"index": {"_id": "42.21Z"}} -{"name": "Construction de réseaux pour fluides", "parent": "F"} -{"index": {"_id": "42.22Z"}} -{"name": "Construction de réseaux électriques et de télécommunications", "parent": "F"} -{"index": {"_id": "42.91Z"}} -{"name": "Construction d'ouvrages maritimes et fluviaux", "parent": "F"} -{"index": {"_id": "42.99Z"}} -{"name": "Construction d'autres ouvrages de génie civil n.c.a.", "parent": "F"} -{"index": {"_id": "43.11Z"}} -{"name": "Travaux de démolition", "parent": "F"} -{"index": {"_id": "43.12A"}} -{"name": "Travaux de terrassement courants et travaux préparatoires", "parent": "F"} -{"index": {"_id": "43.12B"}} -{"name": "Travaux de terrassement spécialisés ou de grande masse", "parent": "F"} -{"index": {"_id": "43.13Z"}} -{"name": "Forages et sondages", "parent": "F"} -{"index": {"_id": "43.21A"}} -{"name": "Travaux d'installation électrique dans tous locaux", "parent": "F"} -{"index": {"_id": "43.21B"}} -{"name": "Travaux d'installation électrique sur la voie publique", "parent": "F"} -{"index": {"_id": "43.22A"}} -{"name": "Travaux d'installation d'eau et de gaz en tous locaux", "parent": "F"} -{"index": {"_id": "43.22B"}} -{"name": "Travaux d'installation d'équipements thermiques et de climatisation", "parent": "F"} -{"index": {"_id": "43.29A"}} -{"name": "Travaux d'isolation", "parent": "F"} -{"index": {"_id": "43.29B"}} -{"name": "Autres travaux d'installation n.c.a.", "parent": "F"} -{"index": {"_id": "43.31Z"}} -{"name": "Travaux de plâtrerie", "parent": "F"} -{"index": {"_id": "43.32A"}} -{"name": "Travaux de menuiserie bois et PVC", "parent": "F"} -{"index": {"_id": "43.32B"}} -{"name": "Travaux de menuiserie métallique et serrurerie", "parent": "F"} -{"index": {"_id": "43.32C"}} -{"name": "Agencement de lieux de vente", "parent": "F"} -{"index": {"_id": "43.33Z"}} -{"name": "Travaux de revêtement des sols et des murs", "parent": "F"} -{"index": {"_id": "43.34Z"}} -{"name": "Travaux de peinture et vitrerie", "parent": "F"} -{"index": {"_id": "43.39Z"}} -{"name": "Autres travaux de finition", "parent": "F"} -{"index": {"_id": "43.91A"}} -{"name": "Travaux de charpente", "parent": "F"} -{"index": {"_id": "43.91B"}} -{"name": "Travaux de couverture par éléments", "parent": "F"} -{"index": {"_id": "43.99A"}} -{"name": "Travaux d'étanchéification", "parent": "F"} -{"index": {"_id": "43.99B"}} -{"name": "Travaux de montage de structures métalliques", "parent": "F"} -{"index": {"_id": "43.99C"}} -{"name": "Travaux de maçonnerie générale et gros œuvre de bâtiment", "parent": "F"} -{"index": {"_id": "43.99D"}} -{"name": "Autres travaux spécialisés de construction", "parent": "F"} -{"index": {"_id": "43.99E"}} -{"name": "Location avec opérateur de matériel de construction", "parent": "F"} -{"index": {"_id": "G"}} -{"name": "Commerce ; réparation d'automobiles et de motocycles", "parent": null} -{"index": {"_id": "45.11Z"}} -{"name": "Commerce de voitures et de véhicules automobiles légers", "parent": "G"} -{"index": {"_id": "45.19Z"}} -{"name": "Commerce d'autres véhicules automobiles", "parent": "G"} -{"index": {"_id": "45.20A"}} -{"name": "Entretien et réparation de véhicules automobiles légers", "parent": "G"} -{"index": {"_id": "45.20B"}} -{"name": "Entretien et réparation d'autres véhicules automobiles", "parent": "G"} -{"index": {"_id": "45.31Z"}} -{"name": "Commerce de gros d'équipements automobiles", "parent": "G"} -{"index": {"_id": "45.32Z"}} -{"name": "Commerce de détail d'équipements automobiles", "parent": "G"} -{"index": {"_id": "45.40Z"}} -{"name": "Commerce et réparation de motocycles", "parent": "G"} -{"index": {"_id": "46.11Z"}} -{"name": "Intermédiaires du commerce en matières premières agricoles, animaux vivants, matières premières textiles et produits semi-finis", "parent": "G"} -{"index": {"_id": "46.12A"}} -{"name": "Centrales d'achat de carburant", "parent": "G"} -{"index": {"_id": "46.12B"}} -{"name": "Autres intermédiaires du commerce en combustibles, métaux, minéraux et produits chimiques", "parent": "G"} -{"index": {"_id": "46.13Z"}} -{"name": "Intermédiaires du commerce en bois et matériaux de construction", "parent": "G"} -{"index": {"_id": "46.14Z"}} -{"name": "Intermédiaires du commerce en machines, équipements industriels, navires et avions", "parent": "G"} -{"index": {"_id": "46.15Z"}} -{"name": "Intermédiaires du commerce en meubles, articles de ménage et quincaillerie", "parent": "G"} -{"index": {"_id": "46.16Z"}} -{"name": "Intermédiaires du commerce en textiles, habillement, fourrures, chaussures et articles en cuir", "parent": "G"} -{"index": {"_id": "46.17A"}} -{"name": "Centrales d'achat alimentaires", "parent": "G"} -{"index": {"_id": "46.17B"}} -{"name": "Autres intermédiaires du commerce en denrées, boissons et tabac", "parent": "G"} -{"index": {"_id": "46.18Z"}} -{"name": "Intermédiaires spécialisés dans le commerce d'autres produits spécifiques", "parent": "G"} -{"index": {"_id": "46.19A"}} -{"name": "Centrales d'achat non alimentaires", "parent": "G"} -{"index": {"_id": "46.19B"}} -{"name": "Autres intermédiaires du commerce en produits divers", "parent": "G"} -{"index": {"_id": "46.21Z"}} -{"name": "Commerce de gros (commerce interentreprises) de céréales, de tabac non manufacturé, de semences et d'aliments pour le bétail", "parent": "G"} -{"index": {"_id": "46.22Z"}} -{"name": "Commerce de gros (commerce interentreprises) de fleurs et plantes", "parent": "G"} -{"index": {"_id": "46.23Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'animaux vivants", "parent": "G"} -{"index": {"_id": "46.24Z"}} -{"name": "Commerce de gros (commerce interentreprises) de cuirs et peaux", "parent": "G"} -{"index": {"_id": "46.31Z"}} -{"name": "Commerce de gros (commerce interentreprises) de fruits et légumes", "parent": "G"} -{"index": {"_id": "46.32A"}} -{"name": "Commerce de gros (commerce interentreprises) de viandes de boucherie", "parent": "G"} -{"index": {"_id": "46.32B"}} -{"name": "Commerce de gros (commerce interentreprises) de produits à base de viande", "parent": "G"} -{"index": {"_id": "46.32C"}} -{"name": "Commerce de gros (commerce interentreprises) de volailles et gibier", "parent": "G"} -{"index": {"_id": "46.33Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits laitiers, œufs, huiles et matières grasses comestibles", "parent": "G"} -{"index": {"_id": "46.34Z"}} -{"name": "Commerce de gros (commerce interentreprises) de boissons", "parent": "G"} -{"index": {"_id": "46.35Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits à base de tabac", "parent": "G"} -{"index": {"_id": "46.36Z"}} -{"name": "Commerce de gros (commerce interentreprises) de sucre, chocolat et confiserie", "parent": "G"} -{"index": {"_id": "46.37Z"}} -{"name": "Commerce de gros (commerce interentreprises) de café, thé, cacao et épices", "parent": "G"} -{"index": {"_id": "46.38A"}} -{"name": "Commerce de gros (commerce interentreprises) de poissons, crustacés et mollusques", "parent": "G"} -{"index": {"_id": "46.38B"}} -{"name": "Commerce de gros (commerce interentreprises) alimentaire spécialisé divers", "parent": "G"} -{"index": {"_id": "46.39A"}} -{"name": "Commerce de gros (commerce interentreprises) de produits surgelés", "parent": "G"} -{"index": {"_id": "46.39B"}} -{"name": "Commerce de gros (commerce interentreprises) alimentaire non spécialisé", "parent": "G"} -{"index": {"_id": "46.41Z"}} -{"name": "Commerce de gros (commerce interentreprises) de textiles", "parent": "G"} -{"index": {"_id": "46.42Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'habillement et de chaussures", "parent": "G"} -{"index": {"_id": "46.43Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'appareils électroménagers", "parent": "G"} -{"index": {"_id": "46.44Z"}} -{"name": "Commerce de gros (commerce interentreprises) de vaisselle, verrerie et produits d'entretien", "parent": "G"} -{"index": {"_id": "46.45Z"}} -{"name": "Commerce de gros (commerce interentreprises) de parfumerie et de produits de beauté", "parent": "G"} -{"index": {"_id": "46.46Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits pharmaceutiques", "parent": "G"} -{"index": {"_id": "46.47Z"}} -{"name": "Commerce de gros (commerce interentreprises) de meubles, de tapis et d'appareils d'éclairage", "parent": "G"} -{"index": {"_id": "46.48Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'articles d'horlogerie et de bijouterie", "parent": "G"} -{"index": {"_id": "46.49Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'autres biens domestiques", "parent": "G"} -{"index": {"_id": "46.51Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'ordinateurs, d'équipements informatiques périphériques et de logiciels", "parent": "G"} -{"index": {"_id": "46.52Z"}} -{"name": "Commerce de gros (commerce interentreprises) de composants et d'équipements électroniques et de télécommunication", "parent": "G"} -{"index": {"_id": "46.61Z"}} -{"name": "Commerce de gros (commerce interentreprises) de matériel agricole", "parent": "G"} -{"index": {"_id": "46.62Z"}} -{"name": "Commerce de gros (commerce interentreprises) de machines-outils", "parent": "G"} -{"index": {"_id": "46.63Z"}} -{"name": "Commerce de gros (commerce interentreprises) de machines pour l'extraction, la construction et le génie civil", "parent": "G"} -{"index": {"_id": "46.64Z"}} -{"name": "Commerce de gros (commerce interentreprises) de machines pour l'industrie textile et l'habillement", "parent": "G"} -{"index": {"_id": "46.65Z"}} -{"name": "Commerce de gros (commerce interentreprises) de mobilier de bureau", "parent": "G"} -{"index": {"_id": "46.66Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'autres machines et équipements de bureau", "parent": "G"} -{"index": {"_id": "46.69A"}} -{"name": "Commerce de gros (commerce interentreprises) de matériel électrique", "parent": "G"} -{"index": {"_id": "46.69B"}} -{"name": "Commerce de gros (commerce interentreprises) de fournitures et équipements industriels divers", "parent": "G"} -{"index": {"_id": "46.69C"}} -{"name": "Commerce de gros (commerce interentreprises) de fournitures et équipements divers pour le commerce et les services", "parent": "G"} -{"index": {"_id": "46.71Z"}} -{"name": "Commerce de gros (commerce interentreprises) de combustibles et de produits annexes", "parent": "G"} -{"index": {"_id": "46.72Z"}} -{"name": "Commerce de gros (commerce interentreprises) de minerais et métaux", "parent": "G"} -{"index": {"_id": "46.73A"}} -{"name": "Commerce de gros (commerce interentreprises) de bois et de matériaux de construction", "parent": "G"} -{"index": {"_id": "46.73B"}} -{"name": "Commerce de gros (commerce interentreprises) d'appareils sanitaires et de produits de décoration", "parent": "G"} -{"index": {"_id": "46.74A"}} -{"name": "Commerce de gros (commerce interentreprises) de quincaillerie", "parent": "G"} -{"index": {"_id": "46.74B"}} -{"name": "Commerce de gros (commerce interentreprises) de fournitures pour la plomberie et le chauffage", "parent": "G"} -{"index": {"_id": "46.75Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits chimiques", "parent": "G"} -{"index": {"_id": "46.76Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'autres produits intermédiaires", "parent": "G"} -{"index": {"_id": "46.77Z"}} -{"name": "Commerce de gros (commerce interentreprises) de déchets et débris", "parent": "G"} -{"index": {"_id": "46.90Z"}} -{"name": "Commerce de gros (commerce interentreprises) non spécialisé", "parent": "G"} -{"index": {"_id": "47.11A"}} -{"name": "Commerce de détail de produits surgelés", "parent": "G"} -{"index": {"_id": "47.11B"}} -{"name": "Commerce d'alimentation générale", "parent": "G"} -{"index": {"_id": "47.11C"}} -{"name": "Supérettes", "parent": "G"} -{"index": {"_id": "47.11D"}} -{"name": "Supermarchés", "parent": "G"} -{"index": {"_id": "47.11E"}} -{"name": "Magasins multi-commerces", "parent": "G"} -{"index": {"_id": "47.11F"}} -{"name": "Hypermarchés", "parent": "G"} -{"index": {"_id": "47.19A"}} -{"name": "Grands magasins", "parent": "G"} -{"index": {"_id": "47.19B"}} -{"name": "Autres commerces de détail en magasin non spécialisé", "parent": "G"} -{"index": {"_id": "47.21Z"}} -{"name": "Commerce de détail de fruits et légumes en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.22Z"}} -{"name": "Commerce de détail de viandes et de produits à base de viande en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.23Z"}} -{"name": "Commerce de détail de poissons, crustacés et mollusques en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.24Z"}} -{"name": "Commerce de détail de pain, pâtisserie et confiserie en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.25Z"}} -{"name": "Commerce de détail de boissons en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.26Z"}} -{"name": "Commerce de détail de produits à base de tabac en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.29Z"}} -{"name": "Autres commerces de détail alimentaires en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.30Z"}} -{"name": "Commerce de détail de carburants en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.41Z"}} -{"name": "Commerce de détail d'ordinateurs, d'unités périphériques et de logiciels en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.42Z"}} -{"name": "Commerce de détail de matériels de télécommunication en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.43Z"}} -{"name": "Commerce de détail de matériels audio et vidéo en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.51Z"}} -{"name": "Commerce de détail de textiles en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.52A"}} -{"name": "Commerce de détail de quincaillerie, peintures et verres en petites surfaces (moins de 400 m²)", "parent": "G"} -{"index": {"_id": "47.52B"}} -{"name": "Commerce de détail de quincaillerie, peintures et verres en grandes surfaces (400 m² et plus)", "parent": "G"} -{"index": {"_id": "47.53Z"}} -{"name": "Commerce de détail de tapis, moquettes et revêtements de murs et de sols en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.54Z"}} -{"name": "Commerce de détail d'appareils électroménagers en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.59A"}} -{"name": "Commerce de détail de meubles", "parent": "G"} -{"index": {"_id": "47.59B"}} -{"name": "Commerce de détail d'autres équipements du foyer", "parent": "G"} -{"index": {"_id": "47.61Z"}} -{"name": "Commerce de détail de livres en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.62Z"}} -{"name": "Commerce de détail de journaux et papeterie en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.63Z"}} -{"name": "Commerce de détail d'enregistrements musicaux et vidéo en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.64Z"}} -{"name": "Commerce de détail d'articles de sport en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.65Z"}} -{"name": "Commerce de détail de jeux et jouets en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.71Z"}} -{"name": "Commerce de détail d'habillement en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.72A"}} -{"name": "Commerce de détail de la chaussure", "parent": "G"} -{"index": {"_id": "47.72B"}} -{"name": "Commerce de détail de maroquinerie et d'articles de voyage", "parent": "G"} -{"index": {"_id": "47.73Z"}} -{"name": "Commerce de détail de produits pharmaceutiques en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.74Z"}} -{"name": "Commerce de détail d'articles médicaux et orthopédiques en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.75Z"}} -{"name": "Commerce de détail de parfumerie et de produits de beauté en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.76Z"}} -{"name": "Commerce de détail de fleurs, plantes, graines, engrais, animaux de compagnie et aliments pour ces animaux en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.77Z"}} -{"name": "Commerce de détail d'articles d'horlogerie et de bijouterie en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.78A"}} -{"name": "Commerces de détail d'optique", "parent": "G"} -{"index": {"_id": "47.78B"}} -{"name": "Commerces de détail de charbons et combustibles", "parent": "G"} -{"index": {"_id": "47.78C"}} -{"name": "Autres commerces de détail spécialisés divers", "parent": "G"} -{"index": {"_id": "47.79Z"}} -{"name": "Commerce de détail de biens d'occasion en magasin", "parent": "G"} -{"index": {"_id": "47.81Z"}} -{"name": "Commerce de détail alimentaire sur éventaires et marchés", "parent": "G"} -{"index": {"_id": "47.82Z"}} -{"name": "Commerce de détail de textiles, d'habillement et de chaussures sur éventaires et marchés", "parent": "G"} -{"index": {"_id": "47.89Z"}} -{"name": "Autres commerces de détail sur éventaires et marchés", "parent": "G"} -{"index": {"_id": "47.91A"}} -{"name": "Vente à distance sur catalogue général", "parent": "G"} -{"index": {"_id": "47.91B"}} -{"name": "Vente à distance sur catalogue spécialisé", "parent": "G"} -{"index": {"_id": "47.99A"}} -{"name": "Vente à domicile", "parent": "G"} -{"index": {"_id": "47.99B"}} -{"name": "Vente par automates et autres commerces de détail hors magasin, éventaires ou marchés n.c.a.", "parent": "G"} -{"index": {"_id": "H"}} -{"name": "Transports et entreposage", "parent": null} -{"index": {"_id": "49.10Z"}} -{"name": "Transport ferroviaire interurbain de voyageurs", "parent": "H"} -{"index": {"_id": "49.20Z"}} -{"name": "Transports ferroviaires de fret", "parent": "H"} -{"index": {"_id": "49.31Z"}} -{"name": "Transports urbains et suburbains de voyageurs", "parent": "H"} -{"index": {"_id": "49.32Z"}} -{"name": "Transports de voyageurs par taxis", "parent": "H"} -{"index": {"_id": "49.39A"}} -{"name": "Transports routiers réguliers de voyageurs", "parent": "H"} -{"index": {"_id": "49.39B"}} -{"name": "Autres transports routiers de voyageurs", "parent": "H"} -{"index": {"_id": "49.39C"}} -{"name": "Téléphériques et remontées mécaniques", "parent": "H"} -{"index": {"_id": "49.41A"}} -{"name": "Transports routiers de fret interurbains", "parent": "H"} -{"index": {"_id": "49.41B"}} -{"name": "Transports routiers de fret de proximité", "parent": "H"} -{"index": {"_id": "49.41C"}} -{"name": "Location de camions avec chauffeur", "parent": "H"} -{"index": {"_id": "49.42Z"}} -{"name": "Services de déménagement", "parent": "H"} -{"index": {"_id": "49.50Z"}} -{"name": "Transports par conduites", "parent": "H"} -{"index": {"_id": "50.10Z"}} -{"name": "Transports maritimes et côtiers de passagers", "parent": "H"} -{"index": {"_id": "50.20Z"}} -{"name": "Transports maritimes et côtiers de fret", "parent": "H"} -{"index": {"_id": "50.30Z"}} -{"name": "Transports fluviaux de passagers", "parent": "H"} -{"index": {"_id": "50.40Z"}} -{"name": "Transports fluviaux de fret", "parent": "H"} -{"index": {"_id": "51.10Z"}} -{"name": "Transports aériens de passagers", "parent": "H"} -{"index": {"_id": "51.21Z"}} -{"name": "Transports aériens de fret", "parent": "H"} -{"index": {"_id": "51.22Z"}} -{"name": "Transports spatiaux", "parent": "H"} -{"index": {"_id": "52.10A"}} -{"name": "Entreposage et stockage frigorifique", "parent": "H"} -{"index": {"_id": "52.10B"}} -{"name": "Entreposage et stockage non frigorifique", "parent": "H"} -{"index": {"_id": "52.21Z"}} -{"name": "Services auxiliaires des transports terrestres", "parent": "H"} -{"index": {"_id": "52.22Z"}} -{"name": "Services auxiliaires des transports par eau", "parent": "H"} -{"index": {"_id": "52.23Z"}} -{"name": "Services auxiliaires des transports aériens", "parent": "H"} -{"index": {"_id": "52.24A"}} -{"name": "Manutention portuaire", "parent": "H"} -{"index": {"_id": "52.24B"}} -{"name": "Manutention non portuaire", "parent": "H"} -{"index": {"_id": "52.29A"}} -{"name": "Messagerie, fret express", "parent": "H"} -{"index": {"_id": "52.29B"}} -{"name": "Affrètement et organisation des transports", "parent": "H"} -{"index": {"_id": "53.10Z"}} -{"name": "Activités de poste dans le cadre d'une obligation de service universel", "parent": "H"} -{"index": {"_id": "53.20Z"}} -{"name": "Autres activités de poste et de courrier", "parent": "H"} -{"index": {"_id": "I"}} -{"name": "Hébergement et restauration", "parent": null} -{"index": {"_id": "55.10Z"}} -{"name": "Hôtels et hébergement similaire", "parent": "I"} -{"index": {"_id": "55.20Z"}} -{"name": "Hébergement touristique et autre hébergement de courte durée", "parent": "I"} -{"index": {"_id": "55.30Z"}} -{"name": "Terrains de camping et parcs pour caravanes ou véhicules de loisirs", "parent": "I"} -{"index": {"_id": "55.90Z"}} -{"name": "Autres hébergements", "parent": "I"} -{"index": {"_id": "56.10A"}} -{"name": "Restauration traditionnelle", "parent": "I"} -{"index": {"_id": "56.10B"}} -{"name": "Cafétérias et autres libres-services", "parent": "I"} -{"index": {"_id": "56.10C"}} -{"name": "Restauration de type rapide", "parent": "I"} -{"index": {"_id": "56.21Z"}} -{"name": "Services des traiteurs", "parent": "I"} -{"index": {"_id": "56.29A"}} -{"name": "Restauration collective sous contrat", "parent": "I"} -{"index": {"_id": "56.29B"}} -{"name": "Autres services de restauration n.c.a.", "parent": "I"} -{"index": {"_id": "56.30Z"}} -{"name": "Débits de boissons", "parent": "I"} -{"index": {"_id": "J"}} -{"name": "Information et communication", "parent": null} -{"index": {"_id": "58.11Z"}} -{"name": "Édition de livres", "parent": "J"} -{"index": {"_id": "58.12Z"}} -{"name": "Édition de répertoires et de fichiers d'adresses", "parent": "J"} -{"index": {"_id": "58.13Z"}} -{"name": "Édition de journaux", "parent": "J"} -{"index": {"_id": "58.14Z"}} -{"name": "Édition de revues et périodiques", "parent": "J"} -{"index": {"_id": "58.19Z"}} -{"name": "Autres activités d'édition", "parent": "J"} -{"index": {"_id": "58.21Z"}} -{"name": "Édition de jeux électroniques", "parent": "J"} -{"index": {"_id": "58.29A"}} -{"name": "Édition de logiciels système et de réseau", "parent": "J"} -{"index": {"_id": "58.29B"}} -{"name": "Édition de logiciels outils de développement et de langages", "parent": "J"} -{"index": {"_id": "58.29C"}} -{"name": "Édition de logiciels applicatifs", "parent": "J"} -{"index": {"_id": "59.11A"}} -{"name": "Production de films et de programmes pour la télévision", "parent": "J"} -{"index": {"_id": "59.11B"}} -{"name": "Production de films institutionnels et publicitaires", "parent": "J"} -{"index": {"_id": "59.11C"}} -{"name": "Production de films pour le cinéma", "parent": "J"} -{"index": {"_id": "59.12Z"}} -{"name": "Post-production de films cinématographiques, de vidéo et de programmes de télévision", "parent": "J"} -{"index": {"_id": "59.13A"}} -{"name": "Distribution de films cinématographiques", "parent": "J"} -{"index": {"_id": "59.13B"}} -{"name": "Édition et distribution vidéo", "parent": "J"} -{"index": {"_id": "59.14Z"}} -{"name": "Projection de films cinématographiques", "parent": "J"} -{"index": {"_id": "59.20Z"}} -{"name": "Enregistrement sonore et édition musicale", "parent": "J"} -{"index": {"_id": "60.10Z"}} -{"name": "Édition et diffusion de programmes radio", "parent": "J"} -{"index": {"_id": "60.20A"}} -{"name": "Édition de chaînes généralistes", "parent": "J"} -{"index": {"_id": "60.20B"}} -{"name": "Édition de chaînes thématiques", "parent": "J"} -{"index": {"_id": "61.10Z"}} -{"name": "Télécommunications filaires", "parent": "J"} -{"index": {"_id": "61.20Z"}} -{"name": "Télécommunications sans fil", "parent": "J"} -{"index": {"_id": "61.30Z"}} -{"name": "Télécommunications par satellite", "parent": "J"} -{"index": {"_id": "61.90Z"}} -{"name": "Autres activités de télécommunication", "parent": "J"} -{"index": {"_id": "62.01Z"}} -{"name": "Programmation informatique", "parent": "J"} -{"index": {"_id": "62.02A"}} -{"name": "Conseil en systèmes et logiciels informatiques", "parent": "J"} -{"index": {"_id": "62.02B"}} -{"name": "Tierce maintenance de systèmes et d'applications informatiques", "parent": "J"} -{"index": {"_id": "62.03Z"}} -{"name": "Gestion d'installations informatiques", "parent": "J"} -{"index": {"_id": "62.09Z"}} -{"name": "Autres activités informatiques", "parent": "J"} -{"index": {"_id": "63.11Z"}} -{"name": "Traitement de données, hébergement et activités connexes", "parent": "J"} -{"index": {"_id": "63.12Z"}} -{"name": "Portails Internet", "parent": "J"} -{"index": {"_id": "63.91Z"}} -{"name": "Activités des agences de presse", "parent": "J"} -{"index": {"_id": "63.99Z"}} -{"name": "Autres services d'information n.c.a.", "parent": "J"} -{"index": {"_id": "K"}} -{"name": "Activités financières et d'assurance", "parent": null} -{"index": {"_id": "64.11Z"}} -{"name": "Activités de banque centrale", "parent": "K"} -{"index": {"_id": "64.19Z"}} -{"name": "Autres intermédiations monétaires", "parent": "K"} -{"index": {"_id": "64.20Z"}} -{"name": "Activités des sociétés holding", "parent": "K"} -{"index": {"_id": "64.30Z"}} -{"name": "Fonds de placement et entités financières similaires", "parent": "K"} -{"index": {"_id": "64.91Z"}} -{"name": "Crédit-bail", "parent": "K"} -{"index": {"_id": "64.92Z"}} -{"name": "Autre distribution de crédit", "parent": "K"} -{"index": {"_id": "64.99Z"}} -{"name": "Autres activités des services financiers, hors assurance et caisses de retraite, n.c.a.", "parent": "K"} -{"index": {"_id": "65.11Z"}} -{"name": "Assurance vie", "parent": "K"} -{"index": {"_id": "65.12Z"}} -{"name": "Autres assurances", "parent": "K"} -{"index": {"_id": "65.20Z"}} -{"name": "Réassurance", "parent": "K"} -{"index": {"_id": "65.30Z"}} -{"name": "Caisses de retraite", "parent": "K"} -{"index": {"_id": "66.11Z"}} -{"name": "Administration de marchés financiers", "parent": "K"} -{"index": {"_id": "66.12Z"}} -{"name": "Courtage de valeurs mobilières et de marchandises", "parent": "K"} -{"index": {"_id": "66.19A"}} -{"name": "Supports juridiques de gestion de patrimoine mobilier", "parent": "K"} -{"index": {"_id": "66.19B"}} -{"name": "Autres activités auxiliaires de services financiers, hors assurance et caisses de retraite, n.c.a.", "parent": "K"} -{"index": {"_id": "66.21Z"}} -{"name": "Évaluation des risques et dommages", "parent": "K"} -{"index": {"_id": "66.22Z"}} -{"name": "Activités des agents et courtiers d'assurances", "parent": "K"} -{"index": {"_id": "66.29Z"}} -{"name": "Autres activités auxiliaires d'assurance et de caisses de retraite", "parent": "K"} -{"index": {"_id": "66.30Z"}} -{"name": "Gestion de fonds", "parent": "K"} -{"index": {"_id": "L"}} -{"name": "Activités immobilières", "parent": null} -{"index": {"_id": "68.10Z"}} -{"name": "Activités des marchands de biens immobiliers", "parent": "L"} -{"index": {"_id": "68.20A"}} -{"name": "Location de logements", "parent": "L"} -{"index": {"_id": "68.20B"}} -{"name": "Location de terrains et d'autres biens immobiliers", "parent": "L"} -{"index": {"_id": "68.31Z"}} -{"name": "Agences immobilières", "parent": "L"} -{"index": {"_id": "68.32A"}} -{"name": "Administration d'immeubles et autres biens immobiliers", "parent": "L"} -{"index": {"_id": "68.32B"}} -{"name": "Supports juridiques de gestion de patrimoine immobilier", "parent": "L"} -{"index": {"_id": "M"}} -{"name": "Activités spécialisées, scientifiques et techniques", "parent": null} -{"index": {"_id": "69.10Z"}} -{"name": "Activités juridiques", "parent": "M"} -{"index": {"_id": "69.20Z"}} -{"name": "Activités comptables", "parent": "M"} -{"index": {"_id": "70.10Z"}} -{"name": "Activités des sièges sociaux", "parent": "M"} -{"index": {"_id": "70.21Z"}} -{"name": "Conseil en relations publiques et communication", "parent": "M"} -{"index": {"_id": "70.22Z"}} -{"name": "Conseil pour les affaires et autres conseils de gestion", "parent": "M"} -{"index": {"_id": "71.11Z"}} -{"name": "Activités d'architecture", "parent": "M"} -{"index": {"_id": "71.12A"}} -{"name": "Activité des géomètres", "parent": "M"} -{"index": {"_id": "71.12B"}} -{"name": "Ingénierie, études techniques", "parent": "M"} -{"index": {"_id": "71.20A"}} -{"name": "Contrôle technique automobile", "parent": "M"} -{"index": {"_id": "71.20B"}} -{"name": "Analyses, essais et inspections techniques", "parent": "M"} -{"index": {"_id": "72.11Z"}} -{"name": "Recherche-développement en biotechnologie", "parent": "M"} -{"index": {"_id": "72.19Z"}} -{"name": "Recherche-développement en autres sciences physiques et naturelles", "parent": "M"} -{"index": {"_id": "72.20Z"}} -{"name": "Recherche-développement en sciences humaines et sociales", "parent": "M"} -{"index": {"_id": "73.11Z"}} -{"name": "Activités des agences de publicité", "parent": "M"} -{"index": {"_id": "73.12Z"}} -{"name": "Régie publicitaire de médias", "parent": "M"} -{"index": {"_id": "73.20Z"}} -{"name": "Études de marché et sondages", "parent": "M"} -{"index": {"_id": "74.10Z"}} -{"name": "Activités spécialisées de design", "parent": "M"} -{"index": {"_id": "74.20Z"}} -{"name": "Activités photographiques", "parent": "M"} -{"index": {"_id": "74.30Z"}} -{"name": "Traduction et interprétation", "parent": "M"} -{"index": {"_id": "74.90A"}} -{"name": "Activité des économistes de la construction", "parent": "M"} -{"index": {"_id": "74.90B"}} -{"name": "Activités spécialisées, scientifiques et techniques diverses", "parent": "M"} -{"index": {"_id": "N"}} -{"name": "Activités de services administratifs et de soutien", "parent": null} -{"index": {"_id": "75.00Z"}} -{"name": "Activités vétérinaires", "parent": "N"} -{"index": {"_id": "77.11A"}} -{"name": "Location de courte durée de voitures et de véhicules automobiles légers", "parent": "N"} -{"index": {"_id": "77.11B"}} -{"name": "Location de longue durée de voitures et de véhicules automobiles légers", "parent": "N"} -{"index": {"_id": "77.12Z"}} -{"name": "Location et location-bail de camions", "parent": "N"} -{"index": {"_id": "77.21Z"}} -{"name": "Location et location-bail d'articles de loisirs et de sport", "parent": "N"} -{"index": {"_id": "77.22Z"}} -{"name": "Location de vidéocassettes et disques vidéo", "parent": "N"} -{"index": {"_id": "77.29Z"}} -{"name": "Location et location-bail d'autres biens personnels et domestiques", "parent": "N"} -{"index": {"_id": "77.31Z"}} -{"name": "Location et location-bail de machines et équipements agricoles", "parent": "N"} -{"index": {"_id": "77.32Z"}} -{"name": "Location et location-bail de machines et équipements pour la construction", "parent": "N"} -{"index": {"_id": "77.33Z"}} -{"name": "Location et location-bail de machines de bureau et de matériel informatique", "parent": "N"} -{"index": {"_id": "77.34Z"}} -{"name": "Location et location-bail de matériels de transport par eau", "parent": "N"} -{"index": {"_id": "77.35Z"}} -{"name": "Location et location-bail de matériels de transport aérien", "parent": "N"} -{"index": {"_id": "77.39Z"}} -{"name": "Location et location-bail d'autres machines, équipements et biens matériels n.c.a.", "parent": "N"} -{"index": {"_id": "77.40Z"}} -{"name": "Location-bail de propriété intellectuelle et de produits similaires, à l'exception des œuvres soumises à copyright", "parent": "N"} -{"index": {"_id": "78.10Z"}} -{"name": "Activités des agences de placement de main-d'œuvre", "parent": "N"} -{"index": {"_id": "78.20Z"}} -{"name": "Activités des agences de travail temporaire", "parent": "N"} -{"index": {"_id": "78.30Z"}} -{"name": "Autre mise à disposition de ressources humaines", "parent": "N"} -{"index": {"_id": "79.11Z"}} -{"name": "Activités des agences de voyage", "parent": "N"} -{"index": {"_id": "79.12Z"}} -{"name": "Activités des voyagistes", "parent": "N"} -{"index": {"_id": "79.90Z"}} -{"name": "Autres services de réservation et activités connexes", "parent": "N"} -{"index": {"_id": "80.10Z"}} -{"name": "Activités de sécurité privée", "parent": "N"} -{"index": {"_id": "80.20Z"}} -{"name": "Activités liées aux systèmes de sécurité", "parent": "N"} -{"index": {"_id": "80.30Z"}} -{"name": "Activités d'enquête", "parent": "N"} -{"index": {"_id": "81.10Z"}} -{"name": "Activités combinées de soutien lié aux bâtiments", "parent": "N"} -{"index": {"_id": "81.21Z"}} -{"name": "Nettoyage courant des bâtiments", "parent": "N"} -{"index": {"_id": "81.22Z"}} -{"name": "Autres activités de nettoyage des bâtiments et nettoyage industriel", "parent": "N"} -{"index": {"_id": "81.29A"}} -{"name": "Désinfection, désinsectisation, dératisation", "parent": "N"} -{"index": {"_id": "81.29B"}} -{"name": "Autres activités de nettoyage n.c.a.", "parent": "N"} -{"index": {"_id": "81.30Z"}} -{"name": "Services d'aménagement paysager", "parent": "N"} -{"index": {"_id": "82.11Z"}} -{"name": "Services administratifs combinés de bureau", "parent": "N"} -{"index": {"_id": "82.19Z"}} -{"name": "Photocopie, préparation de documents et autres activités spécialisées de soutien de bureau", "parent": "N"} -{"index": {"_id": "82.20Z"}} -{"name": "Activités de centres d'appels", "parent": "N"} -{"index": {"_id": "82.30Z"}} -{"name": "Organisation de foires, salons professionnels et congrès", "parent": "N"} -{"index": {"_id": "82.91Z"}} -{"name": "Activités des agences de recouvrement de factures et des sociétés d'information financière sur la clientèle", "parent": "N"} -{"index": {"_id": "82.92Z"}} -{"name": "Activités de conditionnement", "parent": "N"} -{"index": {"_id": "82.99Z"}} -{"name": "Autres activités de soutien aux entreprises n.c.a.", "parent": "N"} -{"index": {"_id": "O"}} -{"name": "Administration publique", "parent": null} -{"index": {"_id": "84.11Z"}} -{"name": "Administration publique générale", "parent": "O"} -{"index": {"_id": "84.12Z"}} -{"name": "Administration publique (tutelle) de la santé, de la formation, de la culture et des services sociaux, autre que sécurité sociale", "parent": "O"} -{"index": {"_id": "84.13Z"}} -{"name": "Administration publique (tutelle) des activités économiques", "parent": "O"} -{"index": {"_id": "84.21Z"}} -{"name": "Affaires étrangères", "parent": "O"} -{"index": {"_id": "84.22Z"}} -{"name": "Défense", "parent": "O"} -{"index": {"_id": "84.23Z"}} -{"name": "Justice", "parent": "O"} -{"index": {"_id": "84.24Z"}} -{"name": "Activités d'ordre public et de sécurité", "parent": "O"} -{"index": {"_id": "84.25Z"}} -{"name": "Services du feu et de secours", "parent": "O"} -{"index": {"_id": "84.30A"}} -{"name": "Activités générales de sécurité sociale", "parent": "O"} -{"index": {"_id": "84.30B"}} -{"name": "Gestion des retraites complémentaires", "parent": "O"} -{"index": {"_id": "84.30C"}} -{"name": "Distribution sociale de revenus", "parent": "O"} -{"index": {"_id": "P"}} -{"name": "Enseignement", "parent": null} -{"index": {"_id": "85.10Z"}} -{"name": "Enseignement pré-primaire", "parent": "P"} -{"index": {"_id": "85.20Z"}} -{"name": "Enseignement primaire", "parent": "P"} -{"index": {"_id": "85.31Z"}} -{"name": "Enseignement secondaire général", "parent": "P"} -{"index": {"_id": "85.32Z"}} -{"name": "Enseignement secondaire technique ou professionnel", "parent": "P"} -{"index": {"_id": "85.41Z"}} -{"name": "Enseignement post-secondaire non supérieur", "parent": "P"} -{"index": {"_id": "85.42Z"}} -{"name": "Enseignement supérieur", "parent": "P"} -{"index": {"_id": "85.51Z"}} -{"name": "Enseignement de disciplines sportives et d'activités de loisirs", "parent": "P"} -{"index": {"_id": "85.52Z"}} -{"name": "Enseignement culturel", "parent": "P"} -{"index": {"_id": "85.53Z"}} -{"name": "Enseignement de la conduite", "parent": "P"} -{"index": {"_id": "85.59A"}} -{"name": "Formation continue d'adultes", "parent": "P"} -{"index": {"_id": "85.59B"}} -{"name": "Autres enseignements", "parent": "P"} -{"index": {"_id": "85.60Z"}} -{"name": "Activités de soutien à l'enseignement", "parent": "P"} -{"index": {"_id": "Q"}} -{"name": "Santé humaine et action sociale", "parent": null} -{"index": {"_id": "86.10Z"}} -{"name": "Activités hospitalières", "parent": "Q"} -{"index": {"_id": "86.21Z"}} -{"name": "Activité des médecins généralistes", "parent": "Q"} -{"index": {"_id": "86.22A"}} -{"name": "Activités de radiodiagnostic et de radiothérapie", "parent": "Q"} -{"index": {"_id": "86.22B"}} -{"name": "Activités chirurgicales", "parent": "Q"} -{"index": {"_id": "86.22C"}} -{"name": "Autres activités des médecins spécialistes", "parent": "Q"} -{"index": {"_id": "86.23Z"}} -{"name": "Pratique dentaire", "parent": "Q"} -{"index": {"_id": "86.90A"}} -{"name": "Ambulances", "parent": "Q"} -{"index": {"_id": "86.90B"}} -{"name": "Laboratoires d'analyses médicales", "parent": "Q"} -{"index": {"_id": "86.90C"}} -{"name": "Centres de collecte et banques d'organes", "parent": "Q"} -{"index": {"_id": "86.90D"}} -{"name": "Activités des infirmiers et des sages-femmes", "parent": "Q"} -{"index": {"_id": "86.90E"}} -{"name": "Activités des professionnels de la rééducation, de l'appareillage et des pédicures-podologues", "parent": "Q"} -{"index": {"_id": "86.90F"}} -{"name": "Activités de santé humaine non classées ailleurs", "parent": "Q"} -{"index": {"_id": "87.10A"}} -{"name": "Hébergement médicalisé pour personnes âgées", "parent": "Q"} -{"index": {"_id": "87.10B"}} -{"name": "Hébergement médicalisé pour enfants handicapés", "parent": "Q"} -{"index": {"_id": "87.10C"}} -{"name": "Hébergement médicalisé pour adultes handicapés et autre hébergement médicalisé", "parent": "Q"} -{"index": {"_id": "87.20A"}} -{"name": "Hébergement social pour handicapés mentaux et malades mentaux", "parent": "Q"} -{"index": {"_id": "87.20B"}} -{"name": "Hébergement social pour toxicomanes", "parent": "Q"} -{"index": {"_id": "87.30A"}} -{"name": "Hébergement social pour personnes âgées", "parent": "Q"} -{"index": {"_id": "87.30B"}} -{"name": "Hébergement social pour handicapés physiques", "parent": "Q"} -{"index": {"_id": "87.90A"}} -{"name": "Hébergement social pour enfants en difficultés", "parent": "Q"} -{"index": {"_id": "87.90B"}} -{"name": "Hébergement social pour adultes et familles en difficultés et autre hébergement social", "parent": "Q"} -{"index": {"_id": "88.10A"}} -{"name": "Aide à domicile", "parent": "Q"} -{"index": {"_id": "88.10B"}} -{"name": "Accueil ou accompagnement sans hébergement d'adultes handicapés ou de personnes âgées", "parent": "Q"} -{"index": {"_id": "88.10C"}} -{"name": "Aide par le travail", "parent": "Q"} -{"index": {"_id": "88.91A"}} -{"name": "Accueil de jeunes enfants", "parent": "Q"} -{"index": {"_id": "88.91B"}} -{"name": "Accueil ou accompagnement sans hébergement d'enfants handicapés", "parent": "Q"} -{"index": {"_id": "88.99A"}} -{"name": "Autre accueil ou accompagnement sans hébergement d'enfants et d'adolescents", "parent": "Q"} -{"index": {"_id": "88.99B"}} -{"name": "Action sociale sans hébergement n.c.a.", "parent": "Q"} -{"index": {"_id": "R"}} -{"name": "Arts, spectacles et activités récréatives", "parent": null} -{"index": {"_id": "90.01Z"}} -{"name": "Arts du spectacle vivant", "parent": "R"} -{"index": {"_id": "90.02Z"}} -{"name": "Activités de soutien au spectacle vivant", "parent": "R"} -{"index": {"_id": "90.03A"}} -{"name": "Création artistique relevant des arts plastiques", "parent": "R"} -{"index": {"_id": "90.03B"}} -{"name": "Autre création artistique", "parent": "R"} -{"index": {"_id": "90.04Z"}} -{"name": "Gestion de salles de spectacles", "parent": "R"} -{"index": {"_id": "91.01Z"}} -{"name": "Gestion des bibliothèques et des archives", "parent": "R"} -{"index": {"_id": "91.02Z"}} -{"name": "Gestion des musées", "parent": "R"} -{"index": {"_id": "91.03Z"}} -{"name": "Gestion des sites et monuments historiques et des attractions touristiques similaires", "parent": "R"} -{"index": {"_id": "91.04Z"}} -{"name": "Gestion des jardins botaniques et zoologiques et des réserves naturelles", "parent": "R"} -{"index": {"_id": "92.00Z"}} -{"name": "Organisation de jeux de hasard et d'argent", "parent": "R"} -{"index": {"_id": "93.11Z"}} -{"name": "Gestion d'installations sportives", "parent": "R"} -{"index": {"_id": "93.12Z"}} -{"name": "Activités de clubs de sports", "parent": "R"} -{"index": {"_id": "93.13Z"}} -{"name": "Activités des centres de culture physique", "parent": "R"} -{"index": {"_id": "93.19Z"}} -{"name": "Autres activités liées au sport", "parent": "R"} -{"index": {"_id": "93.21Z"}} -{"name": "Activités des parcs d'attractions et parcs à thèmes", "parent": "R"} -{"index": {"_id": "93.29Z"}} -{"name": "Autres activités récréatives et de loisirs", "parent": "R"} -{"index": {"_id": "S"}} -{"name": "Autres activités de services", "parent": null} -{"index": {"_id": "94.11Z"}} -{"name": "Activités des organisations patronales et consulaires", "parent": "S"} -{"index": {"_id": "94.12Z"}} -{"name": "Activités des organisations professionnelles", "parent": "S"} -{"index": {"_id": "94.20Z"}} -{"name": "Activités des syndicats de salariés", "parent": "S"} -{"index": {"_id": "94.91Z"}} -{"name": "Activités des organisations religieuses", "parent": "S"} -{"index": {"_id": "94.92Z"}} -{"name": "Activités des organisations politiques", "parent": "S"} -{"index": {"_id": "94.99Z"}} -{"name": "Autres organisations fonctionnant par adhésion volontaire", "parent": "S"} -{"index": {"_id": "95.11Z"}} -{"name": "Réparation d'ordinateurs et d'équipements périphériques", "parent": "S"} -{"index": {"_id": "95.12Z"}} -{"name": "Réparation d'équipements de communication", "parent": "S"} -{"index": {"_id": "95.21Z"}} -{"name": "Réparation de produits électroniques grand public", "parent": "S"} -{"index": {"_id": "95.22Z"}} -{"name": "Réparation d'appareils électroménagers et d'équipements pour la maison et le jardin", "parent": "S"} -{"index": {"_id": "95.23Z"}} -{"name": "Réparation de chaussures et d'articles en cuir", "parent": "S"} -{"index": {"_id": "95.24Z"}} -{"name": "Réparation de meubles et d'équipements du foyer", "parent": "S"} -{"index": {"_id": "95.25Z"}} -{"name": "Réparation d'articles d'horlogerie et de bijouterie", "parent": "S"} -{"index": {"_id": "95.29Z"}} -{"name": "Réparation d'autres biens personnels et domestiques", "parent": "S"} -{"index": {"_id": "96.01A"}} -{"name": "Blanchisserie-teinturerie de gros", "parent": "S"} -{"index": {"_id": "96.01B"}} -{"name": "Blanchisserie-teinturerie de détail", "parent": "S"} -{"index": {"_id": "96.02A"}} -{"name": "Coiffure", "parent": "S"} -{"index": {"_id": "96.02B"}} -{"name": "Soins de beauté", "parent": "S"} -{"index": {"_id": "96.03Z"}} -{"name": "Services funéraires", "parent": "S"} -{"index": {"_id": "96.04Z"}} -{"name": "Entretien corporel", "parent": "S"} -{"index": {"_id": "96.09Z"}} -{"name": "Autres services personnels n.c.a.", "parent": "S"} -{"index": {"_id": "T"}} -{"name": "Activités des ménages en tant qu'employeurs ; activités indifférenciées des ménages en tant que producteurs de biens et services pour usage propre", "parent": null} -{"index": {"_id": "97.00Z"}} -{"name": "Activités des ménages en tant qu'employeurs de personnel domestique", "parent": "T"} -{"index": {"_id": "98.10Z"}} -{"name": "Activités indifférenciées des ménages en tant que producteurs de biens pour usage propre", "parent": "T"} -{"index": {"_id": "98.20Z"}} -{"name": "Activités indifférenciées des ménages en tant que producteurs de services pour usage propre", "parent": "T"} -{"index": {"_id": "U"}} -{"name": "Activités extra-territoriales", "parent": null} -{"index": {"_id": "99.00Z"}} -{"name": "Activités des organisations et organismes extraterritoriaux", "parent": "U"} diff --git a/duniter4j-es-user/src/main/resources/plugin-security.policy b/duniter4j-es-user/src/main/resources/plugin-security.policy deleted file mode 100644 index 23b556b1994f188dcf5fd757e8861972b6b63699..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/main/resources/plugin-security.policy +++ /dev/null @@ -1,5 +0,0 @@ -grant codeBase "file:${es.path.home}/plugins/duniter4j-es-user/"{ - 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"; -}; \ No newline at end of file diff --git a/duniter4j-es-user/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-user/src/test/es-home/config/elasticsearch.yml deleted file mode 100644 index 543cdedf19c4da29f2645bbdc9a4577d3830c0c2..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/es-home/config/elasticsearch.yml +++ /dev/null @@ -1,226 +0,0 @@ -# ======================== Elasticsearch Configuration ========================= -# -# NOTE: Elasticsearch comes with reasonable defaults for most settings. -# Before you set out to tweak and tune the configuration, make sure you -# understand what are you trying to accomplish and the consequences. -# -# The primary way of configuring a node is via this file. This template lists -# the most important settings you may want to configure for a production cluster. -# -# Please see the documentation for further information on configuration options: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/setup-configuration.html> -# -# ---------------------------------- Cluster ----------------------------------- -# -# Use a descriptive name for your cluster: -# -# cluster.name: my-application -cluster.name: duniter4j-es-assembly-test-2 -# -# ------------------------------------ Node ------------------------------------ -# -# Use a descriptive name for the node: -# -# node.name: node-1 -# -# Add custom attributes to the node: -# -# node.rack: r1 -# -# ----------------------------------- Paths ------------------------------------ -# -# Path to directory where to store the data (separate multiple locations by comma): -# -# path.data: /path/to/data -# -# Path to log files: -# -# path.logs: /path/to/logs -# -# ----------------------------------- Memory ----------------------------------- -# -# Lock the memory on startup: -# -# bootstrap.mlockall: true -# -# Make sure that the `ES_HEAP_SIZE` environment variable is set to about half the memory -# available on the system and that the owner of the process is allowed to use this limit. -# -# Elasticsearch performs poorly when the system is swapping the memory. -# -# ---------------------------------- Network ----------------------------------- -# -# Set the bind address to a specific IP (IPv4 or IPv6): -# -# network.host: 192.168.233.118 -# -# Set a custom port for HTTP: -# -# http.port: 9200-9300 - -http.cors.allow-origin: "/.*/" -http.cors.enabled: true - -# Internal transport layer -# -# transport.tcp.port: 9210-9220 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-network.html> -# -# --------------------------------- Discovery ---------------------------------- -# -# Pass an initial list of hosts to perform discovery when new node is started: -# The default list of hosts is ["127.0.0.1", "[::1]"] -# -# discovery.zen.ping.unicast.hosts: ["host1", "host2"] -#discovery.zen.ping.unicast.hosts: ["127.0.0.1", ""] -# -# Prevent the "split brain" by configuring the majority of nodes (total number of nodes / 2 + 1): -# -# discovery.zen.minimum_master_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-discovery.html> -# -# ---------------------------------- Gateway ----------------------------------- -# -# Block initial recovery after a full cluster restart until N nodes are started: -# -# gateway.recover_after_nodes: 3 -# -# For more information, see the documentation at: -# <http://www.elastic.co/guide/en/elasticsearch/reference/current/modules-gateway.html> -# -# ---------------------------------- Various ----------------------------------- -# -# Disable starting multiple nodes on a single system: -# -# node.max_local_storage_nodes: 1 -# -# Require explicit names when deleting indices: -# -# rest.destructive_requires_name: true - -security.manager.enabled: false - -# -# ---------------------------------- Duniter4j --------------------------------- -# -# Disbale duniter4j plugin -# -# duniter.enabled: false -# -# Delete then create all indices at startup - DO NOT set to true in production -# -#duniter.indices.reload: true -# -# Default string analyzer -# -duniter.string.analyzer: french -# -# Enabling blockchain synchronization -# -duniter.blockchain.enable: false -# -# Force blockchain reload - WARNING: all user events will be resetted to 'unread' -# -#duniter.blockchain.reload: true -#duniter.blockchain.reload.from: 50999 -# -# Duniter node address -# -duniter.host: g1-test.duniter.org -duniter.port: 10900 -#duniter.useSsl: true -duniter4j.network.timeout: 10000 -# -# ---------------------------------- Duniter4j security module ------------------- -# -# Allow admin actions -# -duniter.keyring.salt: 'abc' -duniter.keyring.password: 'def' -# -# Enable security - will restrict HTTP access to only Duniter4j known indices -# -duniter.security.enable: true -# -# Security token prefix (default: 'duniter-') -# -# duniter.auth.token.prefix: duniter- -# -# Token validity duration, in seconds (default: 600) -# -# duniter.auth.tokenValidityDuration: 3600 # = 1hour - -# ---------------------------------- Duniter4j P2P sync ------------------------- -# -# Should synchronize data from an existing ES node ? -# -duniter.p2p.enable: false - -# ---------------------------------- Duniter4j Mail module ----------------------- -# -# Enable mail module ? -# -duniter.mail.enable: false -# -# Mail: SMTP server configuration (host and port) -# -duniter.mail.smtp.host: localhost -duniter.mail.smtp.port: 25 -# -# Mail: SMTP server SSL security -# -#duniter.mail.smtp.ssl: true -#duniter.mail.smtp.starttls: true -# -# Mail: SMTP server authentication -# -#duniter.mail.smtp.username: -#duniter.mail.smtp.password: -# -# Mail: 'from' address -# -#duniter.mail.from: no-reply@domain.com -duniter.mail.from: 'no-reply@duniter.fr' -# -# Mail: admin address -# -#duniter.mail.admin: user@domain.com -#duniter.mail.admin: blavenie@EIS-DEV -duniter.mail.admin: 'benoit.lavenier@e-is.pro' -# -# Mail: subject prefix -# -#duniter.mail.subject.prefix: '[Cesium+]' - -# ---------------------------------- Duniter4j Websocket server ---------------------- -# -# Websocket port (usefull for listen changes) -# -duniter.ws.port: 9400-9410 - -# ---------------------------------- Duniter4j Subscription module ------------------- -# -# Enable subscription module (Need to enable mail features) -# -duniter.subscription.enable: true -# -# Opions to DEBUG this features -# -#duniter.subscription.email.atStartup: false -#duniter.subscription.email.debug: false -# -# Email subscription: Day of the week to trigger weekly (default: 2 = monday) -# -#duniter.subscription.email.dayOfWeek: 2 -# -# Email subscription: Hour in day to trigger daily email subscription (default: 3 AM) -# -duniter.subscription.email.hourOfDay: 3 -# -# 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 diff --git a/duniter4j-es-user/src/test/es-home/config/logging.yml b/duniter4j-es-user/src/test/es-home/config/logging.yml deleted file mode 100644 index 9e7d36a4db818c4d8d488430427389a8d356d011..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/es-home/config/logging.yml +++ /dev/null @@ -1,103 +0,0 @@ -# you can override this using by setting a system property, for example -Des.logger.level=DEBUG -es.logger.level: INFO -rootLogger: ${es.logger.level}, console, file -logger: - # log rest execution errors for easier debugging - action: DEBUG - - # deprecation logging, turn to DEBUG to see them - deprecation: INFO, deprecation_log_file - - # reduce the logging for aws, too much is logged under the default INFO - com.amazonaws: WARN - # aws will try to do some sketchy JMX stuff, but its not needed. - com.amazonaws.jmx.SdkMBeanRegistrySupport: ERROR - com.amazonaws.metrics.AwsSdkMetrics: ERROR - - org.apache.http: INFO - - org.duniter: INFO - #org.duniter.elasticsearch: DEBUG - - duniter : INFO - #duniter.network.p2p: TRACE - - security: INFO - cluster.metadata: ERROR - cluster.routing.allocation: ERROR - - org.nuiton.i18n: ERROR - org.nuiton.config: ERROR - org.nuiton.converter: WARN - org.apache.http: WARN - org.apache.http.client: ERROR - org.glassfish.grizzly: WARN - org.glassfish.tyrus: WARN - - # gateway - #gateway: DEBUG - #index.gateway: DEBUG - - # peer shard recovery - #indices.recovery: DEBUG - - # discovery - #discovery: TRACE - - index.search.slowlog: TRACE, index_search_slow_log_file - index.indexing.slowlog: TRACE, index_indexing_slow_log_file - -additivity: - index.search.slowlog: false - index.indexing.slowlog: false - deprecation: false - -appender: - console: - type: console - layout: - type: consolePattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %.10000m%n" - - # Use the following log4j-extras RollingFileAppender to enable gzip compression of log files. - # For more information see https://logging.apache.org/log4j/extras/apidocs/org/apache/log4j/rolling/RollingFileAppender.html - #file: - #type: extrasRollingFile - #file: ${path.logs}/${cluster.name}.log - #rollingPolicy: timeBased - #rollingPolicy.FileNamePattern: ${path.logs}/${cluster.name}.log.%d{yyyy-MM-dd}.gz - #layout: - #type: pattern - #conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - deprecation_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_deprecation.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_search_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_search_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" - - index_indexing_slow_log_file: - type: dailyRollingFile - file: ${path.logs}/${cluster.name}_index_indexing_slowlog.log - datePattern: "'.'yyyy-MM-dd" - layout: - type: pattern - conversionPattern: "[%d{ISO8601}][%-5p][%-25c] %m%n" diff --git a/duniter4j-es-user/src/test/es-home/plugins/duniter4j-es-core/plugin-descriptor.properties b/duniter4j-es-user/src/test/es-home/plugins/duniter4j-es-core/plugin-descriptor.properties deleted file mode 100644 index 4a58004ff2b6749e878b33db5cffa077d0ab3a71..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/es-home/plugins/duniter4j-es-core/plugin-descriptor.properties +++ /dev/null @@ -1,9 +0,0 @@ -name=duniter4j-es-core -description=Plugin for Duniter -version=1.0 -site=false -jvm=true -classname=org.duniter.elasticsearch.Plugin -java.version=1.8 -elasticsearch.version=2.4.6 -isolated=false diff --git a/duniter4j-es-user/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties b/duniter4j-es-user/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties deleted file mode 100644 index 49bd43a2e29da14384b67a9b51b727354fe90521..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties +++ /dev/null @@ -1,80 +0,0 @@ -# Elasticsearch plugin descriptor file -# This file must exist as 'plugin-descriptor.properties' at -# the root directory of all plugins. -# -# A plugin can be 'site', 'jvm', or both. -# -### example site plugin for "foo": -# -# foo.zip <-- zip file for the plugin, with this structure: -# _site/ <-- the contents that will be served -# plugin-descriptor.properties <-- example contents below: -# -# site=true -# description=My cool plugin -# version=1.0 -# -### example jvm plugin for "foo" -# -# foo.zip <-- zip file for the plugin, with this structure: -# <arbitrary name1>.jar <-- classes, resources, dependencies -# <arbitrary nameN>.jar <-- any number of jars -# plugin-descriptor.properties <-- example contents below: -# -# jvm=true -# classname=foo.bar.BazPlugin -# description=My cool plugin -# version=2.0.0-rc1 -# elasticsearch.version=2.0 -# java.version=1.7 -# -### mandatory elements for all plugins: -# -# 'description': simple summary of the plugin -description=The mapper attachments plugin adds the attachment type to Elasticsearch using Apache Tika. -# -# 'version': plugin's version -version=2.3.3 -# -# 'name': the plugin name -name=mapper-attachments - -### mandatory elements for site plugins: -# -# 'site': set to true to indicate contents of the _site/ -# directory in the root of the plugin should be served. -site=false -# -### mandatory elements for jvm plugins : -# -# 'jvm': true if the 'classname' class should be loaded -# from jar files in the root directory of the plugin. -# Note that only jar files in the root directory are -# added to the classpath for the plugin! If you need -# other resources, package them into a resources jar. -jvm=true -# -# 'classname': the name of the class to load, fully-qualified. -classname=org.elasticsearch.mapper.attachments.MapperAttachmentsPlugin -# -# 'java.version' version of java the code is built against -# use the system property java.specification.version -# version string must be a sequence of nonnegative decimal integers -# separated by "."'s and may have leading zeros -java.version=1.7 -# -# 'elasticsearch.version' version of elasticsearch compiled against -# You will have to release a new version of the plugin for each new -# elasticsearch release. This version is checked when the plugin -# is loaded so Elasticsearch will refuse to start in the presence of -# plugins with the incorrect elasticsearch.version. -elasticsearch.version=2.4.6 -# -### deprecated elements for jvm plugins : -# -# 'isolated': true if the plugin should have its own classloader. -# passing false is deprecated, and only intended to support plugins -# that have hard dependencies against each other. If this is -# not specified, then the plugin is isolated by default. -isolated=true -# diff --git a/duniter4j-es-user/src/test/es-home/plugins/mapper-attachments/plugin-security.policy b/duniter4j-es-user/src/test/es-home/plugins/mapper-attachments/plugin-security.policy deleted file mode 100644 index 3264766682269e55baae64ca70d1170101260abf..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/es-home/plugins/mapper-attachments/plugin-security.policy +++ /dev/null @@ -1,34 +0,0 @@ -/* - * Licensed to Elasticsearch under one or more contributor - * license agreements. See the NOTICE file distributed with - * this work for additional information regarding copyright - * ownership. Elasticsearch licenses this file to you under - * the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, - * software distributed under the License is distributed on an - * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY - * KIND, either express or implied. See the License for the - * specific language governing permissions and limitations - * under the License. - */ - -// NOTE: when modifying this file, look at restrictions in TikaImpl too -grant { - // needed to apply additional sandboxing to tika parsing - permission java.security.SecurityPermission "createAccessControlContext"; - - // TODO: fix PDFBox not to actually install bouncy castle like this - permission java.security.SecurityPermission "putProviderProperty.BC"; - permission java.security.SecurityPermission "insertProvider"; - // needed only on java 7 - permission java.security.SecurityPermission "insertProvider.BC"; - // TODO: fix POI XWPF to not do this: https://bz.apache.org/bugzilla/show_bug.cgi?id=58597 - permission java.lang.reflect.ReflectPermission "suppressAccessChecks"; - // needed by xmlbeans, as part of POI for MS xml docs - permission java.lang.RuntimePermission "getClassLoader"; -}; diff --git a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestConfiguration.java b/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestConfiguration.java deleted file mode 100644 index d2acaa0c8cab3d30b62ce1541a776e9381e7c066..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestConfiguration.java +++ /dev/null @@ -1,57 +0,0 @@ -package org.duniter.elasticsearch.user; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.exception.TechnicalException; -import org.nuiton.config.ApplicationConfig; -import org.nuiton.config.ArgumentsParserException; - -import static org.nuiton.i18n.I18n.t; - -/** - * Created by blavenie on 13/09/17. - */ -public class TestConfiguration { - - private ApplicationConfig applicationConfig; - - public TestConfiguration(String configFileName) { - applicationConfig = new ApplicationConfig(); - applicationConfig.setConfigFileName(configFileName); - - try { - applicationConfig.parse(new String[]{}); - - } catch (ArgumentsParserException e) { - throw new TechnicalException(t("duniter4j.config.parse.error"), e); - } - } - - public String getDataSyncHost() { - return applicationConfig.getOption("duniter4j.data.sync.host"); - } - - public int getDataSyncPort() { - return applicationConfig.getOptionAsInt("duniter4j.data.sync.port"); - } -} diff --git a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestFixtures.java b/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestFixtures.java deleted file mode 100644 index 52404402f2f6e72e66c96bb2724c02957c6e1347..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestFixtures.java +++ /dev/null @@ -1,26 +0,0 @@ -package org.duniter.elasticsearch.user;/* - * #%L - * Duniter4j :: ElasticSearch Indexer - * %% - * 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% - */ - - -public class TestFixtures extends org.duniter.core.test.TestFixtures { - -} diff --git a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestResource.java b/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestResource.java deleted file mode 100644 index 86993272bcd1a58fd1200dfdb2695723ad49131d..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/TestResource.java +++ /dev/null @@ -1,84 +0,0 @@ -package org.duniter.elasticsearch.user;/* - * #%L - * Duniter4j :: 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 org.apache.commons.io.FileUtils; -import org.elasticsearch.bootstrap.Elasticsearch; -import org.junit.runner.Description; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.File; - -public class TestResource extends org.duniter.core.test.TestResource { - - private static final Logger log = LoggerFactory.getLogger(TestResource.class); - - public static TestResource create() { - return new TestResource(null); - } - - public static TestResource create(String configName) { - return new TestResource(configName); - } - - private TestFixtures fixtures = new TestFixtures(); - - private TestConfiguration testConfiguration; - - protected TestResource(String configName) { - super(configName); - } - - protected void before(Description description) throws Throwable { - super.before(description); - - // Prepare ES home - File esHomeDir = getResourceDirectory("es-home"); - - System.setProperty("es.path.home", esHomeDir.getCanonicalPath()); - - FileUtils.copyDirectory(new File("src/test/es-home"), esHomeDir); - FileUtils.copyDirectory(new File("target/classes"), new File(esHomeDir, "plugins/duniter4j-es-user")); - - Elasticsearch.main(new String[]{"start"}); - - // Init a configuration - testConfiguration = new TestConfiguration(getConfigFileName()); - - } - - public TestFixtures getFixtures() { - return fixtures; - } - - public TestConfiguration getConfiguration() { - return testConfiguration; - } - - /* -- protected method -- */ - - protected String getConfigFilesPrefix() { - return "duniter4j-es-user-test"; - } - -} diff --git a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/service/SynchroServiceTest.java b/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/service/SynchroServiceTest.java deleted file mode 100644 index c75b643f88b7a726394ac32b53485062ffb13a33..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/java/org/duniter/elasticsearch/user/service/SynchroServiceTest.java +++ /dev/null @@ -1,103 +0,0 @@ -package org.duniter.elasticsearch.user.service; - -/*- - * #%L - * Duniter4j :: ElasticSearch User 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.config.Configuration; -import org.duniter.core.client.model.bma.EndpointApi; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.client.model.local.Peers; -import org.duniter.elasticsearch.model.SynchroResult; -import org.duniter.elasticsearch.service.CurrencyService; -import org.duniter.elasticsearch.service.ServiceLocator; -import org.duniter.elasticsearch.synchro.SynchroService; -import org.duniter.elasticsearch.user.TestResource; -import org.junit.*; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -/** - * Created by blavenie on 13/09/17. - */ -public class SynchroServiceTest { - - private static final Logger log = LoggerFactory.getLogger(SynchroServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private CurrencyService currencyService; - private SynchroService service; - private Peer peer; - - @Before - public void setUp() throws Exception { - currencyService = ServiceLocator.instance().getBean(CurrencyService.class); - service = ServiceLocator.instance().getBean(SynchroService.class); - peer = new Peer.Builder() - .setHost(resource.getConfiguration().getDataSyncHost()) - .setPort(resource.getConfiguration().getDataSyncPort()) - .setCurrency(resource.getFixtures().getCurrency()) - .build(); - - while(!service.isReady()) { - Thread.sleep(1000); - } - - // Init the currency index - Configuration config = Configuration.instance(); - currencyService.createIndexIfNotExists().indexCurrencyFromPeer(new Peer.Builder() - .setHost(config.getNodeHost()) - .setPort(config.getNodePort()).build()); - - // Init data indices - ServiceLocator.instance().getBean(UserService.class).createIndexIfNotExists(); - ServiceLocator.instance().getBean(HistoryService.class).createIndexIfNotExists(); - ServiceLocator.instance().getBean(PageService.class).createIndexIfNotExists(); - ServiceLocator.instance().getBean(GroupService.class).createIndexIfNotExists(); - - Thread.sleep(5000); - } - - @Test - public void synchronizePeer() throws Exception { - // Set a id (require for saving the synchro execution) - peer.setId("fake-hash"); - - // Set the API (require for synchro) - peer.setApi(EndpointApi.ES_USER_API.name()); - - SynchroResult result = service.synchronizePeer(peer, false); - Assert.assertNotNull(result); - - Assert.assertTrue(result.getInserts() > 0); - } - - @Test - @Ignore - public void startNode() throws Exception { - - while(true) { - Thread.sleep(10000); - } - } -} diff --git a/duniter4j-es-user/src/test/resources/duniter4j-es-user-test.properties b/duniter4j-es-user/src/test/resources/duniter4j-es-user-test.properties deleted file mode 100644 index 71ab38ca3edad272060c306434e27a2873158570..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/resources/duniter4j-es-user-test.properties +++ /dev/null @@ -1,5 +0,0 @@ -# This is options only used by TestRessource.getTestConfig(). -# For ES config, see files inside 'src/test/es-home/config' - -duniter4j.data.sync.host=g1-test.data.duniter.fr -duniter4j.data.sync.port=443 \ No newline at end of file diff --git a/duniter4j-es-user/src/test/resources/log4j.properties b/duniter4j-es-user/src/test/resources/log4j.properties deleted file mode 100644 index 1c39b0d7477dd9deb2b995c9ad6b05106eda9074..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/resources/log4j.properties +++ /dev/null @@ -1,21 +0,0 @@ -### -# Global logging configuration -log4j.rootLogger=ERROR, stdout - -# Console output -log4j.appender.stdout=org.apache.log4j.ConsoleAppender -log4j.appender.stdout.layout=org.apache.log4j.PatternLayout -log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m%n - -# duniter4j levels -#log4j.logger.org.duniter=INFO -log4j.logger.org.duniter=DEBUG -log4j.logger.org.duniter.core=WARN -log4j.logger.org.duniter.elasticsearch=DEBUG - -# Other frameworks levels -log4j.logger.org.elasticsearch=INFO -log4j.logger.org.apache.http=WARN -log4j.logger.org.glassfish.grizzly=WARN -log4j.logger.org.glassfish.tyrus=WARN - diff --git a/duniter4j-es-user/src/test/resources/services/org.duniter.core.beans.Bean b/duniter4j-es-user/src/test/resources/services/org.duniter.core.beans.Bean deleted file mode 100644 index 6613b03e0d6f746558b3ac98f065d36d74cf3411..0000000000000000000000000000000000000000 --- a/duniter4j-es-user/src/test/resources/services/org.duniter.core.beans.Bean +++ /dev/null @@ -1,13 +0,0 @@ -org.duniter.core.client.service.bma.BlockchainRemoteServiceImpl -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 -org.duniter.core.client.service.local.CurrencyServiceImpl -org.duniter.elasticsearch.dao.impl.CurrencyDaoImpl -org.duniter.elasticsearch.dao.impl.PeerDaoImpl -org.duniter.elasticsearch.dao.impl.BlockDaoImpl diff --git a/pom.xml b/pom.xml index 59bcf4f15f54a8966aead982ebabe199d6c355f1..b970f08a44a4746b309d9fb5509881fb3371ce40 100644 --- a/pom.xml +++ b/pom.xml @@ -36,12 +36,12 @@ <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> <tyrus.version>1.13.1</tyrus.version> - <jackson.version>2.8.1</jackson.version> + <jackson.version>2.9.0</jackson.version> <stringtemplate.version>4.0.2</stringtemplate.version> <jTextUtilsVersion>0.3.3</jTextUtilsVersion> + <lombok.version>1.16.20</lombok.version> <nuitonConfigVersion>3.0</nuitonConfigVersion> <nuitonVersionVersion>1.0-rc-2</nuitonVersionVersion> @@ -144,22 +144,18 @@ <module>duniter4j-core-shared</module> <module>duniter4j-core-client</module> <module>duniter4j-client</module> - <module>duniter4j-es-core</module> - <module>duniter4j-es-user</module> - <module>duniter4j-es-subscription</module> - <module>duniter4j-es-assembly</module> </modules> <scm> - <url>https://github.com/duniter/duniter4j.git</url> - <connection>scm:git:https://github.com/duniter/duniter4j.git</connection> - <developerConnection>scm:git:https://github.com/duniter/duniter4j.git</developerConnection> + <url>https://git.duniter.org/clients/java/duniter4j.git</url> + <connection>scm:git:https://git.duniter.org/clients/java/duniter4j.git</connection> + <developerConnection>scm:git:https://git.duniter.org/clients/java/duniter4j.git</developerConnection> <tag>HEAD</tag> </scm> <issueManagement> - <system>GitHub</system> - <url>https://github.com/duniter/duniter4j/issues</url> + <system>GitLab</system> + <url>https://git.duniter.org/clients/java/duniter4j/issues</url> </issueManagement> <dependencyManagement> @@ -276,6 +272,12 @@ <artifactId>j-text-utils</artifactId> <version>${jTextUtilsVersion}</version> </dependency> + <dependency> + <groupId>org.projectlombok</groupId> + <artifactId>lombok</artifactId> + <version>${lombok.version}</version> + <scope>provided</scope> + </dependency> <!-- NaCL lib --> <dependency> @@ -296,12 +298,6 @@ <artifactId>scrypt</artifactId> <version>${scrypt.version}</version> </dependency> - <!-- elastic search --> - <dependency> - <groupId>org.elasticsearch</groupId> - <artifactId>elasticsearch</artifactId> - <version>${elasticsearch.version}</version> - </dependency> <!-- JNA (need for OS shutdown hook) --> <dependency> <groupId>net.java.dev.jna</groupId> @@ -897,7 +893,7 @@ <escapeHTML>false</escapeHTML> <feedType>rss_2.0</feedType> <issueLinkTemplatePerSystem> - <default>https://github.com/duniter/duniter4j/issues/%ISSUE%</default> + <default>https://git.duniter.org/clients/java/duniter4j/issues/%ISSUE%</default> </issueLinkTemplatePerSystem> </configuration> </plugin> diff --git a/src/site/markdown/ES.md b/src/site/markdown/ES.md deleted file mode 100644 index ece05db4ba9bf2f98b3e53937f1a7c5bc0966426..0000000000000000000000000000000000000000 --- a/src/site/markdown/ES.md +++ /dev/null @@ -1,197 +0,0 @@ -# ElastiSearch node - -## Install - -### Prerequisites - -#### Install Java - - - Install Java JRE 8 or more. - - - Windows: see [Oracle web site](http://oracle.com/java/index.html) - - - Linux (Ubuntu): - -```bash -sudo apt-get install openjdk-8-jre -``` - -### Install libsodium - -[The Sodium crypto library (libsodium)](https://download.libsodium.org/doc/installation/) is a modern, easy-to-use software library for encryption, decryption, signatures, password hashing and more. - -- Get libsodium (version 1.0.14 or newer) - -```bash -wget -kL https://github.com/jedisct1/libsodium/releases/download/1.0.14/libsodium-1.0.14.tar.gz -tar -xvf libsodium-1.0.14.tar.gz -``` - -- Installation: - -```bash -cd libsodium-1.0.14 -sudo apt-get install build-essential -sudo ./configure -sudo make && sudo make check -sudo make install -``` - -### Install bundle (ElasticSearch + Duniter4j) - - - Download [lastest release](https://github.com/duniter/duniter4j/releases) of file duniter4j-es-X.Y-standalone.zip - - - Unzip - -```bash -unzip duniter4j-es-X.Y-standalone.zip -cd duniter4j-es-X.Y/config -``` - - - Edit the configuration file `config/elasticsearch.yml`, in particular this properties: - -```yml -# Your ES cluster name -cluster.name: duniter4j-elasticsearch - -# Use a descriptive name for the node: -node.name: ES-NODE-1 - -# Set the bind address to a specific IP (IPv4 or IPv6): -network.host: 192.168.0.28 - -# Set a custom port for HTTP: -http.port: 9203 - -# Duniter node to connect with -duniter.host: g1-test.duniter.org -duniter.port: 10900 - -# Initial list of hosts to perform synchronization -duniter.p2p.includes.endpoints: [ - "ES_CORE_API g1-test.data.duniter.fr 443", - "ES_USER_API g1-test.data.duniter.fr 443", - "ES_SUBSCRIPTION_API g1-test.data.duniter.fr 443" -] - -``` - - - Launch the node - -```bash -cd duniter4j-es-X.Y/bin -./elasticsearch -``` - -Output example (on [G1-test](http://g1-test.duniter.fr) currency): - -```bash -$ ./elasticsearch -[2016-09-24 00:16:45,803][INFO ][node ] [ES-NODE-1] version[2.3.3], pid[15365], build[218bdf1/2016-05-17T15:40:04Z] -[2016-09-24 00:16:45,804][INFO ][node ] [ES-NODE-1] initializing ... -[2016-09-24 00:16:46,257][INFO ][plugins ] [ES-NODE-1] modules [reindex, lang-expression, lang-groovy], plugins [mapper-attachments, duniter4j-elasticsearch], sites [duniter4j-elasticsearch] -[2016-09-24 00:16:46,270][INFO ][env ] [ES-NODE-1] using [1] data paths, mounts [[/home (/dev/mapper/isw_defjaaicfj_Volume1p1)]], net usable_space [1tb], net total_space [1.7tb], spins? [possibly], types [ext4] -[2016-09-24 00:16:46,270][INFO ][env ] [ES-NODE-1] heap size [989.8mb], compressed ordinary object pointers [true] -[2016-09-24 00:16:47,757][INFO ][node ] [ES-NODE-1] initialized -[2016-09-24 00:16:47,757][INFO ][node ] [ES-NODE-1] starting ... -[2016-09-24 00:16:47,920][INFO ][transport ] [ES-NODE-1] publish_address {192.168.0.5:9300}, bound_addresses {192.168.0.5:9300} -[2016-09-24 00:16:47,924][INFO ][discovery ] [ES-NODE-1] duniter4j-elasticsearch/jdzzh_jUTbuN26Enl-9whQ -[2016-09-24 00:16:50,982][INFO ][cluster.service ] [ES-NODE-1] detected_master {EIS-DEV}{FD0IzkxETM6tyOqzrKuVYw}{192.168.0.28}{192.168.0.28:9300}, added {{EIS-DEV}{FD0IzkxETM6tyOqzrKuVYw}{192.168.0.28}{192.168.0.28:9300},}, reason: zen-disco-receive(from master [{EIS-DEV}{FD0IzkxETM6tyOqzrKuVYw}{192.168.0.28}{192.168.0.28:9300}]) -[2016-09-24 00:16:53,570][INFO ][http ] [ES-NODE-1] publish_address {192.168.0.5:9203}, bound_addresses {192.168.0.5:9203} -[2016-09-24 00:16:53,570][INFO ][node ] [ES-NODE-1] started -[2016-09-24 00:16:57,850][INFO ][node ] Checking Duniter indices... -[2016-09-24 00:16:57,859][INFO ][node ] Checking Duniter indices... [OK] -[2016-09-24 00:17:08,026][INFO ][duniter.blockchain ] [g1-test] [g1-test.duniter.org:10900] Indexing last blocks... -[2016-09-24 00:17:08,026][INFO ][duniter.blockchain ] [g1-test] [g1-test.duniter.org:10900] Indexing block #999 / 41282 (2%)... -[2016-09-24 00:17:08,045][INFO ][duniter.blockchain ] [g1-test] [g1-test.duniter.org:10900] Indexing block #1998 / 41282 (4%)... -[2016-09-24 00:17:09,026][INFO ][duniter.blockchain ] [g1-test] [g1-test.duniter.org:10900] Indexing block #2997 / 41282 (6%)... -[2016-09-24 00:17:10,057][INFO ][duniter.blockchain ] [g1-test] [g1-test.duniter.org:10900] Indexing block #3996 / 41282 (8%)... -... -[2016-09-24 00:17:11,026][INFO ][duniter.blockchain ] [g1-gtest] [g1-test.duniter.org:10900] Indexing block #41282 - hash [00000AAD73B0E76B870E6779CD7ACCCE175802D7867C13B5C8ED077F380548C5] -``` - -### Test your node - -#### Using a web browser - -The following web address should works: http://localhost:9200/node/summary - -#### Using Cesium - -You should also be able to use your node in the [Cesium](https://github.com/duniter/cesium) application: - - - in the Cesium+ settings, replace the data node address; - - check if graph and profil avatar are display correctly. - - -## Request the ES node - -When a blockchain currency has been indexed, you can test some fun queries : - - - get a block by number (e.g the block #0): - - http://localhost:9200/g1-test/block/0 -> with some additional metadata given by ES - - http://localhost:9200/gtest/block/0/_source -> the original JSON block - - - Block #125 with only hash, dividend and memberCount: - - http://localhost:9200/gtest/block/125/_source?_source=number,hash,dividend,membersCount - - - All blocks using a pubkey (or whatever): - - http://localhost:9200/gtest/block/_search?q=9sbUKBMvJVxtEVhC4N9zV1GFTdaempezehAmtwA8zjKQ1 - - - All blocks with a dividend, with only some selected fields (like dividend, number, hahs). - Note : Query executed in command line, using CURL: - -```bash -curl -XGET 'http://localhost:9200/gtest/block/_search' -d '{ -"query": { - "filtered" : { - "filter": { - "exists" : { "field" : "dividend" } - } - } - }, - "_source": ["number", "dividend", "hash", "membersCount"] - }' -``` - - -More documentation here : - -- [a development tutorial](./development_tutorial.html) (french); - -- [ElasticSearch official web site](http://www.elastic.co/guide/en/elasticsearch/reference/1.3/docs-get.html#get-source-filtering) - -- [a good tutorial on ES query](http://okfnlabs.org/blog/2013/07/01/elasticsearch-query-tutorial.html) - - -## Troubleshooting - -### Could not find an implementation class. - -Message: - -```bash -java.lang.RuntimeException: java.lang.RuntimeException: Could not find an implementation class. - at org.duniter.core.util.websocket.WebsocketClientEndpoint.<init>(WebsocketClientEndpoint.java:56) - at org.duniter.core.client.service.bma.BlockchainRemoteServiceImpl.addNewBlockListener(BlockchainRemoteServiceImpl.java:545) - at org.duniter.elasticsearch.service.BlockchainService.listenAndIndexNewBlock(BlockchainService.java:106) -``` - -Cause: - -Plugin use Websocket to get notification from a Duniter nodes. The current library ([Tyrus](https://tyrus.java.net/)) is loaded throw java Service Loader, that need access to file `META-INF/services/javax.websocket.ContainerProvider` contains by Tyrus. -ElasticSearch use separated classloader, for each plugin, that disable access to META-INF resource. - -Solution : - -Move Tyrus libraries into elasticsearch `lib/` directory : - -```bash -cd <ES_HOME> -mv plugins/duniter4j-elasticsearch/tyrus-*.jar lib -mv plugins/duniter4j-elasticsearch/javax.websocket-api-*.jar lib -``` \ No newline at end of file diff --git a/src/site/markdown/ES_API.md b/src/site/markdown/ES_API.md deleted file mode 100644 index 1f161392d07abc0a5679383d663aae43fe3f0dd2..0000000000000000000000000000000000000000 --- a/src/site/markdown/ES_API.md +++ /dev/null @@ -1,191 +0,0 @@ - -# ES HTTP API - -## Contents - -- [Contents](#contents) -- [Overview](#overview) -- [ES CORE API](#ES_CORE_API) - * [currency](#acurrency) - * [currency/block](#acurrencyblock) - * [currency/blockstat](#acurrencyblockstat) - * [currency/peer](#acurrencypeer) - * [currency/tx](#acurrencytx) -- [ES USER API](#es_user_api) - * [user](#user) - * [user/event](#userevent) - * [user/profile](#userprofile) - * [user/settings](#usersettings) - * [message](#message) - * [message/inbox](#messageinbox) - * [message/oubox](#messageoutbox) - * [invitation](#invitation) - * [invitation/certification](#invitationcertification) -- [ES SUBSCRIPTION API](#ES SUBSCRIPTION API) - -## Overview - -Duniter4j Elasticsearch offer HTTP access to this sub-API: - -- `ES CORE API`: BlockChain indexation; -- `ES USER API`: User data indexation, such as: profiles, private messages, settings (crypted); -- `ES SUBSCRIPTION API`: User service configuration, such as: email notification service; - -Data is made accessible through an HTTP API : - -```text - http[s]://node[:port]/... - |-- <currency_name>/ - | |-- block - | |-- blockstat - | |-- peer - | `-- tx - |-- user/ - | |-- profile - | `-- settings - |-- message/ - | |-- inbox - | `-- outbox - `-- invitation/ - `-- certification -``` - -### Document format - -All stored documents use a JSON format. - -#### Data document - -Every document have the following mandatory fields: - -- `version` : The document's version. -- `issuer` : The document's emitter -- `hash`: the document's hash -- `signature`: the signature emitted by the issuer. Since `version: 2`, only the `hash` is signed. - -#### Deletion - -Document deletion use a document with this mandatory fields: - -- `index` : The document's index -- `type` : The document's type -- `issuer`: The deletion issuer. Should correspond to the document's `issuer`, or the `recipient` in some special case ([inbox message](#messageinbox) or [invitation](#invitation)) -- `time`: the current time -- `hash` -- `signature` - -For instance, a deletion on `message/inbox` should send this document: - -```json -{ - "version" : 2, - "index" : "message", - "type" : "inbox", - "id" : "AV9VOeOuTvXJwYisNfU6", - "issuer" : "F13aXKWQPGCjSQAxxTyJYyRyPm5SqzFSsYYWSDEQGi2A", - "time" : 1509806623, - "hash" : "61EBBFBCA630E8B715C360DDE1CD6CABD92B9267CA4B724A2F1F36F0FF7E3455", - "signature" : "FOkYCX1b05LTAbtz72F/LMWZb8F8zhQKEqcvbuiQy1N6AXtCUC5Xmjcn+NeO9sCLdcmA0HxsJx42GnWZOmKCDA==" -} -``` - -## ES CORE API - -### `<currency>/*` - -#### `<currency>/block` - - - Get the current block: `<currency>/block/current` - - Get a block by number: `<currency>/block/<number>` - - Search on blocks: `<currency>/block/_search` (POST or GET) - -#### `<currency>/blockstat` - -#### `<currency>/peer` - -#### `<currency>/tx` - -## ES USER API - -### `user/*` - -#### `user/event` - - - Get events on an account, by pubkey: `user/event/_search?q=issuer:<pubkey>` (GET) - - Search on events: `user/event/_search` (POST or GET) - -#### `user/profile` - - - - Get an profile, by public key: `user/profile/<pubkey>` - - Add a new profile: `user/profile` (POST) - - Update an existing profile: `user/profile/_update` (POST) - - Delete an existing invitation: `invitation/certification/_delete` (POST) - - Search on profiles: `user/profile/_search` (POST or GET) - -A profile document is a JSON document. Mandatory fields are: - - - `title`: user name (Lastanem, firstname...) - - `time`: submission time, in seconds - - `issuer`: user public key - - `hash`: hash of the JSON document (without fields `hash` and `signature`) - - `signature`: signature of the JSON document (without fields `hash` and `signature`) - -Example with only mandatory fields: - -```json -{ - "version" : 2, - "title" : "Pecquot Ludovic", - "description" : "Développeur Java et techno client-serveur\nParticipation aux #RML7, #EIS et #Sou", - "time" : 1488359903, - "issuer" : "2v6tXNxGC1BWaJtUFyPJ1wJ8rbz9v1ZVU1E1LEV2v4ss", - "hash" : "F66D43ECD4D38785F424ADB68B3EA13DD56DABDE275BBE780E81E8D4E1D0C5FA", - "signature" : "3CWxdLtyY8dky97RZBFLfP6axnfW8KUmhlkiaXC7BN98yg6xE9CkijRBGmuyrx3llPx5HeoGLG99DyvVIKZuCg==" -} -``` - -Some additional fields are `description`, `socials`, `tags` and `avatar` : - -```json -{ - "version" : 2, - "title" : "My profile name", - "description" : "#developer", - "city" : "Rennes", - "socials" : [ { - "type" : "diaspora", - "url" : "https://diaspora-fr.org/people/f9d13420f9ssqzq97aa01beea1f31e2" - } ], - "time" : 1487422234, - "tags" : [ "developer" ], - "issuer" : "2ny7YAdmzReQxAayyJZsyVYwYhVyax2thKcGknmQy5nQ", - "avatar" : { - "_content_type" : "image/png", - "_content" : "iVBORw0KGgoAAAANSUhEUgAAAGQAAABkC(...)" // base 64 encoding - } - "hash" : "85F527077D060E03ECAC6D1AE38A74CCC900ACAF5D52F194BA34F5A5E8A55139", - "signature" : "WeP7JEwttAoSkHcuiFwo6N4SM0uVakTYBQ09H1+K8/nPFyxO3ak1U9EQ6qaQFoAx9IdDp5qO2EX662wP/pcEAg==", -} -``` - -#### `user/settings` - -### `message/*` - -#### `message/inbox` - -#### `message/outbox` - -### `invitation/*` - -#### `invitation/certification` - - - Get an invitation, by id: `invitation/certification/<id>` - - Add a new invitation: `invitation/certification` (POST) - - Delete an existing invitation: `invitation/certification/_delete` (POST) - - Search on invitations: `invitation/certification/_search` (POST or GET) - -## ES SUBSCRIPTION API - -TODO \ No newline at end of file diff --git a/src/site/markdown/development_tutorial.md b/src/site/markdown/development_tutorial.md index cee4419bae560ba57c51bff327193e0e79f9be19..c2404b49d71c8c0d2cd804f334252d9c9f871201 100644 --- a/src/site/markdown/development_tutorial.md +++ b/src/site/markdown/development_tutorial.md @@ -12,32 +12,22 @@ Le projet Duniter4j est composé de plusieurs sous-modules : - `duniter4j-core-shared`: Classes utilitaires Java. Réutilisable dans d'autres projets Java autour de Duniter. - `duniter4j-core-client`: Ensemble de services Java permettant d'accéder à un réseau Duniter (c'est à dire une API Java client Duniter) . Cette partie est **réutilisable dans d'autres applications Java**. - -- `duniter4j-es-*`: Les plugins ElasticSearch, qui implémentent : - - * `duniter4j-es-core`: Indexation de BlockChain Duniter (ESA ou ES API); - - * `duniter4j-es-user`: Indexation de données utilisateurs (profils, des messages privées, paramètres chiffrés) (ESUA ou ES USER API); - - * `duniter4j-es-subscription`: Indexation des abonnements en ligne (notifications par email); - - * `duniter4j-es-assembly`: gestion des livrables (packaging). ## Niveau I : récupérer le code source Ce premier niveau consiste à créer *votre propre version* des sources du logiciel et de récupérer cette copie sur votre ordinateur. Vous y produirez : -- Votre propre compte *GitHub* +- Votre propre compte *GitLab* - Votre propre version du logiciel, votre *fork* - Une copie locale des fichiers de code source provenant de votre *fork* -### Créez un compte GitHub +### Créez un compte GitLab > Si vous disposez déjà d'un compte GitHub, vous pouvez passer cette étape. -Rendez-vous sur https://github.com (site en anglais). Renseigner les 3 champs proposés : +Rendez-vous sur https://git.duniter.org (site en anglais). Renseigner les 3 champs proposés : - Nom d'utilisateur @@ -45,15 +35,11 @@ Rendez-vous sur https://github.com (site en anglais). Renseigner les 3 champs pr - Mot de passe -<img src="https://forum.duniter.org/uploads/default/original/1X/13ade346327b73bbf1acc97027af147eeb4e9089.png" width="346" height="325"/> - Vous recevrez probablement un e-mail de confirmation qu'il vous faudra valider. Une fois cette étape passée, vous devriez disposer d'un compte GitHub . ### Forkez le dépôt principal -Rendez-vous à l'adresse https://github.com/duniter/duniter4j. Cliquez sur le bouton « Fork » en dans le coin supérieur droit de la page : - -<img src="https://forum.duniter.org/uploads/default/original/1X/3b9228c664520496d6a7e86e3f9c4c438f111914.png" width="388" height="98"/> +Rendez-vous à l'adresse https://git.duniter.org/clients/java/duniter4j. Cliquez sur le bouton « Fourcher » (en haut de la page) ### Installer Git @@ -85,7 +71,7 @@ Vous n'avez plus qu'à retourner dans votre console Git et saisir : ce qui donne dans mon cas : ``` -git clone https://github.com/blavenie/duniter4j.git +git clone git@git.duniter.org:clients/java/duniter4j.git Cloning into 'duniter4j'... (...) Checking connectivity... done. @@ -159,39 +145,28 @@ Ce troisième niveau permet de découvrir les quelques commandes que vous utilis ### Configurer le projet -La configuration utilisée pour le développement est visible dans le fichier : `/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml` +La configuration utilisée pour le développement est visible dans le fichier : `/duniter4j-client/src/main/filtered-resources/duniter4j-client.config` #### Configuration du noeud Duniter Si vous avez un noeud Duniter qui est lancé localement, configurez le en modifiant les propriétés suivantes : -```yml +```properties # -# Duniter node to synchronize +# Duniter node: # -duniter.host: localhost -duniter.port: 10901 <- à remplacer par le port de votre noeud +duniter.node.host=localhost +duniter.node.port=10901 <- à remplacer par le port de votre noeud ``` -Si vous n'avez pas de noeud local, conservez la configuration par défaut : +Si vous n'avez pas de noeud local, utiliser la configuration suivante : -```yml +```properties # -# Duniter node to synchronize +# Duniter node: # -duniter.host: g1.duniter.org -duniter.port: 10901 -``` - -#### Désactivation de la couche de sécurité - -Duniter4j a une couche de sécurité, qui empêche l'appel à des URL non autorisées. -Nous allons **désactiver cette couche de sécurité** pour faciliter l'apprentissage qui va suivre. - -Modifiez comme suit, modifier le fichier de configuration `duniter4j-elasticsearch/src/main/assembly/config/elasticsearch.yml` : - -```bash -duniter.security.enable: false +duniter.node.host=g1.duniter.org +duniter.node.port=10901 ``` ### Compiler le projet @@ -210,22 +185,18 @@ Si tout c'est bien passé, vous devriez obtenir quelque chose qui ressemble à c ```bash (...) -[INFO] Building zip: /home/eis/git/duniter/duniter4j/duniter4j-es-assembly/target/duniter4j-es-0.3.5-SNAPSHOT-standalone.zip +[INFO] Building jar: /media/data/home/blavenie/git/duniter/duniter4j/duniter4j-client/target/duniter4j-client-1.0.4-SNAPSHOT.jar [INFO] -[INFO] --- maven-install-plugin:2.4:install (default-install) @ duniter4j-es-assembly --- -[INFO] Installing /home/eis/git/duniter/duniter4j/duniter4j-es-assembly/pom.xml to /home/eis/.m2/repository/org/duniter/duniter4j-es-assembly/0.3.5-SNAPSHOT/duniter4j-es-assembly-0.3.5-SNAPSHOT.pom -[INFO] Installing /home/eis/git/duniter/duniter4j/duniter4j-es-assembly/target/duniter4j-es-0.3.5-SNAPSHOT-standalone.zip to /home/eis/.m2/repository/org/duniter/duniter4j-es-assembly/0.3.5-SNAPSHOT/duniter4j-es-assembly-0.3.5-SNAPSHOT-standalone.zip +[INFO] --- maven-install-plugin:2.5.2:install (default-install) @ duniter4j-client --- +[INFO] Installing /media/data/home/blavenie/git/duniter/duniter4j/duniter4j-client/target/duniter4j-client-1.0.4-SNAPSHOT.jar to /home/blavenie/.m2/repository/org/duniter/duniter4j-client/1.0.4-SNAPSHOT/duniter4j-client-1.0.4-SNAPSHOT.jar +[INFO] Installing /media/data/home/blavenie/git/duniter/duniter4j/duniter4j-client/pom.xml to /home/blavenie/.m2/repository/org/duniter/duniter4j-client/1.0.4-SNAPSHOT/duniter4j-client-1.0.4-SNAPSHOT.pom [INFO] ------------------------------------------------------------------------ [INFO] Reactor Summary: [INFO] -[INFO] Duniter4j : a Duniter Java Client API ............. SUCCESS [0.476s] -[INFO] Duniter4j :: Core Shared .......................... SUCCESS [4.152s] -[INFO] Duniter4j :: Core Client API ...................... SUCCESS [5.633s] -[INFO] Duniter4j :: ElasticSearch Core plugin ............ SUCCESS [8.954s] -[INFO] Duniter4j :: ElasticSearch User plugin ............ SUCCESS [1.039s] -[INFO] Duniter4j :: ElasticSearch Subscription plugin .... SUCCESS [0.804s] -[INFO] Duniter4j :: ElasticSearch Assembly ............... SUCCESS [4.747s] - +[INFO] Duniter4j .......................................... SUCCESS [ 0.466 s] +[INFO] Duniter4j :: Core Shared ........................... SUCCESS [ 1.790 s] +[INFO] Duniter4j :: Core Client API ....................... SUCCESS [ 9.069 s] +[INFO] Duniter4j :: Client ................................ SUCCESS [ 3.383 s] [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ diff --git a/src/site/markdown/index.md b/src/site/markdown/index.md index 528a3f5886759ea23cd45401d52606eb750b94bf..d4e02f81b3f77537740af91ae519f018282c6fae 100644 --- a/src/site/markdown/index.md +++ b/src/site/markdown/index.md @@ -13,9 +13,3 @@ Duniter4j has tree main modules : - `duniter4j-client`: [a command line tool](./CLI.html), to execute basic operation on a Duniter currency: transfer, view peers, ... - `duniter4j-core-client`: [a Java API](./Java_API.html) to help Java developers to communicate with a Duniter network. - -- `duniter4j-elasticsearch`: [a ElastiSearch node](./ES.html) used to store (with full-text capabilities) all blockchain data, and additional user data. - - * It comes with an [HTTP API](./ES_API.html) to store and retrieve all this data. - - * This API is used by [Cesium+](https://www.github.com/duniter/cesium) (a Duniter wallet).