@vint.tri/report_gen_mcp 1.0.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/dist/charts/bar.js +42 -0
- package/dist/charts/line.js +42 -0
- package/dist/charts/pie.js +41 -0
- package/dist/index.js +42 -0
- package/dist/utils/reportGenerator.js +59 -0
- package/generated-report.html +1 -0
- package/package.json +33 -0
- package/src/charts/bar.ts +42 -0
- package/src/charts/line.ts +42 -0
- package/src/charts/pie.ts +41 -0
- package/src/index.ts +43 -0
- package/src/utils/reportGenerator.ts +60 -0
- package/test-cli-report-fixed.html +14 -0
- package/test-cli-report.html +0 -0
- package/test-report.html +19 -0
- package/test-report.json +22 -0
- package/tsconfig.json +13 -0
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.barSchema = void 0;
|
|
4
|
+
exports.renderBarChart = renderBarChart;
|
|
5
|
+
const zod_1 = require("zod");
|
|
6
|
+
const chartjs_node_canvas_1 = require("chartjs-node-canvas");
|
|
7
|
+
exports.barSchema = zod_1.z.object({
|
|
8
|
+
labels: zod_1.z.array(zod_1.z.string()),
|
|
9
|
+
datasets: zod_1.z.array(zod_1.z.object({
|
|
10
|
+
label: zod_1.z.string(),
|
|
11
|
+
data: zod_1.z.array(zod_1.z.number()),
|
|
12
|
+
backgroundColor: zod_1.z.array(zod_1.z.string()).optional(),
|
|
13
|
+
})),
|
|
14
|
+
options: zod_1.z.object({
|
|
15
|
+
title: zod_1.z.string().optional(),
|
|
16
|
+
// Add more Chart.js options as needed
|
|
17
|
+
}).optional(),
|
|
18
|
+
});
|
|
19
|
+
async function renderBarChart(config) {
|
|
20
|
+
const chartConfig = {
|
|
21
|
+
type: 'bar',
|
|
22
|
+
data: {
|
|
23
|
+
labels: config.labels,
|
|
24
|
+
datasets: config.datasets,
|
|
25
|
+
},
|
|
26
|
+
options: {
|
|
27
|
+
...config.options,
|
|
28
|
+
plugins: {
|
|
29
|
+
title: config.options?.title ? {
|
|
30
|
+
display: true,
|
|
31
|
+
text: config.options.title
|
|
32
|
+
} : undefined
|
|
33
|
+
}
|
|
34
|
+
},
|
|
35
|
+
};
|
|
36
|
+
const width = 800;
|
|
37
|
+
const height = 600;
|
|
38
|
+
const chartJSNodeCanvas = new chartjs_node_canvas_1.ChartJSNodeCanvas({ width, height });
|
|
39
|
+
const buffer = await chartJSNodeCanvas.renderToBuffer(chartConfig);
|
|
40
|
+
const base64Image = buffer.toString('base64');
|
|
41
|
+
return `<img src="data:image/png;base64,${base64Image}" alt="Chart" />`;
|
|
42
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.lineSchema = void 0;
|
|
4
|
+
exports.renderLineChart = renderLineChart;
|
|
5
|
+
const zod_1 = require("zod");
|
|
6
|
+
const chartjs_node_canvas_1 = require("chartjs-node-canvas");
|
|
7
|
+
exports.lineSchema = zod_1.z.object({
|
|
8
|
+
labels: zod_1.z.array(zod_1.z.string()),
|
|
9
|
+
datasets: zod_1.z.array(zod_1.z.object({
|
|
10
|
+
label: zod_1.z.string(),
|
|
11
|
+
data: zod_1.z.array(zod_1.z.number()),
|
|
12
|
+
borderColor: zod_1.z.string().optional(),
|
|
13
|
+
})),
|
|
14
|
+
options: zod_1.z.object({
|
|
15
|
+
title: zod_1.z.string().optional(),
|
|
16
|
+
// Add more Chart.js options as needed
|
|
17
|
+
}).optional(),
|
|
18
|
+
});
|
|
19
|
+
async function renderLineChart(config) {
|
|
20
|
+
const chartConfig = {
|
|
21
|
+
type: 'line',
|
|
22
|
+
data: {
|
|
23
|
+
labels: config.labels,
|
|
24
|
+
datasets: config.datasets,
|
|
25
|
+
},
|
|
26
|
+
options: {
|
|
27
|
+
...config.options,
|
|
28
|
+
plugins: {
|
|
29
|
+
title: config.options?.title ? {
|
|
30
|
+
display: true,
|
|
31
|
+
text: config.options.title
|
|
32
|
+
} : undefined
|
|
33
|
+
}
|
|
34
|
+
},
|
|
35
|
+
};
|
|
36
|
+
const width = 800;
|
|
37
|
+
const height = 600;
|
|
38
|
+
const chartJSNodeCanvas = new chartjs_node_canvas_1.ChartJSNodeCanvas({ width, height });
|
|
39
|
+
const buffer = await chartJSNodeCanvas.renderToBuffer(chartConfig);
|
|
40
|
+
const base64Image = buffer.toString('base64');
|
|
41
|
+
return `<img src="data:image/png;base64,${base64Image}" alt="Chart" />`;
|
|
42
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.pieSchema = void 0;
|
|
4
|
+
exports.renderPieChart = renderPieChart;
|
|
5
|
+
const zod_1 = require("zod");
|
|
6
|
+
const chartjs_node_canvas_1 = require("chartjs-node-canvas");
|
|
7
|
+
exports.pieSchema = zod_1.z.object({
|
|
8
|
+
labels: zod_1.z.array(zod_1.z.string()),
|
|
9
|
+
datasets: zod_1.z.array(zod_1.z.object({
|
|
10
|
+
data: zod_1.z.array(zod_1.z.number()),
|
|
11
|
+
backgroundColor: zod_1.z.array(zod_1.z.string()).optional(),
|
|
12
|
+
})),
|
|
13
|
+
options: zod_1.z.object({
|
|
14
|
+
title: zod_1.z.string().optional(),
|
|
15
|
+
// Add more Chart.js options as needed
|
|
16
|
+
}).optional(),
|
|
17
|
+
});
|
|
18
|
+
async function renderPieChart(config) {
|
|
19
|
+
const chartConfig = {
|
|
20
|
+
type: 'pie',
|
|
21
|
+
data: {
|
|
22
|
+
labels: config.labels,
|
|
23
|
+
datasets: config.datasets,
|
|
24
|
+
},
|
|
25
|
+
options: {
|
|
26
|
+
...config.options,
|
|
27
|
+
plugins: {
|
|
28
|
+
title: config.options?.title ? {
|
|
29
|
+
display: true,
|
|
30
|
+
text: config.options.title
|
|
31
|
+
} : undefined
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
};
|
|
35
|
+
const width = 800;
|
|
36
|
+
const height = 600;
|
|
37
|
+
const chartJSNodeCanvas = new chartjs_node_canvas_1.ChartJSNodeCanvas({ width, height });
|
|
38
|
+
const buffer = await chartJSNodeCanvas.renderToBuffer(chartConfig);
|
|
39
|
+
const base64Image = buffer.toString('base64');
|
|
40
|
+
return `<img src="data:image/png;base64,${base64Image}" alt="Chart" />`;
|
|
41
|
+
}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
"use strict";
|
|
3
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
4
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
5
|
+
};
|
|
6
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
7
|
+
const express_1 = __importDefault(require("express"));
|
|
8
|
+
const commander_1 = require("commander");
|
|
9
|
+
const reportGenerator_1 = require("./utils/reportGenerator");
|
|
10
|
+
const path_1 = __importDefault(require("path"));
|
|
11
|
+
const app = (0, express_1.default)();
|
|
12
|
+
const port = 3000;
|
|
13
|
+
app.use(express_1.default.json());
|
|
14
|
+
app.post('/generate-report', async (req, res) => {
|
|
15
|
+
const { document, charts, outputFile = 'report.html' } = req.body;
|
|
16
|
+
try {
|
|
17
|
+
const result = await (0, reportGenerator_1.generateReport)(document, charts, path_1.default.resolve(outputFile));
|
|
18
|
+
res.json(result);
|
|
19
|
+
}
|
|
20
|
+
catch (error) {
|
|
21
|
+
res.status(500).json({ error: error.message });
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
commander_1.program
|
|
25
|
+
.command('start-server')
|
|
26
|
+
.description('Start the MCP report generation server')
|
|
27
|
+
.action(() => {
|
|
28
|
+
app.listen(port, () => {
|
|
29
|
+
console.log(`Server running at http://localhost:${port}`);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
commander_1.program
|
|
33
|
+
.command('generate')
|
|
34
|
+
.option('--document <md>', 'Markdown document with placeholders [[chart:id]]')
|
|
35
|
+
.option('--charts <json>', 'JSON string of charts {id: {type: "bar", config: {...}}}')
|
|
36
|
+
.option('--output <file>', 'Output HTML file', 'report.html')
|
|
37
|
+
.action(async (opts) => {
|
|
38
|
+
const charts = JSON.parse(opts.charts);
|
|
39
|
+
const result = await (0, reportGenerator_1.generateReport)(opts.document, charts, opts.output);
|
|
40
|
+
console.log(result);
|
|
41
|
+
});
|
|
42
|
+
commander_1.program.parse();
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.generateReport = generateReport;
|
|
7
|
+
const fs_extra_1 = __importDefault(require("fs-extra"));
|
|
8
|
+
const ejs_1 = __importDefault(require("ejs"));
|
|
9
|
+
const marked_1 = require("marked");
|
|
10
|
+
const bar_1 = require("../charts/bar");
|
|
11
|
+
const line_1 = require("../charts/line");
|
|
12
|
+
const pie_1 = require("../charts/pie");
|
|
13
|
+
// Add more imports for other chart types as needed
|
|
14
|
+
const chartRenderers = {
|
|
15
|
+
bar: { schema: bar_1.barSchema, renderer: bar_1.renderBarChart },
|
|
16
|
+
line: { schema: line_1.lineSchema, renderer: line_1.renderLineChart },
|
|
17
|
+
pie: { schema: pie_1.pieSchema, renderer: pie_1.renderPieChart },
|
|
18
|
+
// Add more chart types here
|
|
19
|
+
};
|
|
20
|
+
async function generateReport(document, charts, outputFile) {
|
|
21
|
+
// Validate charts
|
|
22
|
+
for (const [id, { type, config }] of Object.entries(charts)) {
|
|
23
|
+
const { schema } = chartRenderers[type];
|
|
24
|
+
if (!schema)
|
|
25
|
+
throw new Error(`Unsupported chart type: ${type}`);
|
|
26
|
+
schema.parse(config);
|
|
27
|
+
}
|
|
28
|
+
// Render charts to HTML img tags
|
|
29
|
+
const renderedCharts = {};
|
|
30
|
+
for (const [id, { type, config }] of Object.entries(charts)) {
|
|
31
|
+
const htmlImg = await chartRenderers[type].renderer(config);
|
|
32
|
+
renderedCharts[id] = htmlImg;
|
|
33
|
+
}
|
|
34
|
+
// Replace placeholders in Markdown, e.g., [[chart:id]]
|
|
35
|
+
let mdWithCharts = document;
|
|
36
|
+
for (const [id, htmlImg] of Object.entries(renderedCharts)) {
|
|
37
|
+
mdWithCharts = mdWithCharts.replace(new RegExp(`\\[\\[chart:${id}\\]\\]`, 'g'), htmlImg);
|
|
38
|
+
}
|
|
39
|
+
// Convert Markdown to HTML
|
|
40
|
+
const htmlContent = await (0, marked_1.marked)(mdWithCharts);
|
|
41
|
+
// Wrap in full HTML template
|
|
42
|
+
const template = `
|
|
43
|
+
<!DOCTYPE html>
|
|
44
|
+
<html lang="en">
|
|
45
|
+
<head>
|
|
46
|
+
<meta charset="UTF-8">
|
|
47
|
+
<title>Report</title>
|
|
48
|
+
<style> body { font-family: Arial, sans-serif; } img { max-width: 100%; } </style>
|
|
49
|
+
</head>
|
|
50
|
+
<body>
|
|
51
|
+
<%- htmlContent %>
|
|
52
|
+
</body>
|
|
53
|
+
</html>
|
|
54
|
+
`;
|
|
55
|
+
const fullHtml = ejs_1.default.render(template, { htmlContent }, { escape: (str) => str });
|
|
56
|
+
// Save to file
|
|
57
|
+
await fs_extra_1.default.writeFile(outputFile, fullHtml);
|
|
58
|
+
return { success: true, filePath: outputFile };
|
|
59
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"success":true,"filePath":"/Applications/Python/report_gen_mcp/test-report.html"}
|
package/package.json
ADDED
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@vint.tri/report_gen_mcp",
|
|
3
|
+
"version": "1.0.0",
|
|
4
|
+
"description": "CLI tool for generating HTML reports with embedded charts",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"bin": {
|
|
7
|
+
"report_gen_mcp": "dist/index.js"
|
|
8
|
+
},
|
|
9
|
+
"scripts": {
|
|
10
|
+
"build": "tsc",
|
|
11
|
+
"start": "node dist/index.js",
|
|
12
|
+
"test": "echo \"No tests yet\""
|
|
13
|
+
},
|
|
14
|
+
"dependencies": {
|
|
15
|
+
"chart.js": "^3.9.1",
|
|
16
|
+
"chartjs-node-canvas": "^4.1.6",
|
|
17
|
+
"commander": "^12.1.0",
|
|
18
|
+
"ejs": "^3.1.10",
|
|
19
|
+
"express": "^4.19.2",
|
|
20
|
+
"fs-extra": "^11.2.0",
|
|
21
|
+
"marked": "^13.0.2",
|
|
22
|
+
"zod": "^3.23.8"
|
|
23
|
+
},
|
|
24
|
+
"devDependencies": {
|
|
25
|
+
"@types/ejs": "^3.1.5",
|
|
26
|
+
"@types/express": "^4.17.21",
|
|
27
|
+
"@types/fs-extra": "^11.0.4",
|
|
28
|
+
"@types/node": "^22.4.1",
|
|
29
|
+
"typescript": "^5.5.4"
|
|
30
|
+
},
|
|
31
|
+
"license": "MIT",
|
|
32
|
+
"author": ""
|
|
33
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import { ChartConfiguration } from 'chart.js';
|
|
3
|
+
import { ChartJSNodeCanvas } from 'chartjs-node-canvas';
|
|
4
|
+
|
|
5
|
+
export const barSchema = z.object({
|
|
6
|
+
labels: z.array(z.string()),
|
|
7
|
+
datasets: z.array(z.object({
|
|
8
|
+
label: z.string(),
|
|
9
|
+
data: z.array(z.number()),
|
|
10
|
+
backgroundColor: z.array(z.string()).optional(),
|
|
11
|
+
})),
|
|
12
|
+
options: z.object({
|
|
13
|
+
title: z.string().optional(),
|
|
14
|
+
// Add more Chart.js options as needed
|
|
15
|
+
}).optional(),
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
export async function renderBarChart(config: z.infer<typeof barSchema>): Promise<string> {
|
|
19
|
+
const chartConfig: ChartConfiguration = {
|
|
20
|
+
type: 'bar',
|
|
21
|
+
data: {
|
|
22
|
+
labels: config.labels,
|
|
23
|
+
datasets: config.datasets,
|
|
24
|
+
},
|
|
25
|
+
options: {
|
|
26
|
+
...config.options,
|
|
27
|
+
plugins: {
|
|
28
|
+
title: config.options?.title ? {
|
|
29
|
+
display: true,
|
|
30
|
+
text: config.options.title
|
|
31
|
+
} : undefined
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
const width = 800;
|
|
37
|
+
const height = 600;
|
|
38
|
+
const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height });
|
|
39
|
+
const buffer = await chartJSNodeCanvas.renderToBuffer(chartConfig);
|
|
40
|
+
const base64Image = buffer.toString('base64');
|
|
41
|
+
return `<img src="data:image/png;base64,${base64Image}" alt="Chart" />`;
|
|
42
|
+
}
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import { ChartConfiguration } from 'chart.js';
|
|
3
|
+
import { ChartJSNodeCanvas } from 'chartjs-node-canvas';
|
|
4
|
+
|
|
5
|
+
export const lineSchema = z.object({
|
|
6
|
+
labels: z.array(z.string()),
|
|
7
|
+
datasets: z.array(z.object({
|
|
8
|
+
label: z.string(),
|
|
9
|
+
data: z.array(z.number()),
|
|
10
|
+
borderColor: z.string().optional(),
|
|
11
|
+
})),
|
|
12
|
+
options: z.object({
|
|
13
|
+
title: z.string().optional(),
|
|
14
|
+
// Add more Chart.js options as needed
|
|
15
|
+
}).optional(),
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
export async function renderLineChart(config: z.infer<typeof lineSchema>): Promise<string> {
|
|
19
|
+
const chartConfig: ChartConfiguration = {
|
|
20
|
+
type: 'line',
|
|
21
|
+
data: {
|
|
22
|
+
labels: config.labels,
|
|
23
|
+
datasets: config.datasets,
|
|
24
|
+
},
|
|
25
|
+
options: {
|
|
26
|
+
...config.options,
|
|
27
|
+
plugins: {
|
|
28
|
+
title: config.options?.title ? {
|
|
29
|
+
display: true,
|
|
30
|
+
text: config.options.title
|
|
31
|
+
} : undefined
|
|
32
|
+
}
|
|
33
|
+
},
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
const width = 800;
|
|
37
|
+
const height = 600;
|
|
38
|
+
const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height });
|
|
39
|
+
const buffer = await chartJSNodeCanvas.renderToBuffer(chartConfig);
|
|
40
|
+
const base64Image = buffer.toString('base64');
|
|
41
|
+
return `<img src="data:image/png;base64,${base64Image}" alt="Chart" />`;
|
|
42
|
+
}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import { ChartConfiguration } from 'chart.js';
|
|
3
|
+
import { ChartJSNodeCanvas } from 'chartjs-node-canvas';
|
|
4
|
+
|
|
5
|
+
export const pieSchema = z.object({
|
|
6
|
+
labels: z.array(z.string()),
|
|
7
|
+
datasets: z.array(z.object({
|
|
8
|
+
data: z.array(z.number()),
|
|
9
|
+
backgroundColor: z.array(z.string()).optional(),
|
|
10
|
+
})),
|
|
11
|
+
options: z.object({
|
|
12
|
+
title: z.string().optional(),
|
|
13
|
+
// Add more Chart.js options as needed
|
|
14
|
+
}).optional(),
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
export async function renderPieChart(config: z.infer<typeof pieSchema>): Promise<string> {
|
|
18
|
+
const chartConfig: ChartConfiguration = {
|
|
19
|
+
type: 'pie',
|
|
20
|
+
data: {
|
|
21
|
+
labels: config.labels,
|
|
22
|
+
datasets: config.datasets,
|
|
23
|
+
},
|
|
24
|
+
options: {
|
|
25
|
+
...config.options,
|
|
26
|
+
plugins: {
|
|
27
|
+
title: config.options?.title ? {
|
|
28
|
+
display: true,
|
|
29
|
+
text: config.options.title
|
|
30
|
+
} : undefined
|
|
31
|
+
}
|
|
32
|
+
},
|
|
33
|
+
};
|
|
34
|
+
|
|
35
|
+
const width = 800;
|
|
36
|
+
const height = 600;
|
|
37
|
+
const chartJSNodeCanvas = new ChartJSNodeCanvas({ width, height });
|
|
38
|
+
const buffer = await chartJSNodeCanvas.renderToBuffer(chartConfig);
|
|
39
|
+
const base64Image = buffer.toString('base64');
|
|
40
|
+
return `<img src="data:image/png;base64,${base64Image}" alt="Chart" />`;
|
|
41
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import express from 'express';
|
|
4
|
+
import { program } from 'commander';
|
|
5
|
+
import { generateReport } from './utils/reportGenerator';
|
|
6
|
+
import path from 'path';
|
|
7
|
+
|
|
8
|
+
const app = express();
|
|
9
|
+
const port = 3000;
|
|
10
|
+
|
|
11
|
+
app.use(express.json());
|
|
12
|
+
|
|
13
|
+
app.post('/generate-report', async (req, res) => {
|
|
14
|
+
const { document, charts, outputFile = 'report.html' } = req.body;
|
|
15
|
+
try {
|
|
16
|
+
const result = await generateReport(document, charts, path.resolve(outputFile));
|
|
17
|
+
res.json(result);
|
|
18
|
+
} catch (error) {
|
|
19
|
+
res.status(500).json({ error: (error as Error).message });
|
|
20
|
+
}
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
program
|
|
24
|
+
.command('start-server')
|
|
25
|
+
.description('Start the MCP report generation server')
|
|
26
|
+
.action(() => {
|
|
27
|
+
app.listen(port, () => {
|
|
28
|
+
console.log(`Server running at http://localhost:${port}`);
|
|
29
|
+
});
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
program
|
|
33
|
+
.command('generate')
|
|
34
|
+
.option('--document <md>', 'Markdown document with placeholders [[chart:id]]')
|
|
35
|
+
.option('--charts <json>', 'JSON string of charts {id: {type: "bar", config: {...}}}')
|
|
36
|
+
.option('--output <file>', 'Output HTML file', 'report.html')
|
|
37
|
+
.action(async (opts) => {
|
|
38
|
+
const charts = JSON.parse(opts.charts);
|
|
39
|
+
const result = await generateReport(opts.document, charts, opts.output);
|
|
40
|
+
console.log(result);
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
program.parse();
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
import fs from 'fs-extra';
|
|
2
|
+
import ejs from 'ejs';
|
|
3
|
+
import { marked } from 'marked';
|
|
4
|
+
import { z } from 'zod';
|
|
5
|
+
import { barSchema, renderBarChart } from '../charts/bar';
|
|
6
|
+
import { lineSchema, renderLineChart } from '../charts/line';
|
|
7
|
+
import { pieSchema, renderPieChart } from '../charts/pie';
|
|
8
|
+
// Add more imports for other chart types as needed
|
|
9
|
+
|
|
10
|
+
const chartRenderers: Record<string, { schema: z.ZodObject<any>; renderer: (config: any) => Promise<string> }> = {
|
|
11
|
+
bar: { schema: barSchema, renderer: renderBarChart },
|
|
12
|
+
line: { schema: lineSchema, renderer: renderLineChart },
|
|
13
|
+
pie: { schema: pieSchema, renderer: renderPieChart },
|
|
14
|
+
// Add more chart types here
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
export async function generateReport(document: string, charts: Record<string, { type: string; config: any }>, outputFile: string) {
|
|
18
|
+
// Validate charts
|
|
19
|
+
for (const [id, { type, config }] of Object.entries(charts)) {
|
|
20
|
+
const { schema } = chartRenderers[type];
|
|
21
|
+
if (!schema) throw new Error(`Unsupported chart type: ${type}`);
|
|
22
|
+
schema.parse(config);
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
// Render charts to HTML img tags
|
|
26
|
+
const renderedCharts: Record<string, string> = {};
|
|
27
|
+
for (const [id, { type, config }] of Object.entries(charts)) {
|
|
28
|
+
const htmlImg = await chartRenderers[type].renderer(config);
|
|
29
|
+
renderedCharts[id] = htmlImg;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// Replace placeholders in Markdown, e.g., [[chart:id]]
|
|
33
|
+
let mdWithCharts = document;
|
|
34
|
+
for (const [id, htmlImg] of Object.entries(renderedCharts)) {
|
|
35
|
+
mdWithCharts = mdWithCharts.replace(new RegExp(`\\[\\[chart:${id}\\]\\]`, 'g'), htmlImg);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// Convert Markdown to HTML
|
|
39
|
+
const htmlContent = await marked(mdWithCharts);
|
|
40
|
+
|
|
41
|
+
// Wrap in full HTML template
|
|
42
|
+
const template = `
|
|
43
|
+
<!DOCTYPE html>
|
|
44
|
+
<html lang="en">
|
|
45
|
+
<head>
|
|
46
|
+
<meta charset="UTF-8">
|
|
47
|
+
<title>Report</title>
|
|
48
|
+
<style> body { font-family: Arial, sans-serif; } img { max-width: 100%; } </style>
|
|
49
|
+
</head>
|
|
50
|
+
<body>
|
|
51
|
+
<%- htmlContent %>
|
|
52
|
+
</body>
|
|
53
|
+
</html>
|
|
54
|
+
`;
|
|
55
|
+
const fullHtml = ejs.render(template, { htmlContent }, { escape: (str) => str });
|
|
56
|
+
|
|
57
|
+
// Save to file
|
|
58
|
+
await fs.writeFile(outputFile, fullHtml);
|
|
59
|
+
return { success: true, filePath: outputFile };
|
|
60
|
+
}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
|
|
2
|
+
<!DOCTYPE html>
|
|
3
|
+
<html lang="en">
|
|
4
|
+
<head>
|
|
5
|
+
<meta charset="UTF-8">
|
|
6
|
+
<title>Report</title>
|
|
7
|
+
<style> body { font-family: Arial, sans-serif; } svg { max-width: 100%; } </style>
|
|
8
|
+
</head>
|
|
9
|
+
<body>
|
|
10
|
+
<h1>Test Report\n\nThis is a test report.\n\n<img src="" alt="Chart" /></h1>
|
|
11
|
+
|
|
12
|
+
</body>
|
|
13
|
+
</html>
|
|
14
|
+
|
|
Binary file
|
package/test-report.html
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
|
|
2
|
+
<!DOCTYPE html>
|
|
3
|
+
<html lang="en">
|
|
4
|
+
<head>
|
|
5
|
+
<meta charset="UTF-8">
|
|
6
|
+
<title>Report</title>
|
|
7
|
+
<style> body { font-family: Arial, sans-serif; } img { max-width: 100%; } </style>
|
|
8
|
+
</head>
|
|
9
|
+
<body>
|
|
10
|
+
<h1>Sales Report</h1>
|
|
11
|
+
<p>This is our monthly sales report.</p>
|
|
12
|
+
<img src="" alt="Chart" />
|
|
13
|
+
|
|
14
|
+
<h2>Conclusion</h2>
|
|
15
|
+
<p>This is the end of our report.</p>
|
|
16
|
+
|
|
17
|
+
</body>
|
|
18
|
+
</html>
|
|
19
|
+
|
package/test-report.json
ADDED
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
{
|
|
2
|
+
"document": "# Sales Report\n\nThis is our monthly sales report.\n\n[[chart:sales]]\n\n## Conclusion\n\nThis is the end of our report.",
|
|
3
|
+
"charts": {
|
|
4
|
+
"sales": {
|
|
5
|
+
"type": "bar",
|
|
6
|
+
"config": {
|
|
7
|
+
"labels": ["January", "February", "March", "April", "May"],
|
|
8
|
+
"datasets": [
|
|
9
|
+
{
|
|
10
|
+
"label": "Sales",
|
|
11
|
+
"data": [100, 150, 200, 175, 225],
|
|
12
|
+
"backgroundColor": ["#FF6384", "#36A2EB", "#FFCE56", "#4BC0C0", "#9966FF"]
|
|
13
|
+
}
|
|
14
|
+
],
|
|
15
|
+
"options": {
|
|
16
|
+
"title": "Monthly Sales Data"
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
},
|
|
21
|
+
"outputFile": "test-report.html"
|
|
22
|
+
}
|
package/tsconfig.json
ADDED
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
{
|
|
2
|
+
"compilerOptions": {
|
|
3
|
+
"target": "ES2020",
|
|
4
|
+
"module": "commonjs",
|
|
5
|
+
"outDir": "./dist",
|
|
6
|
+
"strict": true,
|
|
7
|
+
"esModuleInterop": true,
|
|
8
|
+
"skipLibCheck": true,
|
|
9
|
+
"forceConsistentCasingInFileNames": true
|
|
10
|
+
},
|
|
11
|
+
"include": ["src/**/*"],
|
|
12
|
+
"exclude": ["node_modules"]
|
|
13
|
+
}
|