@invoicer/cli 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (50) hide show
  1. package/.env.example +15 -0
  2. package/config.json +19 -0
  3. package/data/last.json +11 -0
  4. package/dist/cli.js +38 -0
  5. package/dist/cli.js.map +1 -0
  6. package/dist/commands/clients.js +184 -0
  7. package/dist/commands/clients.js.map +1 -0
  8. package/dist/commands/generate.js +294 -0
  9. package/dist/commands/generate.js.map +1 -0
  10. package/dist/commands/timesheet.js +34 -0
  11. package/dist/commands/timesheet.js.map +1 -0
  12. package/dist/core/json-store.js +27 -0
  13. package/dist/core/json-store.js.map +1 -0
  14. package/dist/core/paths.js +29 -0
  15. package/dist/core/paths.js.map +1 -0
  16. package/dist/core/validators.js +131 -0
  17. package/dist/core/validators.js.map +1 -0
  18. package/dist/domain/types.js +2 -0
  19. package/dist/domain/types.js.map +1 -0
  20. package/dist/services/email.js +32 -0
  21. package/dist/services/email.js.map +1 -0
  22. package/dist/services/invoice-number.js +33 -0
  23. package/dist/services/invoice-number.js.map +1 -0
  24. package/dist/services/pdf-render.js +69 -0
  25. package/dist/services/pdf-render.js.map +1 -0
  26. package/dist/services/timesheet.js +99 -0
  27. package/dist/services/timesheet.js.map +1 -0
  28. package/dist/utils/sanitize.js +8 -0
  29. package/dist/utils/sanitize.js.map +1 -0
  30. package/index.html +2584 -0
  31. package/logo.svg +17 -0
  32. package/package.json +39 -0
  33. package/scripts/invoice.mjs +23 -0
  34. package/src/cli.ts +44 -0
  35. package/src/commands/clients.ts +221 -0
  36. package/src/commands/generate.ts +379 -0
  37. package/src/commands/timesheet.ts +47 -0
  38. package/src/core/json-store.ts +33 -0
  39. package/src/core/paths.ts +42 -0
  40. package/src/core/validators.ts +168 -0
  41. package/src/domain/types.ts +129 -0
  42. package/src/services/email.ts +40 -0
  43. package/src/services/invoice-number.ts +46 -0
  44. package/src/services/pdf-render.ts +95 -0
  45. package/src/services/timesheet.ts +173 -0
  46. package/src/utils/sanitize.ts +8 -0
  47. package/test/cli-wiring.test.ts +25 -0
  48. package/test/invoice-number.test.ts +45 -0
  49. package/test/timesheet.test.ts +104 -0
  50. package/tsconfig.json +17 -0
@@ -0,0 +1,27 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ export function readJsonFile(filePath) {
4
+ if (!fs.existsSync(filePath)) {
5
+ return undefined;
6
+ }
7
+ const raw = fs.readFileSync(filePath, "utf8");
8
+ try {
9
+ return JSON.parse(raw);
10
+ }
11
+ catch (error) {
12
+ const message = error instanceof Error ? error.message : String(error);
13
+ throw new Error(`Invalid JSON in ${filePath}: ${message}`);
14
+ }
15
+ }
16
+ export function loadJsonFile(filePath, fallback, validate) {
17
+ const parsed = readJsonFile(filePath);
18
+ if (parsed === undefined) {
19
+ return fallback;
20
+ }
21
+ return validate(parsed);
22
+ }
23
+ export function writeJsonFile(filePath, value) {
24
+ fs.mkdirSync(path.dirname(filePath), { recursive: true });
25
+ fs.writeFileSync(filePath, JSON.stringify(value, null, 2));
26
+ }
27
+ //# sourceMappingURL=json-store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"json-store.js","sourceRoot":"","sources":["../../src/core/json-store.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAE7B,MAAM,UAAU,YAAY,CAAC,QAAgB;IAC3C,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC7B,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC9C,IAAI,CAAC;QACH,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,MAAM,OAAO,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACvE,MAAM,IAAI,KAAK,CAAC,mBAAmB,QAAQ,KAAK,OAAO,EAAE,CAAC,CAAC;IAC7D,CAAC;AACH,CAAC;AAED,MAAM,UAAU,YAAY,CAC1B,QAAgB,EAChB,QAAW,EACX,QAA+B;IAE/B,MAAM,MAAM,GAAG,YAAY,CAAC,QAAQ,CAAC,CAAC;IACtC,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;QACzB,OAAO,QAAQ,CAAC;IAClB,CAAC;IACD,OAAO,QAAQ,CAAC,MAAM,CAAC,CAAC;AAC1B,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,QAAgB,EAAE,KAAc;IAC5D,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAC1D,EAAE,CAAC,aAAa,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;AAC7D,CAAC"}
@@ -0,0 +1,29 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { fileURLToPath } from "node:url";
4
+ export function findProjectRoot(startDir = process.cwd()) {
5
+ let currentDir = startDir;
6
+ const root = path.parse(currentDir).root;
7
+ while (currentDir !== root) {
8
+ const configPath = path.join(currentDir, "config.json");
9
+ if (fs.existsSync(configPath)) {
10
+ return currentDir;
11
+ }
12
+ currentDir = path.dirname(currentDir);
13
+ }
14
+ return null;
15
+ }
16
+ export function createProjectPaths(startDir = process.cwd()) {
17
+ const moduleRoot = path.resolve(path.dirname(fileURLToPath(import.meta.url)), "..", "..");
18
+ const projectRoot = findProjectRoot(startDir) ?? (fs.existsSync(path.join(moduleRoot, "config.json")) ? moduleRoot : process.cwd());
19
+ return {
20
+ projectRoot,
21
+ configPath: path.join(projectRoot, "config.json"),
22
+ lastPath: path.join(projectRoot, "data", "last.json"),
23
+ timesheetPath: path.join(projectRoot, "data", "timesheet.json"),
24
+ archivePath: path.join(projectRoot, "data", "archive.json"),
25
+ outDir: path.join(projectRoot, "out"),
26
+ indexHtmlPath: path.join(projectRoot, "index.html"),
27
+ };
28
+ }
29
+ //# sourceMappingURL=paths.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"paths.js","sourceRoot":"","sources":["../../src/core/paths.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AAYzC,MAAM,UAAU,eAAe,CAAC,QAAQ,GAAG,OAAO,CAAC,GAAG,EAAE;IACtD,IAAI,UAAU,GAAG,QAAQ,CAAC;IAC1B,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC;IAEzC,OAAO,UAAU,KAAK,IAAI,EAAE,CAAC;QAC3B,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC;QACxD,IAAI,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9B,OAAO,UAAU,CAAC;QACpB,CAAC;QACD,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IACxC,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,UAAU,kBAAkB,CAAC,QAAQ,GAAG,OAAO,CAAC,GAAG,EAAE;IACzD,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;IAC1F,MAAM,WAAW,GAAG,eAAe,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACpI,OAAO;QACL,WAAW;QACX,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC;QACjD,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE,WAAW,CAAC;QACrD,aAAa,EAAE,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE,gBAAgB,CAAC;QAC/D,WAAW,EAAE,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE,cAAc,CAAC;QAC3D,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,CAAC;QACrC,aAAa,EAAE,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,YAAY,CAAC;KACpD,CAAC;AACJ,CAAC"}
@@ -0,0 +1,131 @@
1
+ function ensureObject(value, label) {
2
+ if (typeof value !== "object" || value === null || Array.isArray(value)) {
3
+ throw new Error(`${label} must be an object`);
4
+ }
5
+ return value;
6
+ }
7
+ function ensureString(value, label, fallback = "") {
8
+ if (value === undefined || value === null) {
9
+ return fallback;
10
+ }
11
+ if (typeof value !== "string") {
12
+ throw new Error(`${label} must be a string`);
13
+ }
14
+ return value;
15
+ }
16
+ function ensureNumber(value, label) {
17
+ if (typeof value !== "number" || Number.isNaN(value)) {
18
+ throw new Error(`${label} must be a number`);
19
+ }
20
+ return value;
21
+ }
22
+ function ensureOptionalNumber(value, label, fallback) {
23
+ if (value === undefined || value === null || value === "") {
24
+ return fallback;
25
+ }
26
+ if (typeof value === "string") {
27
+ const parsed = Number(value);
28
+ if (!Number.isFinite(parsed)) {
29
+ throw new Error(`${label} must be numeric`);
30
+ }
31
+ return parsed;
32
+ }
33
+ return ensureNumber(value, label);
34
+ }
35
+ export function validateClientConfig(value, indexLabel = "client") {
36
+ const client = ensureObject(value, indexLabel);
37
+ return {
38
+ name: ensureString(client.name, `${indexLabel}.name`).trim(),
39
+ email: ensureString(client.email, `${indexLabel}.email`, "").trim(),
40
+ address: ensureString(client.address, `${indexLabel}.address`, "").trim(),
41
+ currency: ensureString(client.currency, `${indexLabel}.currency`, "USD") || "USD",
42
+ defaultRate: ensureOptionalNumber(client.defaultRate, `${indexLabel}.defaultRate`, 0),
43
+ payPeriodType: ensureString(client.payPeriodType, `${indexLabel}.payPeriodType`, "").trim(),
44
+ };
45
+ }
46
+ export function validateConfig(value) {
47
+ const obj = ensureObject(value, "config");
48
+ const clientsRaw = obj.client;
49
+ if (!Array.isArray(clientsRaw)) {
50
+ throw new Error("config.client must be an array");
51
+ }
52
+ const fromRaw = obj.from === undefined ? {} : ensureObject(obj.from, "config.from");
53
+ return {
54
+ client: clientsRaw.map((client, index) => validateClientConfig(client, `config.client[${index}]`)),
55
+ defaultDescription: ensureString(obj.defaultDescription, "config.defaultDescription", ""),
56
+ from: {
57
+ name: ensureString(fromRaw.name, "config.from.name", ""),
58
+ email: ensureString(fromRaw.email, "config.from.email", ""),
59
+ address: ensureString(fromRaw.address, "config.from.address", ""),
60
+ },
61
+ invoicePrefix: ensureString(obj.invoicePrefix, "config.invoicePrefix", "INV"),
62
+ note: ensureString(obj.note, "config.note", ""),
63
+ };
64
+ }
65
+ function validateClientHistory(value, label) {
66
+ const obj = ensureObject(value, label);
67
+ const seqByMonthRaw = obj.seqByMonth === undefined ? {} : ensureObject(obj.seqByMonth, `${label}.seqByMonth`);
68
+ const seqByMonth = Object.entries(seqByMonthRaw).reduce((acc, [month, seq]) => {
69
+ acc[month] = ensureOptionalNumber(seq, `${label}.seqByMonth.${month}`, 0);
70
+ return acc;
71
+ }, {});
72
+ return {
73
+ lastHours: ensureOptionalNumber(obj.lastHours, `${label}.lastHours`, 160),
74
+ lastRate: ensureOptionalNumber(obj.lastRate, `${label}.lastRate`, 0),
75
+ seqByMonth,
76
+ };
77
+ }
78
+ export function validateLastState(value) {
79
+ const obj = ensureObject(value, "last");
80
+ const clientsRaw = obj.clients === undefined ? {} : ensureObject(obj.clients, "last.clients");
81
+ const clients = Object.entries(clientsRaw).reduce((acc, [name, history]) => {
82
+ acc[name] = validateClientHistory(history, `last.clients.${name}`);
83
+ return acc;
84
+ }, {});
85
+ return { clients };
86
+ }
87
+ function validateTimesheetEntry(value, label) {
88
+ const obj = ensureObject(value, label);
89
+ return {
90
+ id: ensureOptionalNumber(obj.id, `${label}.id`, Date.now()),
91
+ client: ensureString(obj.client, `${label}.client`),
92
+ date: ensureString(obj.date, `${label}.date`),
93
+ hours: ensureOptionalNumber(obj.hours, `${label}.hours`, 0),
94
+ description: ensureString(obj.description, `${label}.description`, ""),
95
+ rate: ensureOptionalNumber(obj.rate, `${label}.rate`, 0),
96
+ amount: ensureOptionalNumber(obj.amount, `${label}.amount`, 0),
97
+ createdAt: ensureString(obj.createdAt, `${label}.createdAt`, ""),
98
+ };
99
+ }
100
+ export function validateTimesheetData(value) {
101
+ const obj = ensureObject(value, "timesheet");
102
+ const entriesRaw = obj.entries === undefined ? [] : obj.entries;
103
+ if (!Array.isArray(entriesRaw)) {
104
+ throw new Error("timesheet.entries must be an array");
105
+ }
106
+ return {
107
+ entries: entriesRaw.map((entry, index) => validateTimesheetEntry(entry, `timesheet.entries[${index}]`)),
108
+ };
109
+ }
110
+ function validateArchivedEntry(value, label) {
111
+ const entry = validateTimesheetEntry(value, label);
112
+ const obj = ensureObject(value, label);
113
+ return {
114
+ ...entry,
115
+ archivedAt: ensureString(obj.archivedAt, `${label}.archivedAt`),
116
+ archivedReason: ensureString(obj.archivedReason, `${label}.archivedReason`, "invoiced"),
117
+ invoiceNumber: ensureString(obj.invoiceNumber, `${label}.invoiceNumber`),
118
+ invoiceMonth: ensureString(obj.invoiceMonth, `${label}.invoiceMonth`),
119
+ };
120
+ }
121
+ export function validateArchiveData(value) {
122
+ const obj = ensureObject(value, "archive");
123
+ const archivedRaw = obj.archivedEntries === undefined ? [] : obj.archivedEntries;
124
+ if (!Array.isArray(archivedRaw)) {
125
+ throw new Error("archive.archivedEntries must be an array");
126
+ }
127
+ return {
128
+ archivedEntries: archivedRaw.map((entry, index) => validateArchivedEntry(entry, `archive.archivedEntries[${index}]`)),
129
+ };
130
+ }
131
+ //# sourceMappingURL=validators.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validators.js","sourceRoot":"","sources":["../../src/core/validators.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/domain/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,32 @@
1
+ import path from "node:path";
2
+ import nodemailer from "nodemailer";
3
+ export function hasSmtpConfig(env = process.env) {
4
+ return Boolean(env.SMTP_HOST && env.SMTP_USER && env.SMTP_PASS);
5
+ }
6
+ export async function sendInvoiceEmail(input, env = process.env) {
7
+ const smtpPort = Number(env.SMTP_PORT || 465);
8
+ const transporter = nodemailer.createTransport({
9
+ host: env.SMTP_HOST,
10
+ port: smtpPort,
11
+ secure: smtpPort === 465,
12
+ auth: {
13
+ user: env.SMTP_USER,
14
+ pass: env.SMTP_PASS,
15
+ },
16
+ });
17
+ const emailSubject = `Invoice ${input.invoiceNumber} - ${input.clientName}`;
18
+ const emailBody = `Dear ${input.clientName},\n\nPlease find attached invoice ${input.invoiceNumber}.\n\nInvoice Details:\n- Period: ${input.periodFrom} to ${input.periodTo}\n- Amount Due: ${input.currency} ${input.total.toFixed(2)}\n\nThank you for your business!\n\nBest regards,\n${input.senderName || ""}`;
19
+ await transporter.sendMail({
20
+ from: env.INVOICE_FROM || env.SMTP_USER,
21
+ to: input.clientEmail,
22
+ subject: emailSubject,
23
+ text: emailBody,
24
+ attachments: [
25
+ {
26
+ filename: path.basename(input.outPath),
27
+ path: input.outPath,
28
+ },
29
+ ],
30
+ });
31
+ }
32
+ //# sourceMappingURL=email.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"email.js","sourceRoot":"","sources":["../../src/services/email.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,UAAU,MAAM,YAAY,CAAC;AAGpC,MAAM,UAAU,aAAa,CAAC,MAAyB,OAAO,CAAC,GAAG;IAChE,OAAO,OAAO,CAAC,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,SAAS,CAAC,CAAC;AAClE,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,gBAAgB,CACpC,KAA4B,EAC5B,MAAyB,OAAO,CAAC,GAAG;IAEpC,MAAM,QAAQ,GAAG,MAAM,CAAC,GAAG,CAAC,SAAS,IAAI,GAAG,CAAC,CAAC;IAE9C,MAAM,WAAW,GAAG,UAAU,CAAC,eAAe,CAAC;QAC7C,IAAI,EAAE,GAAG,CAAC,SAAS;QACnB,IAAI,EAAE,QAAQ;QACd,MAAM,EAAE,QAAQ,KAAK,GAAG;QACxB,IAAI,EAAE;YACJ,IAAI,EAAE,GAAG,CAAC,SAAS;YACnB,IAAI,EAAE,GAAG,CAAC,SAAS;SACpB;KACF,CAAC,CAAC;IAEH,MAAM,YAAY,GAAG,WAAW,KAAK,CAAC,aAAa,MAAM,KAAK,CAAC,UAAU,EAAE,CAAC;IAC5E,MAAM,SAAS,GAAG,QAAQ,KAAK,CAAC,UAAU,qCAAqC,KAAK,CAAC,aAAa,oCAAoC,KAAK,CAAC,UAAU,OAAO,KAAK,CAAC,QAAQ,mBAAmB,KAAK,CAAC,QAAQ,IAAI,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,sDAAsD,KAAK,CAAC,UAAU,IAAI,EAAE,EAAE,CAAC;IAErT,MAAM,WAAW,CAAC,QAAQ,CAAC;QACzB,IAAI,EAAE,GAAG,CAAC,YAAY,IAAI,GAAG,CAAC,SAAS;QACvC,EAAE,EAAE,KAAK,CAAC,WAAW;QACrB,OAAO,EAAE,YAAY;QACrB,IAAI,EAAE,SAAS;QACf,WAAW,EAAE;YACX;gBACE,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,OAAO,CAAC;gBACtC,IAAI,EAAE,KAAK,CAAC,OAAO;aACpB;SACF;KACF,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,33 @@
1
+ export function getDefaultMonth(now = new Date()) {
2
+ return `${now.getFullYear()}-${String(now.getMonth() + 1).padStart(2, "0")}`;
3
+ }
4
+ export function validateMonth(month) {
5
+ return /^\d{4}-\d{2}$/.test(month);
6
+ }
7
+ export function resolveInvoiceNumber(params) {
8
+ const { month, invoicePrefix, overrideInvoice, seqByMonth } = params;
9
+ const trimmedOverride = overrideInvoice?.trim();
10
+ if (trimmedOverride) {
11
+ return trimmedOverride;
12
+ }
13
+ if (!seqByMonth[month]) {
14
+ seqByMonth[month] = 0;
15
+ }
16
+ seqByMonth[month] += 1;
17
+ const yyyymm = month.replace("-", "");
18
+ const prefix = invoicePrefix?.trim() || "INV";
19
+ return `${prefix}-${yyyymm}-${String(seqByMonth[month]).padStart(3, "0")}`;
20
+ }
21
+ export function getInvoicePeriod(month) {
22
+ const year = Number(month.slice(0, 4));
23
+ const monthNumber = Number(month.slice(5, 7));
24
+ const monthEnd = new Date(year, monthNumber, 0);
25
+ return {
26
+ periodFrom: `${month}-01`,
27
+ periodTo: `${month}-${String(monthEnd.getDate()).padStart(2, "0")}`,
28
+ };
29
+ }
30
+ export function getInvoiceDate(today = new Date()) {
31
+ return today.toISOString().slice(0, 10);
32
+ }
33
+ //# sourceMappingURL=invoice-number.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"invoice-number.js","sourceRoot":"","sources":["../../src/services/invoice-number.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,eAAe,CAAC,GAAG,GAAG,IAAI,IAAI,EAAE;IAC9C,OAAO,GAAG,GAAG,CAAC,WAAW,EAAE,IAAI,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;AAC/E,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,KAAa;IACzC,OAAO,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;AACrC,CAAC;AAED,MAAM,UAAU,oBAAoB,CAAC,MAKpC;IACC,MAAM,EAAE,KAAK,EAAE,aAAa,EAAE,eAAe,EAAE,UAAU,EAAE,GAAG,MAAM,CAAC;IACrE,MAAM,eAAe,GAAG,eAAe,EAAE,IAAI,EAAE,CAAC;IAEhD,IAAI,eAAe,EAAE,CAAC;QACpB,OAAO,eAAe,CAAC;IACzB,CAAC;IAED,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;QACvB,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACxB,CAAC;IAED,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACvB,MAAM,MAAM,GAAG,KAAK,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IACtC,MAAM,MAAM,GAAG,aAAa,EAAE,IAAI,EAAE,IAAI,KAAK,CAAC;IAE9C,OAAO,GAAG,MAAM,IAAI,MAAM,IAAI,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;AAC7E,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,KAAa;IAC5C,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IACvC,MAAM,WAAW,GAAG,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IAC9C,MAAM,QAAQ,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC,CAAC;IAEhD,OAAO;QACL,UAAU,EAAE,GAAG,KAAK,KAAK;QACzB,QAAQ,EAAE,GAAG,KAAK,IAAI,MAAM,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE;KACpE,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,cAAc,CAAC,KAAK,GAAG,IAAI,IAAI,EAAE;IAC/C,OAAO,KAAK,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;AAC1C,CAAC"}
@@ -0,0 +1,69 @@
1
+ import fs from "node:fs";
2
+ import path from "node:path";
3
+ import { pathToFileURL } from "node:url";
4
+ import puppeteer from "puppeteer";
5
+ import { sanitizePathSegment } from "../utils/sanitize.js";
6
+ export async function renderInvoicePdf(paths, payload, month) {
7
+ const clientOutDir = path.join(paths.outDir, sanitizePathSegment(payload.client.name, "client"));
8
+ fs.mkdirSync(clientOutDir, { recursive: true });
9
+ const fileUrl = pathToFileURL(paths.indexHtmlPath).toString();
10
+ const browser = await puppeteer.launch({ headless: true });
11
+ try {
12
+ const page = await browser.newPage();
13
+ await page.goto(fileUrl, { waitUntil: "networkidle0" });
14
+ await page.evaluate((browserPayload) => {
15
+ const setVal = (id, value) => {
16
+ const el = document.getElementById(id);
17
+ if (!el)
18
+ return;
19
+ el.value = String(value ?? "");
20
+ el.dispatchEvent(new Event("input", { bubbles: true }));
21
+ el.dispatchEvent(new Event("change", { bubbles: true }));
22
+ };
23
+ setVal("in_freelancer", browserPayload.from.name);
24
+ setVal("in_email", browserPayload.from.email);
25
+ setVal("in_freelancer_address", browserPayload.from.address);
26
+ setVal("in_client", browserPayload.client.name);
27
+ setVal("in_client_address", browserPayload.client.address);
28
+ setVal("in_currency", browserPayload.currency);
29
+ setVal("in_number", browserPayload.invoiceNumber);
30
+ setVal("in_date", browserPayload.invoiceDate);
31
+ setVal("in_payment_terms", browserPayload.paymentTerms || "");
32
+ setVal("in_note", browserPayload.note || "");
33
+ const win = window;
34
+ if (win.__invoicer?.state && win.__invoicer.builder) {
35
+ win.__invoicer.state.lineItems = browserPayload.lineItems.map((item) => ({
36
+ description: item.description,
37
+ periodFrom: browserPayload.periodFrom,
38
+ periodTo: browserPayload.periodTo,
39
+ days: "",
40
+ hoursPerDay: "",
41
+ quantity: String(item.hours),
42
+ rate: Number(item.rate),
43
+ amount: Number(item.amount),
44
+ }));
45
+ win.__invoicer.builder.renderLineItemsUI();
46
+ win.__invoicer.builder.buildPreview();
47
+ }
48
+ const preview = document.querySelector("#preview");
49
+ if (preview) {
50
+ preview.style.display = "block";
51
+ document.body.innerHTML = "";
52
+ document.body.appendChild(preview);
53
+ }
54
+ }, payload);
55
+ const safeInvoiceNumber = sanitizePathSegment(payload.invoiceNumber, "invoice");
56
+ const outPath = path.join(clientOutDir, `invoice-${month}-${safeInvoiceNumber}.pdf`);
57
+ await page.pdf({
58
+ path: outPath,
59
+ format: "A4",
60
+ printBackground: true,
61
+ margin: { top: "12mm", right: "12mm", bottom: "12mm", left: "12mm" },
62
+ });
63
+ return outPath;
64
+ }
65
+ finally {
66
+ await browser.close();
67
+ }
68
+ }
69
+ //# sourceMappingURL=pdf-render.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pdf-render.js","sourceRoot":"","sources":["../../src/services/pdf-render.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,99 @@
1
+ import { loadJsonFile, writeJsonFile } from "../core/json-store.js";
2
+ import { validateArchiveData, validateTimesheetData } from "../core/validators.js";
3
+ function round2(value) {
4
+ return Math.round(value * 100) / 100;
5
+ }
6
+ export function readTimesheetEntries(paths, filters = {}) {
7
+ const data = loadJsonFile(paths.timesheetPath, { entries: [] }, validateTimesheetData);
8
+ return data.entries.filter((entry) => {
9
+ const matchesClient = !filters.client || entry.client === filters.client;
10
+ const matchesMonth = !filters.month || entry.date.startsWith(filters.month);
11
+ return matchesClient && matchesMonth;
12
+ });
13
+ }
14
+ export function readArchivedEntries(paths, filters = {}) {
15
+ const data = loadJsonFile(paths.archivePath, { archivedEntries: [] }, validateArchiveData);
16
+ return data.archivedEntries.filter((entry) => {
17
+ const matchesClient = !filters.client || entry.client === filters.client;
18
+ const matchesMonth = !filters.month || entry.date.startsWith(filters.month);
19
+ return matchesClient && matchesMonth;
20
+ });
21
+ }
22
+ export function aggregateEntriesByDescription(entries) {
23
+ const groups = {};
24
+ for (const entry of entries) {
25
+ const key = entry.description || "Unnamed Service";
26
+ if (!groups[key]) {
27
+ groups[key] = {
28
+ description: key,
29
+ entries: [],
30
+ totalHours: 0,
31
+ totalAmount: 0,
32
+ rates: [],
33
+ dates: [],
34
+ };
35
+ }
36
+ groups[key].entries.push(entry);
37
+ groups[key].totalHours += entry.hours;
38
+ groups[key].totalAmount += entry.amount;
39
+ groups[key].rates.push(entry.rate);
40
+ groups[key].dates.push(entry.date);
41
+ }
42
+ return Object.values(groups).map((group) => {
43
+ const avgRate = group.totalHours === 0 ? 0 : group.totalAmount / group.totalHours;
44
+ const minRate = Math.min(...group.rates);
45
+ const maxRate = Math.max(...group.rates);
46
+ const dates = [...group.dates].sort();
47
+ return {
48
+ description: group.description,
49
+ hours: round2(group.totalHours),
50
+ rate: round2(avgRate),
51
+ amount: round2(group.totalAmount),
52
+ periodFrom: dates[0],
53
+ periodTo: dates[dates.length - 1],
54
+ rateVariance: minRate !== maxRate,
55
+ minRate,
56
+ maxRate,
57
+ entryIds: group.entries.map((entry) => entry.id),
58
+ };
59
+ });
60
+ }
61
+ export function archiveTimesheetEntries(paths, entryIds, invoiceNumber, invoiceMonth) {
62
+ const entryIdSet = new Set(entryIds);
63
+ const timesheetData = loadJsonFile(paths.timesheetPath, { entries: [] }, validateTimesheetData);
64
+ const archiveData = loadJsonFile(paths.archivePath, { archivedEntries: [] }, validateArchiveData);
65
+ const archivedAt = new Date().toISOString();
66
+ const moved = [];
67
+ timesheetData.entries = timesheetData.entries.filter((entry) => {
68
+ if (!entryIdSet.has(entry.id)) {
69
+ return true;
70
+ }
71
+ moved.push({
72
+ ...entry,
73
+ archivedAt,
74
+ archivedReason: "invoiced",
75
+ invoiceNumber,
76
+ invoiceMonth,
77
+ });
78
+ return false;
79
+ });
80
+ if (moved.length > 0) {
81
+ archiveData.archivedEntries.push(...moved);
82
+ writeJsonFile(paths.timesheetPath, timesheetData);
83
+ writeJsonFile(paths.archivePath, archiveData);
84
+ }
85
+ return moved.length;
86
+ }
87
+ export function toTimesheetCsv(entries) {
88
+ return ["date,client,description,hours,rate,amount"]
89
+ .concat(entries.map((entry) => `${entry.date},${entry.client},"${entry.description}",${entry.hours},${entry.rate},${entry.amount}`))
90
+ .join("\n");
91
+ }
92
+ export function toArchiveCsv(entries) {
93
+ return [
94
+ "date,client,description,hours,rate,amount,invoiceNumber,invoiceMonth,archivedAt",
95
+ ]
96
+ .concat(entries.map((entry) => `${entry.date},${entry.client},"${entry.description}",${entry.hours},${entry.rate},${entry.amount},${entry.invoiceNumber},${entry.invoiceMonth},${entry.archivedAt}`))
97
+ .join("\n");
98
+ }
99
+ //# sourceMappingURL=timesheet.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timesheet.js","sourceRoot":"","sources":["../../src/services/timesheet.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ export function sanitizePathSegment(value, fallback = "invoice") {
2
+ const sanitized = String(value ?? "")
3
+ .replace(/[<>:"/\\|?*\u0000-\u001F]/g, "")
4
+ .replace(/[. ]+$/g, "")
5
+ .trim();
6
+ return sanitized || fallback;
7
+ }
8
+ //# sourceMappingURL=sanitize.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sanitize.js","sourceRoot":"","sources":["../../src/utils/sanitize.ts"],"names":[],"mappings":"AAAA,MAAM,UAAU,mBAAmB,CAAC,KAAc,EAAE,QAAQ,GAAG,SAAS;IACtE,MAAM,SAAS,GAAG,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC;SAClC,OAAO,CAAC,4BAA4B,EAAE,EAAE,CAAC;SACzC,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC;SACtB,IAAI,EAAE,CAAC;IAEV,OAAO,SAAS,IAAI,QAAQ,CAAC;AAC/B,CAAC"}