-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathroutes.ts
118 lines (105 loc) · 2.41 KB
/
routes.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
import * as actual from './actual';
import { q, runQuery } from '@actual-app/api';
import express from 'express';
const router = express.Router();
const transactionsQuery = q('transactions').select([
'id',
'is_parent',
'is_child',
'parent_id',
'account',
'category',
'amount',
'payee',
'notes',
'date',
'imported_id',
'error',
'imported_payee',
'starting_balance_flag',
'transfer_id',
'sort_order',
'cleared',
'reconciled',
'tombstone',
'schedule',
'account.closed',
'account.name',
'account.offbudget',
'category.name',
'category.is_income',
'payee.name',
'schedule.name',
]);
const formatHandler: express.RequestHandler = (req, res) => {
const format = req.query['format'] === 'csv' ? 'csv' : 'json';
if (format === 'csv') {
const data = jsonToCSV(res.locals.data);
res.status(200).header('Content-Type', 'text/csv').send(data);
} else if (format === 'json') {
res.status(200).json(res.locals.data);
}
};
router.get('/budget/:month', async (req, res) => {
const budget = await actual.getBudgetAtMonth(req.params.month);
res.status(200).json(budget);
});
router.get(
'/accounts',
async (_, res, next) => {
const accounts = await actual.getAccounts();
res.locals.data = accounts;
next();
},
formatHandler,
);
router.get(
'/accounts/:accountid',
async (req, res, next) => {
const account = await actual.getAccount(req.params.accountid);
res.locals.data = account;
next();
},
formatHandler,
);
router.get(
'/accounts/:accountid/transactions',
async (req, res, next) => {
const { data } = await runQuery(
transactionsQuery.filter({ account: req.params.accountid }),
);
res.locals.data = data;
next();
},
formatHandler,
);
router.get(
'/transactions',
async (_, res, next) => {
const { data } = await runQuery(transactionsQuery);
res.locals.data = data;
next();
},
formatHandler,
);
export { router };
function jsonToCSV(data: any) {
let csv = '';
const unpackHeaders = (x: object) => Object.keys(x).join(',') + '\n';
const unpackValues = (x: object) =>
Object.values(x)
.map((v) => JSON.stringify(v))
.join(',') + '\n';
// object
if (!Array.isArray(data)) {
csv += unpackHeaders(data);
csv += unpackValues(data);
return csv;
}
// array
csv += unpackHeaders(data?.[0]);
for (const row of data) {
csv += unpackValues(row);
}
return csv;
}