@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.
- package/.env.example +15 -0
- package/config.json +19 -0
- package/data/last.json +11 -0
- package/dist/cli.js +38 -0
- package/dist/cli.js.map +1 -0
- package/dist/commands/clients.js +184 -0
- package/dist/commands/clients.js.map +1 -0
- package/dist/commands/generate.js +294 -0
- package/dist/commands/generate.js.map +1 -0
- package/dist/commands/timesheet.js +34 -0
- package/dist/commands/timesheet.js.map +1 -0
- package/dist/core/json-store.js +27 -0
- package/dist/core/json-store.js.map +1 -0
- package/dist/core/paths.js +29 -0
- package/dist/core/paths.js.map +1 -0
- package/dist/core/validators.js +131 -0
- package/dist/core/validators.js.map +1 -0
- package/dist/domain/types.js +2 -0
- package/dist/domain/types.js.map +1 -0
- package/dist/services/email.js +32 -0
- package/dist/services/email.js.map +1 -0
- package/dist/services/invoice-number.js +33 -0
- package/dist/services/invoice-number.js.map +1 -0
- package/dist/services/pdf-render.js +69 -0
- package/dist/services/pdf-render.js.map +1 -0
- package/dist/services/timesheet.js +99 -0
- package/dist/services/timesheet.js.map +1 -0
- package/dist/utils/sanitize.js +8 -0
- package/dist/utils/sanitize.js.map +1 -0
- package/index.html +2584 -0
- package/logo.svg +17 -0
- package/package.json +39 -0
- package/scripts/invoice.mjs +23 -0
- package/src/cli.ts +44 -0
- package/src/commands/clients.ts +221 -0
- package/src/commands/generate.ts +379 -0
- package/src/commands/timesheet.ts +47 -0
- package/src/core/json-store.ts +33 -0
- package/src/core/paths.ts +42 -0
- package/src/core/validators.ts +168 -0
- package/src/domain/types.ts +129 -0
- package/src/services/email.ts +40 -0
- package/src/services/invoice-number.ts +46 -0
- package/src/services/pdf-render.ts +95 -0
- package/src/services/timesheet.ts +173 -0
- package/src/utils/sanitize.ts +8 -0
- package/test/cli-wiring.test.ts +25 -0
- package/test/invoice-number.test.ts +45 -0
- package/test/timesheet.test.ts +104 -0
- 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 @@
|
|
|
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 @@
|
|
|
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"}
|