diff --git a/docs/manual/docs/administrator-guide/configuring-the-catalog/img/application-feedback-link.png b/docs/manual/docs/administrator-guide/configuring-the-catalog/img/application-feedback-link.png new file mode 100644 index 00000000000..a52132a35dd Binary files /dev/null and b/docs/manual/docs/administrator-guide/configuring-the-catalog/img/application-feedback-link.png differ diff --git a/docs/manual/docs/administrator-guide/configuring-the-catalog/img/application-feedback.png b/docs/manual/docs/administrator-guide/configuring-the-catalog/img/application-feedback.png new file mode 100644 index 00000000000..fa54aa2317b Binary files /dev/null and b/docs/manual/docs/administrator-guide/configuring-the-catalog/img/application-feedback.png differ diff --git a/docs/manual/docs/administrator-guide/configuring-the-catalog/system-configuration.md b/docs/manual/docs/administrator-guide/configuring-the-catalog/system-configuration.md index 8f9f567d1e4..d56eb454ffa 100644 --- a/docs/manual/docs/administrator-guide/configuring-the-catalog/system-configuration.md +++ b/docs/manual/docs/administrator-guide/configuring-the-catalog/system-configuration.md @@ -91,15 +91,18 @@ See [Configuring Shibboleth](../managing-users-and-groups/authentication-mode.md Enable the self registration form. See [User Self-Registration](../managing-users-and-groups/user-self-registration.md). -You can configure optionally re-Captcha, to protect you and your users from spam and abuse. And a list of email domains (separated by commas) +You can configure optionally re-Captcha, to protect you and your users from spam and abuse. And a list of email domains (separated by commas) that can request an account. If not configured any email address is allowed. -## system/userFeedback +## User application feedback -!!! warning "Deprecated" +Enabling the setting, displays in the application footer a link to a page that allows sending comments about the application. + +![](img/application-feedback-link.png) - 3.0.0 +![](img/application-feedback.png) +It requires an email server configured. ## Link in metadata records diff --git a/services/src/main/java/org/fao/geonet/api/site/SiteApi.java b/services/src/main/java/org/fao/geonet/api/site/SiteApi.java index d39d42f5134..a2bd724fa59 100644 --- a/services/src/main/java/org/fao/geonet/api/site/SiteApi.java +++ b/services/src/main/java/org/fao/geonet/api/site/SiteApi.java @@ -1,5 +1,5 @@ /* - * Copyright (C) 2001-2023 Food and Agriculture Organization of the + * Copyright (C) 2001-2024 Food and Agriculture Organization of the * United Nations (FAO-UN), United Nations World Food Programme (WFP) * and United Nations Environment Programme (UNEP) * @@ -36,17 +36,16 @@ import jeeves.server.context.ServiceContext; import jeeves.xlink.Processor; import org.apache.commons.lang3.StringUtils; -import org.fao.geonet.ApplicationContextHolder; -import org.fao.geonet.GeonetContext; -import org.fao.geonet.NodeInfo; -import org.fao.geonet.SystemInfo; +import org.fao.geonet.*; import org.fao.geonet.api.ApiParams; import org.fao.geonet.api.ApiUtils; import org.fao.geonet.api.OpenApiConfig; +import org.fao.geonet.api.exception.FeatureNotEnabledException; import org.fao.geonet.api.exception.NotAllowedException; import org.fao.geonet.api.site.model.SettingSet; import org.fao.geonet.api.site.model.SettingsListResponse; import org.fao.geonet.api.tools.i18n.LanguageUtils; +import org.fao.geonet.api.users.recaptcha.RecaptchaChecker; import org.fao.geonet.constants.Geonet; import org.fao.geonet.doi.client.DoiManager; import org.fao.geonet.domain.*; @@ -69,6 +68,7 @@ import org.fao.geonet.repository.*; import org.fao.geonet.repository.specification.MetadataSpecs; import org.fao.geonet.resources.Resources; +import org.fao.geonet.util.MailUtil; import org.fao.geonet.utils.FilePathChecker; import org.fao.geonet.utils.Log; import org.fao.geonet.utils.ProxyInfo; @@ -78,15 +78,10 @@ import org.springframework.http.HttpEntity; import org.springframework.http.HttpStatus; import org.springframework.http.MediaType; +import org.springframework.http.ResponseEntity; import org.springframework.security.access.prepost.PreAuthorize; import org.springframework.stereotype.Controller; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.PutMapping; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RequestMethod; -import org.springframework.web.bind.annotation.RequestParam; -import org.springframework.web.bind.annotation.ResponseBody; -import org.springframework.web.bind.annotation.ResponseStatus; +import org.springframework.web.bind.annotation.*; import javax.imageio.ImageIO; import javax.servlet.http.HttpServletRequest; @@ -100,19 +95,12 @@ import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.StandardCopyOption; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.HashMap; -import java.util.List; -import java.util.ListIterator; -import java.util.Map; -import java.util.Optional; -import java.util.TimeZone; +import java.util.*; import static org.apache.commons.fileupload.util.Streams.checkFileName; import static org.fao.geonet.api.ApiParams.API_CLASS_CATALOG_TAG; import static org.fao.geonet.constants.Geonet.Path.IMPORT_STYLESHEETS_SCHEMA_PREFIX; +import static org.fao.geonet.kernel.setting.Settings.SYSTEM_FEEDBACK_EMAIL; /** * @@ -201,7 +189,7 @@ public static void reloadServices(ServiceContext context) throws Exception { @ResponseBody public SettingsListResponse getSiteOrPortalDescription( @Parameter(hidden = true) - HttpServletRequest request + HttpServletRequest request ) throws Exception { SettingsListResponse response = new SettingsListResponse(); response.setSettings(settingManager.getSettings(new String[]{ @@ -267,7 +255,7 @@ public SettingsListResponse getSettingsSet( @RequestParam( required = false ) - SettingSet[] set, + SettingSet[] set, @Parameter( description = "Setting key", required = false @@ -275,11 +263,11 @@ public SettingsListResponse getSettingsSet( @RequestParam( required = false ) - String[] key, + String[] key, @Parameter( hidden = true ) - HttpSession httpSession + HttpSession httpSession ) throws Exception { ConfigurableApplicationContext appContext = ApplicationContextHolder.get(); UserSession session = ApiUtils.getUserSession(httpSession); @@ -353,7 +341,7 @@ public List getSettingsDetails( @RequestParam( required = false ) - SettingSet[] set, + SettingSet[] set, @Parameter( description = "Setting key", required = false @@ -361,9 +349,9 @@ public List getSettingsDetails( @RequestParam( required = false ) - String[] key, + String[] key, @Parameter(hidden = true) - HttpSession httpSession + HttpSession httpSession ) throws Exception { UserSession session = ApiUtils.getUserSession(httpSession); Profile profile = session == null ? null : session.getProfile(); @@ -415,7 +403,7 @@ public List getSettingsDetails( public void saveSettings( @Parameter(hidden = false) @RequestParam - Map allRequestParams, + Map allRequestParams, HttpServletRequest request ) throws Exception { ApplicationContext applicationContext = ApplicationContextHolder.get(); @@ -450,7 +438,7 @@ public void saveSettings( // Update the system default timezone. If the setting is blank use the timezone user.timezone property from command line or // TZ environment variable String zoneId = StringUtils.defaultIfBlank(settingManager.getValue(Settings.SYSTEM_SERVER_TIMEZONE, true), - SettingManager.DEFAULT_SERVER_TIMEZONE.getId()); + SettingManager.DEFAULT_SERVER_TIMEZONE.getId()); TimeZone.setDefault(TimeZone.getTimeZone(zoneId)); @@ -534,7 +522,7 @@ public boolean isCasEnabled( @PreAuthorize("hasAuthority('Administrator')") public void updateStagingProfile( @PathVariable - SystemInfo.Staging profile) { + SystemInfo.Staging profile) { this.info.setStagingProfile(profile.toString()); } @@ -582,22 +570,22 @@ public HttpEntity indexSite( @Parameter(description = "Drop and recreate index", required = false) @RequestParam(required = false, defaultValue = "true") - boolean reset, + boolean reset, @Parameter(description = "Asynchronous mode (only on all records. ie. no selection bucket)", required = false) @RequestParam(required = false, defaultValue = "false") - boolean asynchronous, + boolean asynchronous, @Parameter(description = "Index. By default only remove record index.", required = false) @RequestParam(required = false, defaultValue = "records") - String[] indices, + String[] indices, @Parameter( description = ApiParams.API_PARAM_BUCKET_NAME, required = false) @RequestParam( required = false ) - String bucket, + String bucket, HttpServletRequest request ) throws Exception { ServiceContext context = ApiUtils.createServiceContext(request); @@ -779,7 +767,7 @@ public ProxyConfiguration getProxyConfiguration( public void setLogo( @Parameter(description = "Logo to use for the catalog") @RequestParam("file") - String file, + String file, @Parameter( description = "Create favicon too", required = false @@ -788,7 +776,7 @@ public void setLogo( defaultValue = "false", required = false ) - boolean asFavicon, + boolean asFavicon, HttpServletRequest request ) throws Exception { @@ -901,4 +889,77 @@ public List getXslTransformations( return list; } } + + + @io.swagger.v3.oas.annotations.Operation( + summary = "Send an email to catalogue administrator with feedback about the application", + description = "") + @PostMapping( + value = "/userfeedback", + produces = MediaType.APPLICATION_JSON_VALUE + ) + @ResponseStatus(HttpStatus.CREATED) + @ResponseBody + public ResponseEntity sendApplicationUserFeedback( + @Parameter( + description = "Recaptcha validation key." + ) + @RequestParam(required = false, defaultValue = "") final String recaptcha, + @Parameter( + description = "User name.", + required = true + ) + @RequestParam final String name, + @Parameter( + description = "User organisation.", + required = true + ) + @RequestParam final String org, + @Parameter( + description = "User email address.", + required = true + ) + @RequestParam final String email, + @Parameter( + description = "A comment or question.", + required = true + ) + @RequestParam final String comments, + @Parameter(hidden = true) final HttpServletRequest request + ) throws Exception { + Locale locale = languageUtils.parseAcceptLanguage(request.getLocales()); + ResourceBundle messages = ResourceBundle.getBundle("org.fao.geonet.api.Messages", locale); + + boolean feedbackEnabled = settingManager.getValueAsBool(Settings.SYSTEM_USERFEEDBACK_ENABLE, false); + if (!feedbackEnabled) { + throw new FeatureNotEnabledException( + "Application feedback is not enabled.") + .withMessageKey("exception.resourceNotEnabled.applicationFeedback") + .withDescriptionKey("exception.resourceNotEnabled.applicationFeedback.description"); + } + + boolean recaptchaEnabled = settingManager.getValueAsBool(Settings.SYSTEM_USERSELFREGISTRATION_RECAPTCHA_ENABLE); + + if (recaptchaEnabled) { + boolean validRecaptcha = RecaptchaChecker.verify(recaptcha, + settingManager.getValue(Settings.SYSTEM_USERSELFREGISTRATION_RECAPTCHA_SECRETKEY)); + if (!validRecaptcha) { + return new ResponseEntity<>( + messages.getString("recaptcha_not_valid"), HttpStatus.PRECONDITION_FAILED); + } + } + + String to = settingManager.getValue(SYSTEM_FEEDBACK_EMAIL); + + Set toAddress = new HashSet<>(); + toAddress.add(to); + + MailUtil.sendMail(new ArrayList<>(toAddress), + messages.getString("site_user_feedback_title"), + String.format( + messages.getString("site_user_feedback_text"), + name, email, org, comments), + settingManager); + return new ResponseEntity<>(HttpStatus.CREATED); + } } diff --git a/web-ui/src/main/resources/catalog/components/contactus/ContactUsDirective.js b/web-ui/src/main/resources/catalog/components/contactus/ContactUsDirective.js index 1eed76abf55..10df5d42a19 100644 --- a/web-ui/src/main/resources/catalog/components/contactus/ContactUsDirective.js +++ b/web-ui/src/main/resources/catalog/components/contactus/ContactUsDirective.js @@ -1,5 +1,5 @@ /* - * Copyright (C) 2001-2016 Food and Agriculture Organization of the + * Copyright (C) 2001-2024 Food and Agriculture Organization of the * United Nations (FAO-UN), United Nations World Food Programme (WFP) * and United Nations Environment Programme (UNEP) * @@ -31,28 +31,81 @@ */ module.directive("gnContactUsForm", [ "$http", - function ($http) { + "$translate", + "vcRecaptchaService", + "gnConfigService", + "gnConfig", + function ($http, $translate, vcRecaptchaService, gnConfigService, gnConfig) { return { restrict: "A", replace: true, scope: { user: "=" }, - templateUrl: "../../catalog/components/share/" + "partials/contactusform.html", + templateUrl: + "../../catalog/components/contactus/" + "partials/contactusform.html", link: function (scope, element, attrs) { - scope.send = function (formId) { - $http({ - url: "contact.send@json", - method: "POST", - data: $(formId).serialize(), - headers: { "Content-Type": "application/x-www-form-urlencoded" } - }).then(function (response) { - // TODO: report no email sent - if (response.status === 200) { - scope.success = true; - } else { + gnConfigService.load().then(function (c) { + scope.recaptchaEnabled = + gnConfig["system.userSelfRegistration.recaptcha.enable"]; + scope.recaptchaKey = + gnConfig["system.userSelfRegistration.recaptcha.publickey"]; + }); + + scope.resolveRecaptcha = false; + + function initModel() { + scope.feedbackModel = { + name: scope.user.name, + email: scope.user.email, + org: "", + comments: "" + }; + } + + initModel(); + + scope.send = function (form, formId) { + if (scope.recaptchaEnabled) { + if (vcRecaptchaService.getResponse() === "") { + scope.resolveRecaptcha = true; + + var deferred = $q.defer(); + deferred.resolve(""); + return deferred.promise; } - }); + scope.resolveRecaptcha = false; + scope.captcha = vcRecaptchaService.getResponse(); + $("#recaptcha").val(scope.captcha); + } + + if (form.$valid) { + $http({ + url: "../api/site/userfeedback", + method: "POST", + data: $(formId).serialize(), + headers: { + "Content-Type": "application/x-www-form-urlencoded" + } + }).then( + function (response) { + scope.$emit("StatusUpdated", { + msg: $translate.instant("feebackSent"), + timeout: 2, + type: "info" + }); + initModel(); + }, + function (response) { + scope.success = false; + scope.$emit("StatusUpdated", { + msg: $translate.instant("feebackSentError"), + timeout: 0, + type: "danger" + }); + } + ); + } }; } }; diff --git a/web-ui/src/main/resources/catalog/components/contactus/partials/contactusform.html b/web-ui/src/main/resources/catalog/components/contactus/partials/contactusform.html index 716d5e9dc26..059b1bcd449 100644 --- a/web-ui/src/main/resources/catalog/components/contactus/partials/contactusform.html +++ b/web-ui/src/main/resources/catalog/components/contactus/partials/contactusform.html @@ -1,31 +1,74 @@ -
+ -
- +
+
- +
-
- +
+
- +
-
- +
+
- +
-
- +
+
-
- -

feebackSent

diff --git a/web-ui/src/main/resources/catalog/js/CatController.js b/web-ui/src/main/resources/catalog/js/CatController.js index 4d3e704d994..79b66951b47 100644 --- a/web-ui/src/main/resources/catalog/js/CatController.js +++ b/web-ui/src/main/resources/catalog/js/CatController.js @@ -1,5 +1,5 @@ /* - * Copyright (C) 2001-2016 Food and Agriculture Organization of the + * Copyright (C) 2001-2024 Food and Agriculture Organization of the * United Nations (FAO-UN), United Nations World Food Programme (WFP) * and United Nations Environment Programme (UNEP) * @@ -1655,6 +1655,10 @@ return gnGlobalSettings.gnCfg.mods.footer.showApplicationInfoAndLinksInFooter; }; + $scope.getContactusVisible = function () { + return gnConfig[gnConfig.key.isFeedbackEnabled]; + }; + function detectNode(detector) { if (detector.regexp) { var res = new RegExp(detector.regexp).exec(location.pathname); diff --git a/web-ui/src/main/resources/catalog/js/ContactUsController.js b/web-ui/src/main/resources/catalog/js/ContactUsController.js index 18876c80ca7..f510439db18 100644 --- a/web-ui/src/main/resources/catalog/js/ContactUsController.js +++ b/web-ui/src/main/resources/catalog/js/ContactUsController.js @@ -26,9 +26,17 @@ goog.require("gn_contactus_directive"); - var module = angular.module("gn_contact_us_controller", ["gn_contactus_directive"]); + var module = angular.module("gn_contact_us_controller", [ + "gn_contactus_directive", + "vcRecaptcha" + ]); - module.constant("$LOCALES", ["core"]); + module.config([ + "$LOCALES", + function ($LOCALES) { + $LOCALES.push("/../api/i18n/packages/search"); + } + ]); /** * diff --git a/web-ui/src/main/resources/catalog/locales/en-admin.json b/web-ui/src/main/resources/catalog/locales/en-admin.json index ec570ee7c95..30282dec251 100644 --- a/web-ui/src/main/resources/catalog/locales/en-admin.json +++ b/web-ui/src/main/resources/catalog/locales/en-admin.json @@ -852,9 +852,9 @@ "userFeedbackList": "Last user feedbacks", "system/userFeedback": "User feedback", "system/userFeedback/enable": "Enable application feedback", - "system/userFeedback/enable-help": "Enabling this option allows to send feedback about the application to the system administrator. It requires the mail server is also configured.", + "system/userFeedback/enable-help": "Enabling this option allows to send feedback about the application to the system administrator. It requires the mail server to be configured.", "system/userFeedback/metadata/enable": "Enable metadata feedback", - "system/userFeedback/metadata/enable-help": "Enabling this option allows to feedback to the metadata owner and system administrator about metadata record. It requires the mail server is also configured.", + "system/userFeedback/metadata/enable-help": "Enabling this option allows to feedback to the metadata owner and system administrator about metadata record. It requires the mail server to be configured.", "system/xlinkResolver": "Metadata XLink", "system/xlinkResolver/enable": "Enable XLink resolution", "system/xlinkResolver/enable-help": "If set, XLinks to metadata fragments in records will be resolved.", diff --git a/web-ui/src/main/resources/catalog/locales/en-core.json b/web-ui/src/main/resources/catalog/locales/en-core.json index e18cb2c1325..cd03dca04eb 100644 --- a/web-ui/src/main/resources/catalog/locales/en-core.json +++ b/web-ui/src/main/resources/catalog/locales/en-core.json @@ -126,6 +126,7 @@ "featureCatalog": "Feature catalog", "frequency": "Frequency", "feebackSent": "Your message has been sent to the catalog manager.", + "feebackSentError": "An error occurred sending your to the catalog manager. Please try again later, contact the service provider, or report this issue.", "feedbackNotEnable": "Feedback is not enabled.", "filter": "Filter", "filterSearch": "Display search options", diff --git a/web-ui/src/main/resources/catalog/views/default/less/gn_contact_us_default.less b/web-ui/src/main/resources/catalog/views/default/less/gn_contact_us_default.less new file mode 100644 index 00000000000..d5c88c43aed --- /dev/null +++ b/web-ui/src/main/resources/catalog/views/default/less/gn_contact_us_default.less @@ -0,0 +1,15 @@ +@import "../../../style/gn_contact_us.less"; + +// nojs styles +.gn-nojs { + .gn-top-search { + padding: 30px 0; + margin-bottom: 30px; + .gn-form-any input.input-lg { + height: 46px; + } + .btn-lg { + padding: 13px 16px; + } + } +} diff --git a/web-ui/src/main/resources/catalog/views/default/templates/footer.html b/web-ui/src/main/resources/catalog/views/default/templates/footer.html index 018b6eedb41..eb006285b99 100644 --- a/web-ui/src/main/resources/catalog/views/default/templates/footer.html +++ b/web-ui/src/main/resources/catalog/views/default/templates/footer.html @@ -12,6 +12,11 @@ about +
  • + + contact + +