From 29f51f524dc87cb6d5dd29d948b77a305757ef28 Mon Sep 17 00:00:00 2001
From: Johannes Hoermann <j.hoermann@adito.de>
Date: Mon, 1 Jul 2019 13:48:06 +0200
Subject: [PATCH] Webservice

---
 .../AddressValidation_entity.aod              |  20 +-
 .../children/type_param/valueProcess.js       |   3 +-
 .../currentvalue_param/valueProcess.js        |   2 +-
 .../_____LANGUAGE_EXTRA.aod                   |   3 +
 .../_____PREFERENCES_PROJECT.aod              |   4 +-
 process/DataCaching_lib/process.js            |   4 +-
 process/WsValidation_lib/process.js           | 181 +++++++++++++++---
 7 files changed, 167 insertions(+), 50 deletions(-)

diff --git a/entity/AddressValidation_entity/AddressValidation_entity.aod b/entity/AddressValidation_entity/AddressValidation_entity.aod
index 2f0920361f..7544eed7e3 100644
--- a/entity/AddressValidation_entity/AddressValidation_entity.aod
+++ b/entity/AddressValidation_entity/AddressValidation_entity.aod
@@ -115,14 +115,6 @@
       <name>STREET</name>
       <title>Street</title>
     </entityField>
-    <entityField>
-      <name>FROM_BN</name>
-      <title>From buildingnumber</title>
-    </entityField>
-    <entityField>
-      <name>TO_BN</name>
-      <title>To buildingnumber</title>
-    </entityField>
     <entityField>
       <name>LON</name>
     </entityField>
@@ -132,10 +124,14 @@
     <entityField>
       <name>CITYEXT</name>
     </entityField>
+    <entityField>
+      <name>BUILDINGNO</name>
+    </entityField>
   </entityFields>
   <recordContainers>
     <jDitoRecordContainer>
       <name>jdito</name>
+      <jDitoRecordAlias></jDitoRecordAlias>
       <isFilterable v="true" />
       <isRequireContainerFiltering v="true" />
       <contentProcess>%aditoprj%/entity/AddressValidation_entity/recordcontainers/jdito/contentProcess.js</contentProcess>
@@ -171,10 +167,7 @@
           <name>STREET.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
-          <name>FROM_BN.value</name>
-        </jDitoRecordFieldMapping>
-        <jDitoRecordFieldMapping>
-          <name>TO_BN.value</name>
+          <name>BUILDINGNO.value</name>
         </jDitoRecordFieldMapping>
         <jDitoRecordFieldMapping>
           <name>LON.value</name>
@@ -182,9 +175,6 @@
         <jDitoRecordFieldMapping>
           <name>LAT.value</name>
         </jDitoRecordFieldMapping>
-        <jDitoRecordFieldMapping>
-          <name>VALUE.value</name>
-        </jDitoRecordFieldMapping>
       </recordFieldMappings>
     </jDitoRecordContainer>
   </recordContainers>
diff --git a/entity/AddressValidation_entity/entityfields/fulladdressvalidation/children/type_param/valueProcess.js b/entity/AddressValidation_entity/entityfields/fulladdressvalidation/children/type_param/valueProcess.js
index a6cfed5664..daa9b19ded 100644
--- a/entity/AddressValidation_entity/entityfields/fulladdressvalidation/children/type_param/valueProcess.js
+++ b/entity/AddressValidation_entity/entityfields/fulladdressvalidation/children/type_param/valueProcess.js
@@ -1,4 +1,5 @@
 import("WsValidation_lib");
 import("system.result");
 
-result.string(WsValidationType.get().TYPE_ADDRESS.key);
\ No newline at end of file
+// result.string(WsValidationType.get().TYPE_ADDRESS.key);
+result.string(WsValidationType.get().TYPE_ADDRESS_NOMINATIM.key);
\ No newline at end of file
diff --git a/entity/Address_entity/entityfields/fulladdressvalidation/children/currentvalue_param/valueProcess.js b/entity/Address_entity/entityfields/fulladdressvalidation/children/currentvalue_param/valueProcess.js
index b51e1bdca2..3312d2f2f0 100644
--- a/entity/Address_entity/entityfields/fulladdressvalidation/children/currentvalue_param/valueProcess.js
+++ b/entity/Address_entity/entityfields/fulladdressvalidation/children/currentvalue_param/valueProcess.js
@@ -1,4 +1,4 @@
 import("system.result");
 import("system.vars");
 
-    result.string(vars.get("$field.AddressSearch"));
\ No newline at end of file
+result.string(vars.get("$field.AddressSearch"));
\ No newline at end of file
diff --git a/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod b/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod
index 3ee7d10b55..c409a2a6cf 100644
--- a/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod
+++ b/language/_____LANGUAGE_EXTRA/_____LANGUAGE_EXTRA.aod
@@ -4461,6 +4461,9 @@
     <entry>
       <key>Descriptions</key>
     </entry>
+    <entry>
+      <key>User Administration</key>
+    </entry>
   </keyValueMap>
   <font name="Dialog" style="0" size="11" />
   <sqlModels>
diff --git a/preferences/_____PREFERENCES_PROJECT/_____PREFERENCES_PROJECT.aod b/preferences/_____PREFERENCES_PROJECT/_____PREFERENCES_PROJECT.aod
index ee8627ef22..b48b6a4811 100644
--- a/preferences/_____PREFERENCES_PROJECT/_____PREFERENCES_PROJECT.aod
+++ b/preferences/_____PREFERENCES_PROJECT/_____PREFERENCES_PROJECT.aod
@@ -102,11 +102,11 @@
       <property>Country</property>
     </customStringProperty>
     <customBooleanProperty>
-      <name>osm.enable</name>
+      <name>nominatim.enable</name>
       <property v="true" />
     </customBooleanProperty>
     <customStringProperty>
-      <name>osm.url</name>
+      <name>nominatim.url</name>
       <property>https://nominatim.openstreetmap.org/search</property>
     </customStringProperty>
   </customProperties>
diff --git a/process/DataCaching_lib/process.js b/process/DataCaching_lib/process.js
index 5eee5316c7..03cb0d54a4 100644
--- a/process/DataCaching_lib/process.js
+++ b/process/DataCaching_lib/process.js
@@ -11,7 +11,7 @@ import("system.project");
  * @class 
  * 
  * @param {String} pIdentifiyingName name to identify the DataCache. This MUST be unique for one data representation (e.g. key-value pair for all Languages with key ISO2-code and value the ISO3-Code). this will affect the storage-name (=name of the global variable on the client for example)
- * @param {bool} [pKeepPerLanguage=false] if true the data is kept per locale (different storing for each requested language), false when not (every language is sharin the same stoarge because only untranslated data is kept)
+ * @param {bool} [pKeepPerLanguage=false] if true the data is kept per locale (different storing for each requested language), false when not (every language is sharing the same stoarge because only untranslated data is kept)
  * @param {String} [pLocaleOverride=current language] sometimes a special locale is required, use this parameter to specify it
  * 
  */
@@ -37,7 +37,7 @@ function CachedData(pIdentifiyingName, pKeepPerLanguage, pLocaleOverride)
  * @param {String} [pLocaleOverride=current language] sometimes a special locale is required, use this parameter to specify it
  * @param {Function} pDataCallbackFunction function that is called to load the data. this functions gets 2 params: 1. if the data has to be translated 2. the locale to translate
  * 
- * @return {Object} returns the data you wanted (of the pDataCallbackFunction)
+ * @return {CachedData} returns the data you wanted (of the pDataCallbackFunction)
  * 
  * @static
  */
diff --git a/process/WsValidation_lib/process.js b/process/WsValidation_lib/process.js
index e59e0829e2..e8778ed46b 100644
--- a/process/WsValidation_lib/process.js
+++ b/process/WsValidation_lib/process.js
@@ -5,6 +5,8 @@ import("system.vars");
 import("system.neon");
 import("system.net");
 import("system.util");
+import("system.auth");
+import("DataCaching_lib");
 
 /**
  * Class used for the webservice validation types.
@@ -12,18 +14,18 @@ import("system.util");
  * Do not create new instances anywhere else than inside of the get-function!
  * 
  * @param {String} pKey the key-name
- * @param {String} pParamName name used by the webservice
  * @param {String} pWebserviceName name of the webservice. Used to find the correct webservice configuraation
  * @param {Function} pProcessResultCallback function for processing Webservice-result to the return value
  * @param {Function} pProcessRequestCallback function for processing Webservice-request
+ * @param {Object} pAdditionalInfo additional Info. Can be an object or something else. (callback specific)
  */
-function WsValidationType(pKey, pParamName, pWebserviceName, pProcessResultCallback, pProcessRequestCallback) 
+function WsValidationType(pKey, pWebserviceName, pProcessResultCallback, pProcessRequestCallback, pAdditionalInfo) 
 {
     this.key = pKey;
-    this.paramName = pParamName;
     this.processResultCallback = pProcessResultCallback;
     this.processRequestCallback = pProcessRequestCallback;
     this.webserviceName = pWebserviceName;
+    this.additionalInfo = pAdditionalInfo;
 }
 
 /**
@@ -37,12 +39,13 @@ WsValidationType.get = function(pKey)
     if (!this._cache)
         this._cache = {
             // Address lookups
-            TYPE_ADDRESS: new WsValidationType("TYPE_ADDRESS", "search", "addressValidation", _processAllAddressLookup, _customRequest),
-            TYPE_ZIP: new WsValidationType("TYPE_ZIP", "zip", "zipCityValidation", _processAddressLookup, _customRequest),
-            TYPE_CITY: new WsValidationType("TYPE_CITY", "city", "zipCityValidation", _processAddressLookup, _customRequest),
+            TYPE_ADDRESS_NOMINATIM: new WsValidationType("TYPE_ADDRESS_NOMINATIM", "nominatim", _processNominatimAddressLookup, _nominatimRequest, "q"),
+            TYPE_ADDRESS: new WsValidationType("TYPE_ADDRESS", "addressValidation", _processAllAddressLookup, _customRequest, "search"),
+            TYPE_ZIP: new WsValidationType("TYPE_ZIP", "zipCityValidation", _processAddressLookup, _customRequest, "zip"),
+            TYPE_CITY: new WsValidationType("TYPE_CITY", "zipCityValidation", _processAddressLookup, _customRequest, "city"),
             
             // Communication validation
-            TYPE_PHONE: new WsValidationType("TYPE_PHONE", "Number", "phoneValidation", _processCommunicationValidation, _customRequest)
+            TYPE_PHONE: new WsValidationType("TYPE_PHONE", "phoneValidation", _processCommunicationValidation, _customRequest, "Number")
         }
     
     if (pKey)
@@ -61,8 +64,7 @@ WsValidationType.get = function(pKey)
     *    region,
     *    state,
     *    street,
-    *    from_bn, // building number
-    *    to_bn,   // building number
+    *    data.from_bn + (data.to_bn ? " - " + data.to_bn : ""),
     *    lon,
     *    lat,
     *    value // pValue
@@ -71,7 +73,7 @@ WsValidationType.get = function(pKey)
     {
         var resultAddresses = [];
         
-        // if error, only add the default value else parse the body
+        // if error, return [] else parse the body
         if (!WsValidationUtils._isError(pWsResult))
         {
             resultAddresses = JSON.parse(pWsResult.body);
@@ -94,8 +96,7 @@ WsValidationType.get = function(pKey)
                             data.region,
                             data.state,
                             data.streat,
-                            data.from_bn,
-                            data.to_bn,
+                            data.from_bn + (data.to_bn ? " - " + data.to_bn : ""),
                             data.lon,
                             data.lat,
                             pValue]
@@ -113,8 +114,7 @@ WsValidationType.get = function(pKey)
     *    region,
     *    state,
     *    street,
-    *    from_bn, // building number
-    *    to_bn,   // building number
+    *    data.from_bn + (data.to_bn ? " - " + data.to_bn : ""),
     *    lon,
     *    lat,
     *    value // (the result. e.g. if pType was ZIP, the value is the found zip)
@@ -123,7 +123,7 @@ WsValidationType.get = function(pKey)
     {
         var resultAddresses = [];
         
-        // if error, only add the default value else parse the body
+        // if error, return [] else parse the body
         if (!WsValidationUtils._isError(pWsResult))
         {
             resultAddresses = JSON.parse(pWsResult.body);
@@ -144,8 +144,7 @@ WsValidationType.get = function(pKey)
                             data.region,
                             data.state,
                             data.streat,
-                            data.from_bn,
-                            data.to_bn,
+                            data.from_bn + (data.to_bn ? " - " + data.to_bn : ""),
                             data.lon,
                             data.lat,
                             data[this.paramName]]
@@ -164,17 +163,64 @@ WsValidationType.get = function(pKey)
         }
     }
     
+    function _processNominatimAddressLookup(pWsResult, pValue)
+    {
+        var resultAddresses = [];
+        
+        if (pWsResult != null)
+        {
+            logging.log(JSON.stringify(pWsResult, null, "\t"))
+            pWsResult.forEach(function(pPlaceData) {
+                if (pPlaceData.address == undefined)
+                    pPlaceData.address = {};
+                
+                var city = pPlaceData.address.town;
+                if (!city)
+                    city = pPlaceData.address.city;
+                if (!city)
+                    city = pPlaceData.address.village;
+                if (!city)
+                    city = pPlaceData.address.hamlet;
+                
+                var cityext = pPlaceData.address.suburb;
+                if (!cityext)
+                    cityext = pPlaceData.address.city_district;
+                
+                var addrData = [
+                    pPlaceData.display_name,
+                    pPlaceData.address.postcode,
+                    city,
+                    cityext,
+                    pPlaceData.address.country,
+                    pPlaceData.address.district,
+                    pPlaceData.address.state_district,
+                    pPlaceData.address.state,
+                    pPlaceData.address.road,
+                    pPlaceData.address.house_number,
+                    pPlaceData.lon,
+                    pPlaceData.lat
+                ];
+                
+                resultAddresses.push([JSON.stringify(addrData.concat([pValue, pPlaceData.place_id]))].concat(addrData));
+            })
+        }
+
+        return resultAddresses;
+    }
+    
     /**
      * request using a custom webservice
      */
-    function _customRequest(pValue, pCountry) {
+    function _customRequest(pValue, pCountry)
+    {
         if (WsValidationUtils.isWsEnabled(this))
         {
             var userName = project.getPreferenceValue("custom." + this.webserviceName + ".user");
             var pw = project.getPreferenceValue("custom." + this.webserviceName + ".pw");
             var url = project.getPreferenceValue("custom." + this.webserviceName + ".url");
             var countryParamName = project.getPreferenceValue("custom." + this.webserviceName + ".countryParamName");
-
+            var parameterName = this.additionalInfo;
+            
             if (!userName || !pw || !url || !countryParamName)
             {
                 throw new Error("if the webservice " + this.key + " is enabled, you have to provide also userName, pw, url and countryParamName")
@@ -190,7 +236,7 @@ WsValidationType.get = function(pKey)
                 {
                     parameters[countryParamName] = pCountry;
                 }
-                parameters[this.paramName] = pValue;
+                parameters[parameterName] = pValue;
 
                 // call webservice
                 var ret = JSON.parse(net.callRestWebserviceBasicAuth(url, actionType, parameters, null, null, "text/plain", "text/plain", util.DATA_TEXT, util.DATA_TEXT, userName, pw, true));
@@ -208,6 +254,83 @@ WsValidationType.get = function(pKey)
         }
         return this.processResultCallback.call(this, null, pValue);
     }
+    
+    function _nominatimRequest(pValue, pCountry)
+    {
+        if (WsValidationUtils.isWsEnabled(this))
+        {
+            var parameterName = this.additionalInfo;
+            var nominatimCache;
+            
+            if (!vars.exists("$context.nominatim"))
+            {
+                nominatimCache = {
+                    last: ""
+                }
+            } 
+            else 
+            {
+                nominatimCache = JSON.parse(vars.getString("$context.nominatim"));
+            }
+            
+            if (nominatimCache.last == parameterName + pValue + (pCountry ? pCountry : ""))
+            {
+                return nominatimCache.data;
+            }
+            else
+            {
+                nominatimCache.last = parameterName + pValue + (pCountry ? pCountry : "")
+
+                var url = project.getPreferenceValue("custom." + this.webserviceName + ".url");
+
+
+                if (!url)
+                {
+                    throw new Error("if the webservice " + this.key + " is enabled, you have to provide also the url")
+                }
+
+                if (pValue)
+                {
+                    var restConf = net.createConfigForRestWebserviceCall()
+                                    .url(url)
+                                    .actionType("GET")
+                                    .dataTypeAccept("application/json")
+                                    .dataTypeJDitoAccept(util.DATA_TEXT)
+                                    .dataTypeJDitoSend(util.DATA_TEXT)
+                                    .addQueryParameter("format", "json")
+                                    .addQueryParameter("addressdetails", "1");
+
+                    // add user input params
+                    if (pCountry)
+                    {
+                        restConf.addQueryParameter("countrycodes", pCountry)
+                    }
+
+                    restConf.addQueryParameter(parameterName, pValue)
+
+
+                    // call webservice
+                    var ret = JSON.parse(net.callRestWebservice(restConf, auth.createConfigForNoAuth()));
+
+                    // if error, log the error. But also process the result and let the callback generate the correct default value
+                    if (ret == null)
+                    {
+                        logging.log("error")
+                        logging.log(translate.withArguments("${WEBSERVICE_ERROR} url:%0 status:%1", [url, (ret.hasHttpSuccessStatusCode ? " StatusCode: " + ret.httpStatusCode : "")]));
+                    }
+
+                    // different handling of the result per type
+                    nominatimCache.data = this.processResultCallback.call(this, ret, pValue);
+                    vars.set("$context.nominatim", JSON.stringify(nominatimCache))
+                    return nominatimCache.data;
+                }
+            }
+        }
+        
+        nominatimCache.data = this.processResultCallback.call(this, null, pValue);
+        vars.set("$context.nominatim", JSON.stringify(nominatimCache)) 
+        return nominatimCache.data;
+    }
 }
 
 /**
@@ -242,7 +365,7 @@ WsValidationUtils.validate = function(pValue, pType, pCountry)
  */
 WsValidationUtils.setAddressFields = function(pFieldToSetToValue)
 {
-    try 
+    try
     {
         var data = JSON.parse(vars.getString("$this.value"));
     } catch (exception) { 
@@ -251,17 +374,17 @@ WsValidationUtils.setAddressFields = function(pFieldToSetToValue)
         return
     }
      
-    _setField("$field.ZIP", data.zip);
-    _setField("$field.COUNTRY", data.country);
-    _setField("$field.CITY", data.city);
-    //_setField("$field.DISTRICT", data.district); not needed currently
-    _setField("$field.REGION", data.region);
-    _setField("$field.STATE", data.state);
-    _setField("$field.ADDRESS", data.streat);
+    _setField("$field.ZIP", data[1]);
+    _setField("$field.CITY", data[2]);
+    _setField("$field.COUNTRY", data[4]);
+    //_setField("$field.DISTRICT",  data[5]); not needed currently
+    _setField("$field.REGION", data[6]);
+    _setField("$field.STATE", data[7]);
+    _setField("$field.ADDRESS", data[8]);
     
     if (pFieldToSetToValue)
     {
-        _setField(pFieldToSetToValue, data.value)
+        _setField(pFieldToSetToValue, data[12])
     }
     
     function _setField(pField, pValue)
-- 
GitLab