Merge branch 'feature/verbs' into overview
This commit is contained in:
commit
c04a104280
@ -5,7 +5,7 @@
|
|||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"test": "node_modules/mocha/bin/mocha",
|
"test": "node_modules/mocha/bin/mocha",
|
||||||
"start": "node index.js"
|
"start": "nodemon index.js"
|
||||||
},
|
},
|
||||||
"author": "",
|
"author": "",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
|
269
server.js
269
server.js
@ -11,198 +11,98 @@ var pg = require('pg');
|
|||||||
var server = express();
|
var server = express();
|
||||||
server.engine('handlebars', handlebars());
|
server.engine('handlebars', handlebars());
|
||||||
server.set('view engine', 'handlebars');
|
server.set('view engine', 'handlebars');
|
||||||
server.use(function(req, res, next) {
|
|
||||||
res.header("Access-Control-Allow-Origin", "*");
|
server.use(function(inReq, inRes, inNext)
|
||||||
res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
|
{
|
||||||
next();
|
inRes.header("Access-Control-Allow-Origin", "*");
|
||||||
|
inRes.header("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE, OPTIONS");
|
||||||
|
inRes.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
|
||||||
|
inNext();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
var Postgres = new pg.Client({
|
var Postgres = new pg.Client({
|
||||||
user: process.env.user,
|
user: process.env.user,
|
||||||
password: process.env.password,
|
password: process.env.password,
|
||||||
host: process.env.host,
|
host: process.env.host,
|
||||||
port: process.env.port,
|
port: process.env.port,
|
||||||
database: process.env.database,
|
database: process.env.database,
|
||||||
application_name: "tps_etl_api",
|
ssl: false,
|
||||||
ssl: true
|
application_name: "tps_etl_api"
|
||||||
});
|
});
|
||||||
|
Postgres.FirstRow = function(inSQL,args, inResponse)
|
||||||
|
{
|
||||||
|
Postgres.query(inSQL,args, (err, res) => {
|
||||||
|
if (err === null)
|
||||||
|
{
|
||||||
|
inResponse.json(res.rows[0]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
inResponse.json(err.message);
|
||||||
|
});
|
||||||
|
};
|
||||||
Postgres.connect();
|
Postgres.connect();
|
||||||
|
|
||||||
//-------------------------------------------------------------list source--------------------------------------------------------------------------
|
//----------------------------------------------------------source definitions------------------------------------------------------------
|
||||||
|
|
||||||
server.use("/srce_list", function (inReq, inRes) {
|
//returns array of all sources
|
||||||
|
server.get("/source", function (inReq, inRes)
|
||||||
var sql = "SELECT jsonb_agg(defn) source_list FROM tps.srce"
|
{
|
||||||
console.log(sql);
|
var sql = "SELECT jsonb_agg(defn) source_list FROM tps.srce";
|
||||||
|
Postgres.FirstRow(sql,[], inRes);
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
console.log("source list request complete");
|
|
||||||
});
|
});
|
||||||
}
|
//returns message about status and error description
|
||||||
);
|
server.post("/source", bodyParser.json(), function (inReq, inRes)// remove body parsing, just pass post body to the sql string build
|
||||||
|
{
|
||||||
//-------------------------------------------------------------list maps--------------------------------------------------------------------------
|
var sql = "SELECT x.message FROM tps.srce_set($1::jsonb) as x(message)";
|
||||||
|
Postgres.FirstRow(sql,[JSON.stringify(inReq.body)], inRes);
|
||||||
server.use("/map_list", function (inReq, inRes) {
|
|
||||||
|
|
||||||
var sql = "SELECT jsonb_agg(regex) regex FROM tps.map_rm"
|
|
||||||
console.log(sql);
|
|
||||||
|
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
|
|
||||||
if (err === null) {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
inRes.json(err.message);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
//--------------------------------------------------------list unmapped items flagged to be mapped---------------------------------------------------
|
|
||||||
|
|
||||||
server.use("/unmapped", function (inReq, inRes) {
|
|
||||||
|
|
||||||
var sql = "SELECT jsonb_agg(row_to_json(x)::jsonb) regex FROM tps.report_unmapped_recs('";
|
|
||||||
sql += inReq.query.srce + "') x"
|
|
||||||
console.log(sql);
|
|
||||||
|
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
|
|
||||||
if (err === null) {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
inRes.json(err.message);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
//-------------------------------------------------------------set source via json in body--------------------------------------------------------------------------
|
|
||||||
|
|
||||||
server.use("/srce_set", bodyParser.json(), function (inReq, inRes) {
|
|
||||||
|
|
||||||
//validate the body contents before pushing to sql?
|
|
||||||
var sql = "SELECT x.message FROM tps.srce_set($$";
|
|
||||||
sql += JSON.stringify( inReq.body);
|
|
||||||
sql += "$$::jsonb) as x(message)";
|
|
||||||
console.log(sql);
|
|
||||||
|
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
|
|
||||||
//Postgres.end();
|
|
||||||
|
|
||||||
if (err === null) {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
inRes.json(err.message);
|
|
||||||
//handle error
|
|
||||||
});
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
//-------------------------------------------------------------set one or more map definitions--------------------------------------------------------------------------
|
|
||||||
|
|
||||||
server.use("/mapdef_set", bodyParser.json(), function (inReq, inRes) {
|
|
||||||
|
|
||||||
//validate the body contents before pushing to sql?
|
|
||||||
var sql = "SELECT x.message FROM tps.srce_map_def_set($$";
|
|
||||||
sql += JSON.stringify( inReq.body);
|
|
||||||
sql += "$$::jsonb) as x(message)";
|
|
||||||
console.log(sql);
|
|
||||||
|
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
|
|
||||||
//Postgres.end();
|
|
||||||
|
|
||||||
if (err === null) {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
inRes.json(err.message);
|
|
||||||
//handle error
|
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
//----------------------------------------------------------regex instractions--------------------------------------------------------------------------
|
||||||
);
|
//list all regex operations
|
||||||
|
server.get("/regex", function (inReq, inRes)
|
||||||
//-------------------------------------------------------------add entries to lookup table--------------------------------------------------------------------------
|
{
|
||||||
|
var sql = "SELECT jsonb_agg(regex) regex FROM tps.map_rm";
|
||||||
server.use("/mapval_set", bodyParser.json(), function (inReq, inRes) {
|
Postgres.FirstRow(sql, [], inRes);
|
||||||
|
|
||||||
//validate the body contents before pushing to sql?
|
|
||||||
var sql = "SELECT x.message FROM tps.map_rv_set($$";
|
|
||||||
sql += JSON.stringify( inReq.body);
|
|
||||||
sql += "$$::jsonb) as x(message)";
|
|
||||||
console.log(sql);
|
|
||||||
|
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
|
|
||||||
//Postgres.end();
|
|
||||||
|
|
||||||
if (err === null) {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
inRes.json(err.message);
|
|
||||||
//handle error
|
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
//set one or more map definitions
|
||||||
);
|
server.post("/regex", bodyParser.json(), function (inReq, inRes)
|
||||||
|
{
|
||||||
|
var sql = "SELECT x.message FROM tps.srce_map_def_set($1::jsonb) as x(message)";
|
||||||
|
Postgres.FirstRow(sql, [JSON.stringify(inReq.body)], inRes);
|
||||||
|
});
|
||||||
|
|
||||||
|
//------------------------------------------------------------mappings-------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
//list unmapped items flagged to be mapped ?srce=
|
||||||
|
server.get("/unmapped", function (inReq, inRes)
|
||||||
|
{
|
||||||
|
var sql = "SELECT jsonb_agg(row_to_json(x)::jsonb) regex FROM tps.report_unmapped_recs($1::text) x";
|
||||||
|
Postgres.FirstRow(sql,[inReq.query.srce], inRes);
|
||||||
|
});
|
||||||
|
|
||||||
|
//add entries to lookup table
|
||||||
|
server.post("/mapping", bodyParser.json(), function (inReq, inRes)
|
||||||
|
{
|
||||||
|
var sql = "SELECT x.message FROM tps.map_rv_set($1::jsonb) as x(message)";
|
||||||
|
Postgres.FirstRow(sql,[JSON.stringify( inReq.body)], inRes);
|
||||||
|
});
|
||||||
|
|
||||||
/*
|
|
||||||
send a csv with powershell:
|
|
||||||
wget -uri http://localhost/import -Method Post -InFile "C:\Users\fleet\Downloads\d.csv"
|
|
||||||
bash
|
|
||||||
curl -v -F upload=@//mnt/c/Users/fleet/Downloads/d.csv localhost/import
|
|
||||||
*/
|
|
||||||
|
|
||||||
//-------------------------------------------------------------import data--------------------------------------------------------------------------
|
//-------------------------------------------------------------import data--------------------------------------------------------------------------
|
||||||
|
|
||||||
server.use("/import", upload.single('upload'), function (inReq, inRes) {
|
server.use("/import", upload.single('upload'), function (inReq, inRes) {
|
||||||
|
|
||||||
//console.log(inReq.file);
|
|
||||||
console.log("should have gotten file as post body here");
|
console.log("should have gotten file as post body here");
|
||||||
var csv = inReq.file.buffer.toString('utf8')
|
var csv = inReq.file.buffer.toString('utf8')
|
||||||
// create a new converter object
|
|
||||||
//var jobj = csvtojson.fromString(csv).
|
|
||||||
//{headers: "true", delimiter: ",", output: "jsonObj", flatKeys: "true"}
|
//{headers: "true", delimiter: ",", output: "jsonObj", flatKeys: "true"}
|
||||||
csvtojson({ flatKeys: "true" }).fromString(csv).then(
|
csvtojson({ flatKeys: "true" }).fromString(csv).then(
|
||||||
(x) => {
|
(x) => {
|
||||||
//console.log(x);
|
var sql = "SELECT x.message FROM tps.srce_import($1, $2::jsonb) as x(message)"
|
||||||
//inRes.json(x);
|
console.log(sql);
|
||||||
|
Postgres.FirstRow(sql, [inReq.query.srce, JSON.stringify(x)], inRes);
|
||||||
//push to db
|
|
||||||
var sql = "SELECT x.message FROM tps.srce_import($$";
|
|
||||||
sql += inReq.query.srce;
|
|
||||||
sql += "$$, $$"
|
|
||||||
sql += JSON.stringify(x)
|
|
||||||
sql += "$$::jsonb) as x(message)"
|
|
||||||
console.log("sql for insert here");
|
|
||||||
//console.log(sql);
|
|
||||||
|
|
||||||
Postgres.query(sql, (err, res) => {
|
|
||||||
|
|
||||||
//Postgres.end();
|
|
||||||
|
|
||||||
if (err === null) {
|
|
||||||
inRes.json(res.rows[0]);
|
|
||||||
Postgres.end();
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
inRes.json(err.message);
|
|
||||||
//Postgres.end();
|
|
||||||
//handle error
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
//const jsonArray = csv().fromFile(csvFilePath);
|
|
||||||
//csvtojson({ output: "csv" }).fromString(csv).then((jsonObj) => { console.log(jsonObj) });
|
|
||||||
//validate the body contents before pushing to sql?
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -211,49 +111,56 @@ server.use("/import", upload.single('upload'), function (inReq, inRes) {
|
|||||||
|
|
||||||
server.use("/csv_suggest", upload.single('upload'), function (inReq, inRes) {
|
server.use("/csv_suggest", upload.single('upload'), function (inReq, inRes) {
|
||||||
|
|
||||||
//console.log(inReq.file);
|
|
||||||
console.log("should have gotten file as post body here");
|
console.log("should have gotten file as post body here");
|
||||||
var csv = inReq.file.buffer.toString('utf8')
|
var csv = inReq.file.buffer.toString('utf8')
|
||||||
// create a new converter object
|
|
||||||
//var jobj = csvtojson.fromString(csv).
|
|
||||||
//{headers: "true", delimiter: ",", output: "jsonObj", flatKeys: "true"}
|
//{headers: "true", delimiter: ",", output: "jsonObj", flatKeys: "true"}
|
||||||
csvtojson({ flatKeys: "true" }).fromString(csv).then(
|
csvtojson({ flatKeys: "true" }).fromString(csv).then(
|
||||||
(x) => {
|
(x) => {
|
||||||
//console.log(x);
|
var sug = {
|
||||||
//inRes.json(x);
|
schemas: {
|
||||||
|
default: []
|
||||||
//push to db
|
},
|
||||||
var sug = {};
|
loading_function: "csv",
|
||||||
|
source:"client_file",
|
||||||
|
name: "",
|
||||||
|
constraint: []
|
||||||
|
};
|
||||||
for (var key in x[0]) {
|
for (var key in x[0]) {
|
||||||
|
var col = {};
|
||||||
|
//test if number
|
||||||
if (!isNaN(parseFloat(x[0][key])) && isFinite(x[0][key])) {
|
if (!isNaN(parseFloat(x[0][key])) && isFinite(x[0][key])) {
|
||||||
|
//if is a number but leading character is -0- then it's text
|
||||||
if (x[0][key].charAt(0) == "0"){
|
if (x[0][key].charAt(0) == "0"){
|
||||||
sug[key] = "text";
|
col["type"] = "text";
|
||||||
}
|
}
|
||||||
|
//if number and leadign character is not 0 then numeric
|
||||||
else {
|
else {
|
||||||
sug[key] = "numeric";
|
col["type"] = "numeric";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
//if can cast to a date within a hundred years its probably a date
|
||||||
else if (Date.parse(x[0][key]) > Date.parse('1950-01-01') && Date.parse(x[0][key]) < Date.parse('2050-01-01')) {
|
else if (Date.parse(x[0][key]) > Date.parse('1950-01-01') && Date.parse(x[0][key]) < Date.parse('2050-01-01')) {
|
||||||
sug[key] = "date";
|
col["type"] = "date";
|
||||||
}
|
}
|
||||||
|
//otherwise its text
|
||||||
else {
|
else {
|
||||||
sug[key] = "text";
|
col["type"] = "text";
|
||||||
}
|
}
|
||||||
|
col["path"] = "{" + key + "}";
|
||||||
|
col["column_name"] = key;
|
||||||
|
sug.schemas.default.push(col);
|
||||||
}
|
}
|
||||||
console.log(sug);
|
console.log(sug);
|
||||||
inRes.json(sug);
|
inRes.json(sug);
|
||||||
//console.log(sql);
|
|
||||||
}
|
}
|
||||||
//const jsonArray = csv().fromFile(csvFilePath);
|
|
||||||
//csvtojson({ output: "csv" }).fromString(csv).then((jsonObj) => { console.log(jsonObj) });
|
|
||||||
//validate the body contents before pushing to sql?
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
server.get("/", function (inReq, inRes) {
|
server.get("/", function (inReq, inRes)
|
||||||
|
{
|
||||||
inRes.render("definition", { title: "definition", layout: "main" });
|
inRes.render("definition", { title: "definition", layout: "main" });
|
||||||
})
|
});
|
||||||
|
|
||||||
module.exports = server;
|
module.exports = server;
|
1
test/dcard_regex/curl
Normal file
1
test/dcard_regex/curl
Normal file
@ -0,0 +1 @@
|
|||||||
|
curl -H "Content-Type: application/json" -X POST -d@./regex.json http://localhost/regex
|
24
test/dcard_regex/regex.json
Normal file
24
test/dcard_regex/regex.json
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"regex": {
|
||||||
|
"function": "extract",
|
||||||
|
"description": "pull first 20 characters from description for mapping",
|
||||||
|
"where": [
|
||||||
|
{}
|
||||||
|
],
|
||||||
|
"defn": [
|
||||||
|
{
|
||||||
|
"regex": ".{1,20}",
|
||||||
|
"map": "y",
|
||||||
|
"field": "f20",
|
||||||
|
"flag": "",
|
||||||
|
"key": "{Description}",
|
||||||
|
"retain": "y"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"sequence": 2,
|
||||||
|
"name": "First 20",
|
||||||
|
"srce": "dcard"
|
||||||
|
}
|
||||||
|
]
|
1
test/dcard_source/curl
Normal file
1
test/dcard_source/curl
Normal file
@ -0,0 +1 @@
|
|||||||
|
curl -H "Content-Type: application/json" -X POST -d@./srce.json http://localhost/srce_set
|
76
test/dcard_source/srce.json
Normal file
76
test/dcard_source/srce.json
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
{
|
||||||
|
"name": "dcard",
|
||||||
|
"source": "client_file",
|
||||||
|
"loading_function": "csv",
|
||||||
|
"constraint": [
|
||||||
|
"{Trans. Date}",
|
||||||
|
"{Post Date}",
|
||||||
|
"{Description}"
|
||||||
|
],
|
||||||
|
"schemas": {
|
||||||
|
"default": [
|
||||||
|
{
|
||||||
|
"path": "{Trans. Date}",
|
||||||
|
"type": "date",
|
||||||
|
"column_name": "Trans. Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Post Date}",
|
||||||
|
"type": "date",
|
||||||
|
"column_name": "Post Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Description}",
|
||||||
|
"type": "text",
|
||||||
|
"column_name": "Description"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Amount}",
|
||||||
|
"type": "numeric",
|
||||||
|
"column_name": "Amount"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Category}",
|
||||||
|
"type": "text",
|
||||||
|
"column_name": "Category"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"mapped": [
|
||||||
|
{
|
||||||
|
"path": "{Trans. Date}",
|
||||||
|
"type": "date",
|
||||||
|
"column_name": "Trans. Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Post Date}",
|
||||||
|
"type": "date",
|
||||||
|
"column_name": "Post Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Description}",
|
||||||
|
"type": "text",
|
||||||
|
"column_name": "Description"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Amount}",
|
||||||
|
"type": "numeric",
|
||||||
|
"column_name": "Amount"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{Category}",
|
||||||
|
"type": "text",
|
||||||
|
"column_name": "Category"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{party}",
|
||||||
|
"type": "text",
|
||||||
|
"column_name": "Party"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "{reason}",
|
||||||
|
"type": "text",
|
||||||
|
"column_name": "Reason"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user