aquameta-datum
Version:
Service layer for the Aquameta database API
187 lines (158 loc) • 18 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = executeConnection;
exports.parseSourceUrl = parseSourceUrl;
var _pg = _interopRequireDefault(require("@micburks/pg"));
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
const anonConfig = {
user: 'anonymous',
database: 'aquameta',
host: 'localhost',
port: 5432,
max: 4,
idleTimeoutMilliseconds: 30000
};
async function getConnection(config) {
const mergedConfig = { ...anonConfig,
...(config && config.connection || {})
}; // Don't allow user to be overridden
mergedConfig.user = anonConfig.user;
const anonClient = new _pg.default.Client(mergedConfig);
await anonClient.connect();
if (!mergedConfig.sessionId) {
return anonClient;
}
let result;
try {
result = await anonClient.query('select (role_id).name as role_name from endpoint.session where id = $1::uuid', [mergedConfig.sessionId]);
} catch (e) {
await anonClient.end();
throw e;
} // If login fails - continue request as anonymous
if (!result || !result.rows || result.rows.length === 0) {
return anonClient;
} // Release Client - TODO: release back to pool
await anonClient.end(); // Logged in
const userResult = new Result(result.rows[0]);
const user = await userResult.json();
console.log(`connection: logged in as ${user.role_name}`);
const userClient = new _pg.default.Client({ ...mergedConfig,
user: user.role_name
});
await userClient.connect();
return userClient;
}
/**
* Execute query server-side
* @returns {Promise}
*/
async function executeConnection(client, query) {
let connection;
try {
connection = await getConnection(client); // let result;
/*
if (query.args && query.args.source) {
const {schemaName, relationName, column, name} = parseSourceUrl(
query.url,
);
const queryResult = await connection.query(
`
select content, mimetype
from (select $1 as content, '$1' as extension from $3.$4 where name='$2') as c
join endpoint.mimetype_extension me on c.extension=me.extension
join endpoint.mimetype m on me.mimetype_id=m.id;
`,
[column, name, schemaName, relationName],
);
result = {...queryResult};
if (result && result.rows && result.rows.length !== 0) {
result.status = 200;
result.message = 'OK';
} else {
result.status = 404;
result.status = 'NOT FOUND';
}
} else {
*/
console.log('trying connection', query.version || client.version, query.method, query.url, JSON.stringify(query.args), JSON.stringify(query.data));
const result = await connection.query('select status, message, response, mimetype ' + 'from endpoint.request($1, $2, $3, $4::json, $5::json)', [query.version || client.version, query.method, query.url, JSON.stringify(query.args), JSON.stringify(query.data)]); //}
// TODO: end connection if userClient, but release if anonClient?
await connection.end();
const res = new Result(result.rows[0]);
if (client.rawResponse) {
return res;
} else {
return res.json().then(r => {
if (r.result) {
return r.result.map(({
row
}) => row);
} else {
return [];
}
});
}
} catch (e) {
// Problem with connecting to database
console.error(`connection: error trying to connect to database`);
console.error(e);
if (connection) {
await connection.end();
}
return null;
}
}
// Mimic response from fetch
class Result {
constructor({
status,
message,
response,
mimetype
}) {
this.status = status;
this.statusText = message;
this.response = response;
this.mimetype = mimetype;
}
async json() {
return JSON.parse(this.response);
}
}
/**
* TODO I want to keep track of how many pools are open and when they connect
* pg-pool has some great events
* pool.on('connect', client => {
* client.count = count++
* })
*/
/**
* TODO in order to do this, I would have to keep track of the open pools,
* instead of doing it with pg.connect()
*
* var pool = new pg.Pool(config)
* pool.connect(callback)
*
* is the same as
*
* pg.connect(config, callback)
*
* and this way, pg will keep track of the pools and not create a new one when
* the same config has been passed in twice
*/
function parseSourceUrl(pathname) {
const [,, schemaName, relationName, ...rest] = pathname.split('/');
const fileName = rest.join('/');
const lastPeriod = fileName.lastIndexOf('.');
const name = fileName.slice(0, lastPeriod);
const column = fileName.slice(lastPeriod + 1);
return {
schemaName,
relationName,
column,
name
};
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["connection.js"],"names":["anonConfig","user","database","host","port","max","idleTimeoutMilliseconds","getConnection","config","mergedConfig","connection","anonClient","pg","Client","connect","sessionId","result","query","e","end","rows","length","userResult","Result","json","console","log","role_name","userClient","executeConnection","client","version","method","url","JSON","stringify","args","data","res","rawResponse","then","r","map","row","error","constructor","status","message","response","mimetype","statusText","parse","parseSourceUrl","pathname","schemaName","relationName","rest","split","fileName","join","lastPeriod","lastIndexOf","name","slice","column"],"mappings":"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","sourcesContent":["// @flow\n\nimport pg from '@micburks/pg';\nimport type {\n  Client,\n  ConnectionOptions,\n  Executable,\n  QueryResult,\n} from '../types.js';\n\nconst anonConfig: ConnectionOptions = {\n  user: 'anonymous',\n  database: 'aquameta',\n  host: 'localhost',\n  port: 5432,\n  max: 4,\n  idleTimeoutMilliseconds: 30000,\n};\n\ntype PgConnection = {\n  connect: () => Promise<void>,\n  query: (string, Array<any>) => Promise<QueryResult>,\n  end: () => Promise<void>,\n};\n\nasync function getConnection(config?: {\n  [string]: any,\n  connection?: {[string]: any},\n}): Promise<PgConnection> {\n  const mergedConfig = {\n    ...anonConfig,\n    ...((config && config.connection) || {}),\n  };\n  // Don't allow user to be overridden\n  mergedConfig.user = anonConfig.user;\n  const anonClient: PgConnection = new pg.Client(mergedConfig);\n\n  await anonClient.connect();\n\n  if (!mergedConfig.sessionId) {\n    return anonClient;\n  }\n\n  let result;\n  try {\n    result = await anonClient.query(\n      'select (role_id).name as role_name from endpoint.session where id = $1::uuid',\n      [mergedConfig.sessionId],\n    );\n  } catch (e) {\n    await anonClient.end();\n    throw e;\n  }\n\n  // If login fails - continue request as anonymous\n  if (!result || !result.rows || result.rows.length === 0) {\n    return anonClient;\n  }\n\n  // Release Client - TODO: release back to pool\n  await anonClient.end();\n\n  // Logged in\n  const userResult = new Result(result.rows[0]);\n  const user = await userResult.json();\n  console.log(`connection: logged in as ${user.role_name}`);\n  const userClient = new pg.Client({\n    ...mergedConfig,\n    user: user.role_name,\n  });\n  await userClient.connect();\n  return userClient;\n}\n\n/**\n * Execute query server-side\n * @returns {Promise}\n */\nexport default async function executeConnection(\n  client: Client,\n  query: Executable,\n): Promise<QueryResult> {\n  let connection;\n\n  try {\n    connection = await getConnection(client);\n    // let result;\n    /*\n    if (query.args && query.args.source) {\n      const {schemaName, relationName, column, name} = parseSourceUrl(\n        query.url,\n      );\n      const queryResult = await connection.query(\n        `\n        select content, mimetype\n        from (select $1 as content, '$1' as extension from $3.$4 where name='$2') as c\n        join endpoint.mimetype_extension me on c.extension=me.extension\n        join endpoint.mimetype m on me.mimetype_id=m.id;\n      `,\n        [column, name, schemaName, relationName],\n      );\n      result = {...queryResult};\n      if (result && result.rows && result.rows.length !== 0) {\n        result.status = 200;\n        result.message = 'OK';\n      } else {\n        result.status = 404;\n        result.status = 'NOT FOUND';\n      }\n    } else {\n      */\n    console.log(\n      'trying connection',\n      query.version || client.version,\n      query.method,\n      query.url,\n      JSON.stringify(query.args),\n      JSON.stringify(query.data),\n    );\n    const result = await connection.query(\n      'select status, message, response, mimetype ' +\n        'from endpoint.request($1, $2, $3, $4::json, $5::json)',\n      [\n        query.version || client.version,\n        query.method,\n        query.url,\n        JSON.stringify(query.args),\n        JSON.stringify(query.data),\n      ],\n    );\n    //}\n\n    // TODO: end connection if userClient, but release if anonClient?\n    await connection.end();\n\n    const res = new Result(result.rows[0]);\n    if (client.rawResponse) {\n      return res;\n    } else {\n      return res.json().then(r => {\n        if (r.result) {\n          return r.result.map(({row}) => row);\n        } else {\n          return [];\n        }\n      });\n    }\n  } catch (e) {\n    // Problem with connecting to database\n    console.error(`connection: error trying to connect to database`);\n    console.error(e);\n    if (connection) {\n      await connection.end();\n    }\n    return null;\n  }\n}\n\ntype ResultArgs = {\n  status: string,\n  message: string,\n  response: string,\n  mimetype: string,\n};\n// Mimic response from fetch\nclass Result {\n  status: string;\n  statusText: string;\n  response: string;\n  mimetype: string;\n  constructor({status, message, response, mimetype}: ResultArgs) {\n    this.status = status;\n    this.statusText = message;\n    this.response = response;\n    this.mimetype = mimetype;\n  }\n  async json(): Promise<{[string]: any}> {\n    return JSON.parse(this.response);\n  }\n}\n\n/**\n * TODO I want to keep track of how many pools are open and when they connect\n * pg-pool has some great events\n * pool.on('connect', client => {\n *   client.count = count++\n * })\n */\n\n/**\n * TODO in order to do this, I would have to keep track of the open pools,\n * instead of doing it with pg.connect()\n *\n * var pool = new pg.Pool(config)\n * pool.connect(callback)\n *\n * is the same as\n *\n * pg.connect(config, callback)\n *\n * and this way, pg will keep track of the pools and not create a new one when\n * the same config has been passed in twice\n */\n\ntype ParsedSourceUrl = {\n  schemaName: string,\n  relationName: string,\n  column: string,\n  name: string,\n};\n\nexport function parseSourceUrl(pathname: string): ParsedSourceUrl {\n  const [, , schemaName, relationName, ...rest] = pathname.split('/');\n  const fileName = rest.join('/');\n  const lastPeriod = fileName.lastIndexOf('.');\n  const name = fileName.slice(0, lastPeriod);\n  const column = fileName.slice(lastPeriod + 1);\n\n  return {schemaName, relationName, column, name};\n}\n"]}