@fairspec/agent 0.9.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/LICENSE.md +9 -0
- package/README.md +3 -0
- package/build/actions/tableSchema/asHtml.d.ts +4 -0
- package/build/actions/tableSchema/asHtml.js +82 -0
- package/build/actions/tableSchema/asHtml.spec.d.ts +1 -0
- package/build/actions/tableSchema/asHtml.spec.js +356 -0
- package/build/actions/tableSchema/asMarkdown.d.ts +4 -0
- package/build/actions/tableSchema/asMarkdown.js +65 -0
- package/build/actions/tableSchema/asMarkdown.spec.d.ts +1 -0
- package/build/actions/tableSchema/asMarkdown.spec.js +133 -0
- package/build/document/Document.d.ts +2 -0
- package/build/document/Document.js +2 -0
- package/build/document/index.d.ts +3 -0
- package/build/document/index.js +3 -0
- package/build/document/types/Base.d.ts +3 -0
- package/build/document/types/Base.js +2 -0
- package/build/document/types/Json.d.ts +5 -0
- package/build/document/types/Json.js +2 -0
- package/build/document/validate.d.ts +5 -0
- package/build/document/validate.js +23 -0
- package/build/document/validate.spec.d.ts +1 -0
- package/build/document/validate.spec.js +71 -0
- package/build/index.d.ts +1 -0
- package/build/index.js +2 -0
- package/build/package.json +39 -0
- package/build/plugin.d.ts +6 -0
- package/build/plugin.js +15 -0
- package/build/schema/convert/toHtml.d.ts +4 -0
- package/build/schema/convert/toHtml.js +107 -0
- package/build/schema/convert/toHtml.spec.d.ts +1 -0
- package/build/schema/convert/toHtml.spec.js +432 -0
- package/build/schema/convert/toMarkdown.d.ts +4 -0
- package/build/schema/convert/toMarkdown.js +75 -0
- package/build/schema/convert/toMarkdown.spec.d.ts +1 -0
- package/build/schema/convert/toMarkdown.spec.js +137 -0
- package/build/schema/index.d.ts +2 -0
- package/build/schema/index.js +3 -0
- package/build/server.d.ts +2 -0
- package/build/server.js +29 -0
- package/build/tableSchema/convert/toHtml.d.ts +4 -0
- package/build/tableSchema/convert/toHtml.js +80 -0
- package/build/tableSchema/convert/toHtml.spec.d.ts +1 -0
- package/build/tableSchema/convert/toHtml.spec.js +375 -0
- package/build/tableSchema/convert/toMarkdown.d.ts +4 -0
- package/build/tableSchema/convert/toMarkdown.js +57 -0
- package/build/tableSchema/convert/toMarkdown.spec.d.ts +1 -0
- package/build/tableSchema/convert/toMarkdown.spec.js +141 -0
- package/build/tableSchema/index.d.ts +2 -0
- package/build/tableSchema/index.js +3 -0
- package/build/tools/data/validate.d.ts +971 -0
- package/build/tools/data/validate.js +15 -0
- package/build/tools/data/validate.spec.d.ts +1 -0
- package/build/tools/data/validate.spec.js +228 -0
- package/build/tools/dataSchema/infer.d.ts +797 -0
- package/build/tools/dataSchema/infer.js +15 -0
- package/build/tools/dataSchema/infer.spec.d.ts +1 -0
- package/build/tools/dataSchema/infer.spec.js +215 -0
- package/build/tools/dataSchema/validate.d.ts +177 -0
- package/build/tools/dataSchema/validate.js +17 -0
- package/build/tools/dataSchema/validate.spec.d.ts +1 -0
- package/build/tools/dataSchema/validate.spec.js +148 -0
- package/build/tools/dataset/infer.d.ts +1985 -0
- package/build/tools/dataset/infer.js +15 -0
- package/build/tools/dataset/infer.spec.d.ts +1 -0
- package/build/tools/dataset/infer.spec.js +181 -0
- package/build/tools/dataset/validate.d.ts +1168 -0
- package/build/tools/dataset/validate.js +17 -0
- package/build/tools/dataset/validate.spec.d.ts +1 -0
- package/build/tools/dataset/validate.spec.js +212 -0
- package/build/tools/dialect/infer.d.ts +889 -0
- package/build/tools/dialect/infer.js +16 -0
- package/build/tools/dialect/infer.spec.d.ts +1 -0
- package/build/tools/dialect/infer.spec.js +138 -0
- package/build/tools/table/query.d.ts +798 -0
- package/build/tools/table/query.js +24 -0
- package/build/tools/table/query.spec.d.ts +1 -0
- package/build/tools/table/query.spec.js +138 -0
- package/build/tools/table/validate.d.ts +999 -0
- package/build/tools/table/validate.js +16 -0
- package/build/tools/table/validate.spec.d.ts +1 -0
- package/build/tools/table/validate.spec.js +106 -0
- package/build/tools/tableSchema/infer.d.ts +1316 -0
- package/build/tools/tableSchema/infer.js +16 -0
- package/build/tools/tableSchema/infer.spec.d.ts +1 -0
- package/build/tools/tableSchema/infer.spec.js +152 -0
- package/build/tools/tableSchema/validate.d.ts +177 -0
- package/build/tools/tableSchema/validate.js +17 -0
- package/build/tools/tableSchema/validate.spec.d.ts +1 -0
- package/build/tools/tableSchema/validate.spec.js +121 -0
- package/package.json +39 -0
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { loadTable, queryTable, Resource } from "@fairspec/library";
|
|
2
|
+
import { createTool } from "@mastra/core/tools";
|
|
3
|
+
import { z } from "zod";
|
|
4
|
+
export const queryTableTool = createTool({
|
|
5
|
+
id: "query-table",
|
|
6
|
+
description: "Query a table using SQL. Loads the table, executes the SQL query, and returns the results as an array of records.",
|
|
7
|
+
inputSchema: z.object({
|
|
8
|
+
resource: Resource.describe("The table resource to query"),
|
|
9
|
+
query: z
|
|
10
|
+
.string()
|
|
11
|
+
.describe("SQL query to execute (use 'self' as table name)"),
|
|
12
|
+
}),
|
|
13
|
+
outputSchema: z.array(z.record(z.string(), z.unknown())),
|
|
14
|
+
execute: async (input) => {
|
|
15
|
+
const table = await loadTable(input.resource, {});
|
|
16
|
+
if (!table) {
|
|
17
|
+
throw new Error("Could not load table");
|
|
18
|
+
}
|
|
19
|
+
const lazyFrame = queryTable(table, input.query);
|
|
20
|
+
const frame = await lazyFrame.collect();
|
|
21
|
+
return frame.toRecords();
|
|
22
|
+
},
|
|
23
|
+
});
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicXVlcnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90b29scy90YWJsZS9xdWVyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLFVBQVUsRUFBRSxRQUFRLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQTtBQUNuRSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sb0JBQW9CLENBQUE7QUFDL0MsT0FBTyxFQUFFLENBQUMsRUFBRSxNQUFNLEtBQUssQ0FBQTtBQUV2QixNQUFNLENBQUMsTUFBTSxjQUFjLEdBQUcsVUFBVSxDQUFDO0lBQ3ZDLEVBQUUsRUFBRSxhQUFhO0lBQ2pCLFdBQVcsRUFDVCxtSEFBbUg7SUFDckgsV0FBVyxFQUFFLENBQUMsQ0FBQyxNQUFNLENBQUM7UUFDcEIsUUFBUSxFQUFFLFFBQVEsQ0FBQyxRQUFRLENBQUMsNkJBQTZCLENBQUM7UUFDMUQsS0FBSyxFQUFFLENBQUM7YUFDTCxNQUFNLEVBQUU7YUFDUixRQUFRLENBQUMsaURBQWlELENBQUM7S0FDL0QsQ0FBQztJQUNGLFlBQVksRUFBRSxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQ3hELE9BQU8sRUFBRSxLQUFLLEVBQUMsS0FBSyxFQUFDLEVBQUU7UUFDckIsTUFBTSxLQUFLLEdBQUcsTUFBTSxTQUFTLENBQUMsS0FBSyxDQUFDLFFBQVEsRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUNqRCxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDWCxNQUFNLElBQUksS0FBSyxDQUFDLHNCQUFzQixDQUFDLENBQUE7UUFDekMsQ0FBQztRQUVELE1BQU0sU0FBUyxHQUFHLFVBQVUsQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBQ2hELE1BQU0sS0FBSyxHQUFHLE1BQU0sU0FBUyxDQUFDLE9BQU8sRUFBRSxDQUFBO1FBQ3ZDLE9BQU8sS0FBSyxDQUFDLFNBQVMsRUFBRSxDQUFBO0lBQzFCLENBQUM7Q0FDRixDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBsb2FkVGFibGUsIHF1ZXJ5VGFibGUsIFJlc291cmNlIH0gZnJvbSBcIkBmYWlyc3BlYy9saWJyYXJ5XCJcbmltcG9ydCB7IGNyZWF0ZVRvb2wgfSBmcm9tIFwiQG1hc3RyYS9jb3JlL3Rvb2xzXCJcbmltcG9ydCB7IHogfSBmcm9tIFwiem9kXCJcblxuZXhwb3J0IGNvbnN0IHF1ZXJ5VGFibGVUb29sID0gY3JlYXRlVG9vbCh7XG4gIGlkOiBcInF1ZXJ5LXRhYmxlXCIsXG4gIGRlc2NyaXB0aW9uOlxuICAgIFwiUXVlcnkgYSB0YWJsZSB1c2luZyBTUUwuIExvYWRzIHRoZSB0YWJsZSwgZXhlY3V0ZXMgdGhlIFNRTCBxdWVyeSwgYW5kIHJldHVybnMgdGhlIHJlc3VsdHMgYXMgYW4gYXJyYXkgb2YgcmVjb3Jkcy5cIixcbiAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICByZXNvdXJjZTogUmVzb3VyY2UuZGVzY3JpYmUoXCJUaGUgdGFibGUgcmVzb3VyY2UgdG8gcXVlcnlcIiksXG4gICAgcXVlcnk6IHpcbiAgICAgIC5zdHJpbmcoKVxuICAgICAgLmRlc2NyaWJlKFwiU1FMIHF1ZXJ5IHRvIGV4ZWN1dGUgKHVzZSAnc2VsZicgYXMgdGFibGUgbmFtZSlcIiksXG4gIH0pLFxuICBvdXRwdXRTY2hlbWE6IHouYXJyYXkoei5yZWNvcmQoei5zdHJpbmcoKSwgei51bmtub3duKCkpKSxcbiAgZXhlY3V0ZTogYXN5bmMgaW5wdXQgPT4ge1xuICAgIGNvbnN0IHRhYmxlID0gYXdhaXQgbG9hZFRhYmxlKGlucHV0LnJlc291cmNlLCB7fSlcbiAgICBpZiAoIXRhYmxlKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJDb3VsZCBub3QgbG9hZCB0YWJsZVwiKVxuICAgIH1cblxuICAgIGNvbnN0IGxhenlGcmFtZSA9IHF1ZXJ5VGFibGUodGFibGUsIGlucHV0LnF1ZXJ5KVxuICAgIGNvbnN0IGZyYW1lID0gYXdhaXQgbGF6eUZyYW1lLmNvbGxlY3QoKVxuICAgIHJldHVybiBmcmFtZS50b1JlY29yZHMoKVxuICB9LFxufSlcbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
import { writeTempFile } from "@fairspec/library";
|
|
2
|
+
import { describe, expect, it } from "vitest";
|
|
3
|
+
import { queryTableTool } from "./query.js";
|
|
4
|
+
describe("queryTableTool", () => {
|
|
5
|
+
it("validates tool structure", () => {
|
|
6
|
+
expect(queryTableTool.id).toBe("query-table");
|
|
7
|
+
expect(queryTableTool.description).toBeTruthy();
|
|
8
|
+
expect(queryTableTool.inputSchema).toBeTruthy();
|
|
9
|
+
expect(queryTableTool.outputSchema).toBeTruthy();
|
|
10
|
+
expect(queryTableTool.execute).toBeTypeOf("function");
|
|
11
|
+
});
|
|
12
|
+
it("queries table with SELECT statement", async () => {
|
|
13
|
+
const csvContent = "id,name,age\n1,alice,25\n2,bob,30\n3,carol,28";
|
|
14
|
+
const csvPath = await writeTempFile(csvContent, { format: "csv" });
|
|
15
|
+
const resource = { data: csvPath, dialect: { format: "csv" } };
|
|
16
|
+
const result = await queryTableTool.execute?.({
|
|
17
|
+
resource,
|
|
18
|
+
query: "SELECT * FROM self WHERE age > 25",
|
|
19
|
+
}, {});
|
|
20
|
+
expect.assert(result);
|
|
21
|
+
expect.assert(!("error" in result));
|
|
22
|
+
expect(Array.isArray(result)).toBe(true);
|
|
23
|
+
expect(result.length).toBe(2);
|
|
24
|
+
expect(result).toContainEqual({ id: 2, name: "bob", age: 30 });
|
|
25
|
+
expect(result).toContainEqual({ id: 3, name: "carol", age: 28 });
|
|
26
|
+
});
|
|
27
|
+
it("queries table with column selection", async () => {
|
|
28
|
+
const csvContent = "id,name,age,city\n1,alice,25,NYC\n2,bob,30,LA\n3,carol,28,SF";
|
|
29
|
+
const csvPath = await writeTempFile(csvContent, { format: "csv" });
|
|
30
|
+
const resource = { data: csvPath, dialect: { format: "csv" } };
|
|
31
|
+
const result = await queryTableTool.execute?.({
|
|
32
|
+
resource,
|
|
33
|
+
query: "SELECT name, city FROM self",
|
|
34
|
+
}, {});
|
|
35
|
+
expect.assert(result);
|
|
36
|
+
expect.assert(!("error" in result));
|
|
37
|
+
expect(Array.isArray(result)).toBe(true);
|
|
38
|
+
expect(result.length).toBe(3);
|
|
39
|
+
expect(result[0]).toHaveProperty("name");
|
|
40
|
+
expect(result[0]).toHaveProperty("city");
|
|
41
|
+
expect(result[0]).not.toHaveProperty("id");
|
|
42
|
+
expect(result[0]).not.toHaveProperty("age");
|
|
43
|
+
});
|
|
44
|
+
it("queries table with ORDER BY clause", async () => {
|
|
45
|
+
const csvContent = "id,name,score\n1,alice,85\n2,bob,90\n3,carol,88";
|
|
46
|
+
const csvPath = await writeTempFile(csvContent, { format: "csv" });
|
|
47
|
+
const resource = { data: csvPath, dialect: { format: "csv" } };
|
|
48
|
+
const result = await queryTableTool.execute?.({
|
|
49
|
+
resource,
|
|
50
|
+
query: "SELECT * FROM self ORDER BY score DESC",
|
|
51
|
+
}, {});
|
|
52
|
+
expect.assert(result);
|
|
53
|
+
expect.assert(!("error" in result));
|
|
54
|
+
expect(Array.isArray(result)).toBe(true);
|
|
55
|
+
expect(result.length).toBe(3);
|
|
56
|
+
expect(result[0]).toMatchObject({ name: "bob", score: 90 });
|
|
57
|
+
expect(result[1]).toMatchObject({ name: "carol", score: 88 });
|
|
58
|
+
expect(result[2]).toMatchObject({ name: "alice", score: 85 });
|
|
59
|
+
});
|
|
60
|
+
it("queries table with aggregation", async () => {
|
|
61
|
+
const csvContent = "id,name,score\n1,alice,85\n2,bob,90\n3,carol,88";
|
|
62
|
+
const csvPath = await writeTempFile(csvContent, { format: "csv" });
|
|
63
|
+
const resource = { data: csvPath, dialect: { format: "csv" } };
|
|
64
|
+
const result = await queryTableTool.execute?.({
|
|
65
|
+
resource,
|
|
66
|
+
query: "SELECT COUNT(*) as count, AVG(score) as avg_score FROM self",
|
|
67
|
+
}, {});
|
|
68
|
+
expect.assert(result);
|
|
69
|
+
expect.assert(!("error" in result));
|
|
70
|
+
expect(Array.isArray(result)).toBe(true);
|
|
71
|
+
expect(result.length).toBe(1);
|
|
72
|
+
expect(result[0]).toHaveProperty("count");
|
|
73
|
+
expect(result[0]).toHaveProperty("avg_score");
|
|
74
|
+
});
|
|
75
|
+
it("queries table with WHERE and LIMIT", async () => {
|
|
76
|
+
const csvContent = "id,name,age\n1,alice,25\n2,bob,30\n3,carol,28\n4,dave,35\n5,eve,22";
|
|
77
|
+
const csvPath = await writeTempFile(csvContent, { format: "csv" });
|
|
78
|
+
const resource = { data: csvPath, dialect: { format: "csv" } };
|
|
79
|
+
const result = await queryTableTool.execute?.({
|
|
80
|
+
resource,
|
|
81
|
+
query: "SELECT * FROM self WHERE age >= 25 ORDER BY age LIMIT 2",
|
|
82
|
+
}, {});
|
|
83
|
+
expect.assert(result);
|
|
84
|
+
expect.assert(!("error" in result));
|
|
85
|
+
expect(Array.isArray(result)).toBe(true);
|
|
86
|
+
expect(result.length).toBe(2);
|
|
87
|
+
});
|
|
88
|
+
it("queries inline data", async () => {
|
|
89
|
+
const resource = {
|
|
90
|
+
data: [
|
|
91
|
+
{ id: 1, name: "alice", score: 85 },
|
|
92
|
+
{ id: 2, name: "bob", score: 90 },
|
|
93
|
+
{ id: 3, name: "carol", score: 88 },
|
|
94
|
+
],
|
|
95
|
+
};
|
|
96
|
+
const result = await queryTableTool.execute?.({
|
|
97
|
+
resource,
|
|
98
|
+
query: "SELECT name, score FROM self WHERE score > 85",
|
|
99
|
+
}, {});
|
|
100
|
+
expect.assert(result);
|
|
101
|
+
expect.assert(!("error" in result));
|
|
102
|
+
expect(Array.isArray(result)).toBe(true);
|
|
103
|
+
expect(result.length).toBe(2);
|
|
104
|
+
expect(result).toContainEqual({ name: "bob", score: 90 });
|
|
105
|
+
expect(result).toContainEqual({ name: "carol", score: 88 });
|
|
106
|
+
});
|
|
107
|
+
it("queries with JOIN operations", async () => {
|
|
108
|
+
const resource = {
|
|
109
|
+
data: [
|
|
110
|
+
{ id: 1, name: "alice", dept_id: 10 },
|
|
111
|
+
{ id: 2, name: "bob", dept_id: 20 },
|
|
112
|
+
],
|
|
113
|
+
};
|
|
114
|
+
const result = await queryTableTool.execute?.({
|
|
115
|
+
resource,
|
|
116
|
+
query: "SELECT * FROM self WHERE dept_id = 10",
|
|
117
|
+
}, {});
|
|
118
|
+
expect.assert(result);
|
|
119
|
+
expect.assert(!("error" in result));
|
|
120
|
+
expect(Array.isArray(result)).toBe(true);
|
|
121
|
+
expect(result.length).toBe(1);
|
|
122
|
+
expect(result[0]).toMatchObject({ name: "alice", dept_id: 10 });
|
|
123
|
+
});
|
|
124
|
+
it("returns all records when querying without filters", async () => {
|
|
125
|
+
const csvContent = "id,name\n1,alice\n2,bob\n3,carol";
|
|
126
|
+
const csvPath = await writeTempFile(csvContent, { format: "csv" });
|
|
127
|
+
const resource = { data: csvPath, dialect: { format: "csv" } };
|
|
128
|
+
const result = await queryTableTool.execute?.({
|
|
129
|
+
resource,
|
|
130
|
+
query: "SELECT * FROM self",
|
|
131
|
+
}, {});
|
|
132
|
+
expect.assert(result);
|
|
133
|
+
expect.assert(!("error" in result));
|
|
134
|
+
expect(Array.isArray(result)).toBe(true);
|
|
135
|
+
expect(result.length).toBe(3);
|
|
136
|
+
});
|
|
137
|
+
});
|
|
138
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"query.spec.js","sourceRoot":"","sources":["../../../tools/table/query.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Resource } from \"@fairspec/library\"\nimport { writeTempFile } from \"@fairspec/library\"\nimport { describe, expect, it } from \"vitest\"\nimport { queryTableTool } from \"./query.ts\"\n\ndescribe(\"queryTableTool\", () => {\n  it(\"validates tool structure\", () => {\n    expect(queryTableTool.id).toBe(\"query-table\")\n    expect(queryTableTool.description).toBeTruthy()\n    expect(queryTableTool.inputSchema).toBeTruthy()\n    expect(queryTableTool.outputSchema).toBeTruthy()\n    expect(queryTableTool.execute).toBeTypeOf(\"function\")\n  })\n\n  it(\"queries table with SELECT statement\", async () => {\n    const csvContent = \"id,name,age\\n1,alice,25\\n2,bob,30\\n3,carol,28\"\n    const csvPath = await writeTempFile(csvContent, { format: \"csv\" })\n    const resource: Resource = { data: csvPath, dialect: { format: \"csv\" } }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT * FROM self WHERE age > 25\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(2)\n    expect(result).toContainEqual({ id: 2, name: \"bob\", age: 30 })\n    expect(result).toContainEqual({ id: 3, name: \"carol\", age: 28 })\n  })\n\n  it(\"queries table with column selection\", async () => {\n    const csvContent =\n      \"id,name,age,city\\n1,alice,25,NYC\\n2,bob,30,LA\\n3,carol,28,SF\"\n    const csvPath = await writeTempFile(csvContent, { format: \"csv\" })\n    const resource: Resource = { data: csvPath, dialect: { format: \"csv\" } }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT name, city FROM self\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(3)\n    expect(result[0]).toHaveProperty(\"name\")\n    expect(result[0]).toHaveProperty(\"city\")\n    expect(result[0]).not.toHaveProperty(\"id\")\n    expect(result[0]).not.toHaveProperty(\"age\")\n  })\n\n  it(\"queries table with ORDER BY clause\", async () => {\n    const csvContent = \"id,name,score\\n1,alice,85\\n2,bob,90\\n3,carol,88\"\n    const csvPath = await writeTempFile(csvContent, { format: \"csv\" })\n    const resource: Resource = { data: csvPath, dialect: { format: \"csv\" } }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT * FROM self ORDER BY score DESC\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(3)\n    expect(result[0]).toMatchObject({ name: \"bob\", score: 90 })\n    expect(result[1]).toMatchObject({ name: \"carol\", score: 88 })\n    expect(result[2]).toMatchObject({ name: \"alice\", score: 85 })\n  })\n\n  it(\"queries table with aggregation\", async () => {\n    const csvContent = \"id,name,score\\n1,alice,85\\n2,bob,90\\n3,carol,88\"\n    const csvPath = await writeTempFile(csvContent, { format: \"csv\" })\n    const resource: Resource = { data: csvPath, dialect: { format: \"csv\" } }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT COUNT(*) as count, AVG(score) as avg_score FROM self\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(1)\n    expect(result[0]).toHaveProperty(\"count\")\n    expect(result[0]).toHaveProperty(\"avg_score\")\n  })\n\n  it(\"queries table with WHERE and LIMIT\", async () => {\n    const csvContent =\n      \"id,name,age\\n1,alice,25\\n2,bob,30\\n3,carol,28\\n4,dave,35\\n5,eve,22\"\n    const csvPath = await writeTempFile(csvContent, { format: \"csv\" })\n    const resource: Resource = { data: csvPath, dialect: { format: \"csv\" } }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT * FROM self WHERE age >= 25 ORDER BY age LIMIT 2\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(2)\n  })\n\n  it(\"queries inline data\", async () => {\n    const resource: Resource = {\n      data: [\n        { id: 1, name: \"alice\", score: 85 },\n        { id: 2, name: \"bob\", score: 90 },\n        { id: 3, name: \"carol\", score: 88 },\n      ],\n    }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT name, score FROM self WHERE score > 85\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(2)\n    expect(result).toContainEqual({ name: \"bob\", score: 90 })\n    expect(result).toContainEqual({ name: \"carol\", score: 88 })\n  })\n\n  it(\"queries with JOIN operations\", async () => {\n    const resource: Resource = {\n      data: [\n        { id: 1, name: \"alice\", dept_id: 10 },\n        { id: 2, name: \"bob\", dept_id: 20 },\n      ],\n    }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT * FROM self WHERE dept_id = 10\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(1)\n    expect(result[0]).toMatchObject({ name: \"alice\", dept_id: 10 })\n  })\n\n  it(\"returns all records when querying without filters\", async () => {\n    const csvContent = \"id,name\\n1,alice\\n2,bob\\n3,carol\"\n    const csvPath = await writeTempFile(csvContent, { format: \"csv\" })\n    const resource: Resource = { data: csvPath, dialect: { format: \"csv\" } }\n\n    const result = await queryTableTool.execute?.(\n      {\n        resource,\n        query: \"SELECT * FROM self\",\n      },\n      {},\n    )\n\n    expect.assert(result)\n    expect.assert(!(\"error\" in result))\n\n    expect(Array.isArray(result)).toBe(true)\n    expect(result.length).toBe(3)\n  })\n})\n"]}
|