@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.
@@ -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
+ &lt;h1&gt;Test Report\n\nThis is a test report.\n\n&lt;img src=&#34;&#34; alt=&#34;Chart&#34; /&gt;&lt;/h1&gt;
11
+
12
+ </body>
13
+ </html>
14
+
Binary file
@@ -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
+
@@ -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
+ }