UNPKG

aquameta-datum

Version:

Service layer for the Aquameta database API

124 lines (102 loc) 13 kB
import { fetch } from './fetch.js'; /** * Fetch query results client-side * @returns {Promise} */ export default async function executeEndpoint(client, query) { const requestUrl = `/${client.url}/${client.version}/${query.url}`.replace(/\/+/g, '/') // remove duplicate slashes .replace(/\/$/, ''); // remove tail slash const queryString = getQueryString(query.args); const options = { method: query.method, credentials: 'same-origin', headers: { 'Content-Type': 'application/json' } }; if (query.data) { options.body = JSON.stringify(query.data); } try { console.log(`endpoint: trying to fetch ${requestUrl}`, options); const response = await fetch(queryString ? `${requestUrl}?${queryString}` : requestUrl, options); // Let client deal with status codes if (client.rawResponse) { return response; } else { return response.json().then(r => { return r.result.map(({ row }) => row); }); } } catch (error) { // Log error in collapsed group console.groupCollapsed(query.method, error.statusCode, error.title); if ('message' in error) { console.error(error.message); } console.groupEnd(); throw error.title; } } // Map the keys of the args object to an array of encoded url components function getQueryString(args) { return Object.keys(args).sort().map(argName => argName in methods && methods[argName](args[argName])).filter(Boolean).join('&').replace(/&&/g, '&'); } const methods = { where(arg) { // where: [{ name: 'column_name', op: '=', value: 'value' }] return asArray(arg).map(w => `where=${urlEncode(w)}`).join('&'); }, order(arg) { // order_by: [{ column: 'column_name', direction: 'asc|desc' }] // TODO: random()? const columnList = asArray(arg).map(col => { const { column, direction = 'asc' } = col; return direction !== 'asc' ? `-${column}` : `${column}`; }); return `order_by=${encodeURIComponent(columnList.join(','))}`; }, limit(arg) { // limit: number const parsedNum = parseInt(arg); if (!isNaN(parsedNum)) { return `limit=${parsedNum}`; } else { return null; } }, offset(arg) { // offset: number const parsedNum = parseInt(arg); if (!isNaN(parsedNum)) { return `order=${parsedNum}`; } else { return null; } }, evented() { return 'session_id'; }, metaData(arg) { return `meta_data=${urlEncode(arg)}`; }, args(arg) { return `args=${urlEncode(arg)}`; }, exclude(arg) { return `exclude=${urlEncode(arg)}`; }, include(arg) { return `include=${urlEncode(arg)}`; } }; function urlEncode(str) { return encodeURIComponent(JSON.stringify(str)); } function asArray(arg) { return arg instanceof Array ? arg : [arg]; } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["endpoint.js"],"names":["fetch","executeEndpoint","client","query","requestUrl","url","version","replace","queryString","getQueryString","args","options","method","credentials","headers","data","body","JSON","stringify","console","log","response","rawResponse","json","then","r","result","map","row","error","groupCollapsed","statusCode","title","message","groupEnd","Object","keys","sort","argName","methods","filter","Boolean","join","where","arg","asArray","w","urlEncode","order","columnList","col","column","direction","encodeURIComponent","limit","parsedNum","parseInt","isNaN","offset","evented","metaData","exclude","include","str","Array"],"mappings":"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","sourcesContent":["// @flow\n\nimport {fetch} from './fetch.js';\nimport type {Client, Executable, QueryResult, WhereOps} from '../types.js';\n\ntype Request = {\n  method: string,\n  credentials?: string,\n  headers: {[string]: string},\n  body?: string,\n};\n\n/**\n * Fetch query results client-side\n * @returns {Promise}\n */\nexport default async function executeEndpoint(\n  client: Client,\n  query: Executable,\n): Promise<QueryResult> {\n  const requestUrl = `/${client.url}/${client.version}/${query.url}`\n    .replace(/\\/+/g, '/') // remove duplicate slashes\n    .replace(/\\/$/, ''); // remove tail slash\n  const queryString = getQueryString(query.args);\n  const options: Request = {\n    method: query.method,\n    credentials: 'same-origin',\n    headers: {\n      'Content-Type': 'application/json',\n    },\n  };\n  if (query.data) {\n    options.body = JSON.stringify(query.data);\n  }\n\n  try {\n    console.log(`endpoint: trying to fetch ${requestUrl}`, options);\n    const response = await fetch(\n      queryString ? `${requestUrl}?${queryString}` : requestUrl,\n      options,\n    );\n\n    // Let client deal with status codes\n    if (client.rawResponse) {\n      return response;\n    } else {\n      return response.json().then(r => {\n        return r.result.map(({row}) => row);\n      });\n    }\n  } catch (error) {\n    // Log error in collapsed group\n    console.groupCollapsed(query.method, error.statusCode, error.title);\n    if ('message' in error) {\n      console.error(error.message);\n    }\n    console.groupEnd();\n    throw error.title;\n  }\n}\n\n// Map the keys of the args object to an array of encoded url components\nfunction getQueryString(args: {[string]: mixed}): string {\n  return Object.keys(args)\n    .sort()\n    .map(argName => argName in methods && methods[argName](args[argName]))\n    .filter(Boolean)\n    .join('&')\n    .replace(/&&/g, '&');\n}\n\ntype WhereArg = {\n  name: string,\n  op: WhereOps,\n  value: any,\n};\ntype OrderArg = {\n  column: string,\n  direction?: string,\n};\nconst methods = {\n  where(arg: WhereArg | Array<WhereArg>): string {\n    // where: [{ name: 'column_name', op: '=', value: 'value' }]\n    return asArray(arg)\n      .map(w => `where=${urlEncode(w)}`)\n      .join('&');\n  },\n  order(arg: OrderArg | Array<OrderArg>): string {\n    // order_by: [{ column: 'column_name', direction: 'asc|desc' }]\n    // TODO: random()?\n    const columnList = asArray(arg).map(col => {\n      const {column, direction = 'asc'} = col;\n      return direction !== 'asc' ? `-${column}` : `${column}`;\n    });\n    return `order_by=${encodeURIComponent(columnList.join(','))}`;\n  },\n  limit(arg: number): string | null {\n    // limit: number\n    const parsedNum = parseInt(arg);\n    if (!isNaN(parsedNum)) {\n      return `limit=${parsedNum}`;\n    } else {\n      return null;\n    }\n  },\n  offset(arg: number): string | null {\n    // offset: number\n    const parsedNum = parseInt(arg);\n    if (!isNaN(parsedNum)) {\n      return `order=${parsedNum}`;\n    } else {\n      return null;\n    }\n  },\n  evented(): string {\n    return 'session_id';\n  },\n  metaData(arg: any): string {\n    return `meta_data=${urlEncode(arg)}`;\n  },\n  args(arg: any): string {\n    return `args=${urlEncode(arg)}`;\n  },\n  exclude(arg: any): string {\n    return `exclude=${urlEncode(arg)}`;\n  },\n  include(arg: any): string {\n    return `include=${urlEncode(arg)}`;\n  },\n};\n\nfunction urlEncode(str: any): string {\n  return encodeURIComponent(JSON.stringify(str));\n}\n\nfunction asArray(arg: any): any {\n  return arg instanceof Array ? arg : [arg];\n}\n"]}