X-Git-Url: https://git.openstreetmap.org./nominatim-ui.git/blobdiff_plain/c636646759d9c541f7baf19812be18d002a8ce1d..d092630e94af1e5c38c8cb22335b939f6759ab7d:/src/lib/api_utils.js?ds=sidebyside diff --git a/src/lib/api_utils.js b/src/lib/api_utils.js index 066543f..3cbc0df 100644 --- a/src/lib/api_utils.js +++ b/src/lib/api_utils.js @@ -1,77 +1,117 @@ +import { last_api_request_url_store, error_store } from './stores.js'; -import { get_config_value } from './config_reader.js'; -import { last_updated_store } from './stores.js'; - +function api_request_progress(status) { + var loading_el = document.getElementById('loading'); + if (!loading_el) return; // might not be on page yet + loading_el.style.display = (status === 'start') ? 'block' : null; +} export async function fetch_from_api(endpoint_name, params, callback) { var api_url = generate_nominatim_api_url(endpoint_name, params); - document.getElementById('loading').style.display = 'block'; - await fetch(api_url) - .then(response => response.json()) - .then(data => { - callback(data); - document.getElementById('loading').style.display = 'none'; - }); - - - fetch(generate_nominatim_api_url('status', { format: 'json' })) - .then(response => response.json()) - .then(data => { - let last_updated = { - api_request_url: api_url, - api_request_url_debug: api_url + '&debug=1', - date: data.data_updated - }; - last_updated_store.set(last_updated); - }); + const mock_api_error = (new URLSearchParams(window.location.search)).get('mock_api_error'); + + api_request_progress('start'); + if (endpoint_name !== 'status') last_api_request_url_store.set(null); + + try { + await fetch(api_url, { headers: Nominatim_Config.Nominatim_API_Endpoint_Headers || {} }) + .then(async (response) => { + if ((!((response.status >= 200 && response.status < 300) || response.status === 404)) + || mock_api_error === 'fetch' + ) { + error_store.set(`Error fetching data from ${api_url} (${response.statusText})`); + return undefined; + } + + // Parse JSON here instead of returning a promise so we can catch possible + // errors. + var data; + try { + if (mock_api_error === 'parse') { + data = JSON.parse('{'); + } else { + data = await response.json(); + } + } catch (err) { + // e.g. 'JSON.parse: unexpected non-whitespace character after JSON data at line 1' + error_store.set(`Error parsing JSON data from ${api_url} (${err})`); + return undefined; + } + return data; + }) + .then((data) => { + if (data) { + if (data.error) { + error_store.set(data.error.message); + } + callback(data); + } + api_request_progress('finish'); + }); + } catch (error) { + error_store.set(`Error fetching data from ${api_url} (${error})`); + api_request_progress('finish'); + } + + if (endpoint_name !== 'status') last_api_request_url_store.set(api_url); +} + +var fetch_content_cache = {}; +export async function fetch_content_into_element(url, dom_element) { + if (!window.location.protocol.match(/^http/)) { + dom_element.innerHTML = `Cannot display data from ${url} here. ` + + 'Browser security prevents loading content from file:// URLs.'; + return; + } + + if (fetch_content_cache[url]) { + dom_element.innerHTML = fetch_content_cache[url]; + return; + } + try { + await fetch(url) + .then(response => response.text()) + .then(html => { + html = html.replace('Nominatim_API_Endpoint', generate_nominatim_endpoint_url()); + dom_element.innerHTML = html; + fetch_content_cache[url] = html; + }); + } catch (error) { + dom_element.innerHTML = `Error fetching content from ${url} (${error})`; + } +} + +function generate_nominatim_endpoint_url(endpoint_name) { + var conf_endpoint = Nominatim_Config.Nominatim_API_Endpoint; + + if (typeof conf_endpoint === 'function') { + return conf_endpoint(endpoint_name); + } + + if (!endpoint_name) return conf_endpoint; + + return conf_endpoint + endpoint_name + '.php'; } function generate_nominatim_api_url(endpoint_name, params) { - return get_config_value('Nominatim_API_Endpoint') + endpoint_name + '.php?' + // default value for /search + if (params.dedupe === 1) delete params.dedupe; + + extend_parameters(params, Nominatim_Config.Nominatim_API_Endpoint_Params); + return generate_nominatim_endpoint_url(endpoint_name) + + '?' + Object.keys(clean_up_parameters(params)).map((k) => { return encodeURIComponent(k) + '=' + encodeURIComponent(params[k]); }).join('&'); } -/*! - * Serialize all form data into a SearchParams string - * (c) 2020 Chris Ferdinandi, MIT License, https://gomakethings.com - * @param {Node} form The form to serialize - * @return {String} The serialized form data - */ -export function serialize_form(form) { - var arr = []; - Array.prototype.slice.call(form.elements).forEach(function (field) { - if (!field.name || field.disabled || ['submit', 'button'].indexOf(field.type) > -1) return; - // if (field.type === 'select-multiple') { - // Array.prototype.slice.call(field.options).forEach(function (option) { - // if (!option.selected) return; - // arr.push(encodeURIComponent(field.name) + '=' + encodeURIComponent(option.value)); - // }); - // return; - // } - if (['checkbox', 'radio'].indexOf(field.type) > -1 && !field.checked) return; - if (typeof field.value === 'undefined') return; - arr.push(encodeURIComponent(field.name) + '=' + encodeURIComponent(field.value)); - }); - return arr.join('&'); -} - - -// remove any URL paramters with empty values -// '&empty=&filled=value' => 'filled=value' -export function clean_up_url_parameters(url) { - var url_params = new URLSearchParams(url); - var to_delete = []; // deleting inside loop would skip iterations - url_params.forEach(function (value, key) { - if (value === '') to_delete.push(key); - }); - for (var i = 0; i < to_delete.length; i += 1) { - url_params.delete(to_delete[i]); +function extend_parameters(params, params2) { + var param_names = Object.keys(params2); + for (var i = 0; i < param_names.length; i += 1) { + params[param_names[i]] = params2[param_names[i]]; } - return url_params.toString(); } function clean_up_parameters(params) { @@ -87,7 +127,7 @@ function clean_up_parameters(params) { } export function update_html_title(title) { - document.title = [title, 'OpenStreetMap Nominatim'] + document.title = [title, Nominatim_Config.Page_Title] .filter((val) => val && val.length > 1) .join(' | '); }