qwed-open-responses 0.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/README.md ADDED
@@ -0,0 +1,135 @@
1
+ # QWED Open Responses (Node.js)
2
+
3
+ [![npm version](https://badge.fury.io/js/qwed-open-responses.svg)](https://badge.fury.io/js/qwed-open-responses)
4
+ [![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)
5
+
6
+ **Verification guards for AI agent outputs in Node.js/Express applications.**
7
+
8
+ ## Installation
9
+
10
+ ```bash
11
+ npm install qwed-open-responses
12
+ ```
13
+
14
+ ## Quick Start
15
+
16
+ ```typescript
17
+ import express from 'express';
18
+ import { createQWEDMiddleware, ToolGuard, SafetyGuard } from 'qwed-open-responses';
19
+
20
+ const app = express();
21
+ app.use(express.json());
22
+
23
+ // Add verification middleware
24
+ app.use(createQWEDMiddleware({
25
+ guards: [new ToolGuard(), new SafetyGuard()],
26
+ blockOnFailure: true,
27
+ }));
28
+
29
+ app.post('/api/agent', (req, res) => {
30
+ // Response will be verified before sending
31
+ res.json({
32
+ tool_calls: [
33
+ { name: 'search', arguments: { query: 'weather' } }
34
+ ]
35
+ });
36
+ });
37
+ ```
38
+
39
+ ## Guards
40
+
41
+ ### ToolGuard
42
+
43
+ Blocks dangerous tool calls.
44
+
45
+ ```typescript
46
+ const guard = new ToolGuard({
47
+ blockedTools: ['execute_shell', 'delete_file'],
48
+ allowedTools: ['search', 'calculator'], // Whitelist mode
49
+ });
50
+ ```
51
+
52
+ ### SafetyGuard
53
+
54
+ Detects PII and prompt injection.
55
+
56
+ ```typescript
57
+ const guard = new SafetyGuard({
58
+ checkPii: true,
59
+ checkInjection: true,
60
+ });
61
+ ```
62
+
63
+ ### SchemaGuard
64
+
65
+ Validates JSON structure.
66
+
67
+ ```typescript
68
+ const guard = new SchemaGuard({
69
+ type: 'object',
70
+ required: ['name', 'age'],
71
+ });
72
+ ```
73
+
74
+ ### MathGuard
75
+
76
+ Verifies calculations.
77
+
78
+ ```typescript
79
+ const guard = new MathGuard({ tolerance: 0.01 });
80
+ ```
81
+
82
+ ## Middleware Options
83
+
84
+ ```typescript
85
+ createQWEDMiddleware({
86
+ guards: [], // Guards to apply
87
+ blockOnFailure: true, // Block failed responses
88
+ verbose: false, // Log verification results
89
+ skipPaths: ['/health'], // Paths to skip
90
+ onError: (result, req, res) => {
91
+ // Custom error handler
92
+ },
93
+ });
94
+ ```
95
+
96
+ ## Verify Request Bodies
97
+
98
+ ```typescript
99
+ import { verifyRequestBody, ToolGuard } from 'qwed-open-responses';
100
+
101
+ app.post('/api/execute',
102
+ verifyRequestBody({ guards: [new ToolGuard()] }),
103
+ (req, res) => {
104
+ // Request body is verified
105
+ }
106
+ );
107
+ ```
108
+
109
+ ## Direct Verification
110
+
111
+ ```typescript
112
+ import { ResponseVerifier, ToolGuard } from 'qwed-open-responses';
113
+
114
+ const verifier = new ResponseVerifier([new ToolGuard()]);
115
+
116
+ const result = verifier.verify({
117
+ tool_calls: [{ name: 'search', arguments: {} }]
118
+ });
119
+
120
+ if (result.verified) {
121
+ console.log('✅ Safe to execute');
122
+ } else {
123
+ console.log('❌ Blocked:', result.blockReason);
124
+ }
125
+ ```
126
+
127
+ ## Links
128
+
129
+ - **GitHub:** [QWED-AI/qwed-open-responses](https://github.com/QWED-AI/qwed-open-responses)
130
+ - **PyPI (Python):** [qwed-open-responses](https://pypi.org/project/qwed-open-responses/)
131
+ - **Docs:** [docs.qwedai.com](https://docs.qwedai.com/docs/open-responses/overview)
132
+
133
+ ## License
134
+
135
+ Apache 2.0
@@ -0,0 +1,73 @@
1
+ /**
2
+ * QWED Open Responses - Guards
3
+ */
4
+ import { GuardResult, ParsedResponse } from './types';
5
+ /**
6
+ * Base class for all guards.
7
+ */
8
+ export declare abstract class BaseGuard {
9
+ abstract name: string;
10
+ abstract description: string;
11
+ abstract check(response: ParsedResponse, context?: Record<string, any>): GuardResult;
12
+ protected passResult(message?: string, details?: Record<string, any>): GuardResult;
13
+ protected failResult(message: string, details?: Record<string, any>, severity?: 'error' | 'warning'): GuardResult;
14
+ }
15
+ /**
16
+ * Tool Guard - Blocks dangerous tool calls.
17
+ */
18
+ export declare class ToolGuard extends BaseGuard {
19
+ name: string;
20
+ description: string;
21
+ private blockedTools;
22
+ private allowedTools;
23
+ private dangerousPatterns;
24
+ private static DEFAULT_BLOCKED_TOOLS;
25
+ private static DEFAULT_DANGEROUS_PATTERNS;
26
+ constructor(options?: {
27
+ blockedTools?: string[];
28
+ allowedTools?: string[];
29
+ useDefaultBlocklist?: boolean;
30
+ dangerousPatterns?: RegExp[];
31
+ });
32
+ check(response: ParsedResponse, context?: Record<string, any>): GuardResult;
33
+ private extractToolCalls;
34
+ }
35
+ /**
36
+ * Schema Guard - Validates JSON schema.
37
+ */
38
+ export declare class SchemaGuard extends BaseGuard {
39
+ name: string;
40
+ description: string;
41
+ private schema;
42
+ constructor(schema: Record<string, any>);
43
+ check(response: ParsedResponse, context?: Record<string, any>): GuardResult;
44
+ }
45
+ /**
46
+ * Safety Guard - Comprehensive safety checks.
47
+ */
48
+ export declare class SafetyGuard extends BaseGuard {
49
+ name: string;
50
+ description: string;
51
+ private checkPii;
52
+ private checkInjection;
53
+ private static PII_PATTERNS;
54
+ private static INJECTION_PATTERNS;
55
+ constructor(options?: {
56
+ checkPii?: boolean;
57
+ checkInjection?: boolean;
58
+ });
59
+ check(response: ParsedResponse, context?: Record<string, any>): GuardResult;
60
+ private extractContent;
61
+ }
62
+ /**
63
+ * Math Guard - Verifies calculations.
64
+ */
65
+ export declare class MathGuard extends BaseGuard {
66
+ name: string;
67
+ description: string;
68
+ private tolerance;
69
+ constructor(options?: {
70
+ tolerance?: number;
71
+ });
72
+ check(response: ParsedResponse, context?: Record<string, any>): GuardResult;
73
+ }
package/dist/guards.js ADDED
@@ -0,0 +1,228 @@
1
+ "use strict";
2
+ /**
3
+ * QWED Open Responses - Guards
4
+ */
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.MathGuard = exports.SafetyGuard = exports.SchemaGuard = exports.ToolGuard = exports.BaseGuard = void 0;
7
+ /**
8
+ * Base class for all guards.
9
+ */
10
+ class BaseGuard {
11
+ passResult(message, details) {
12
+ return {
13
+ guardName: this.name,
14
+ passed: true,
15
+ message: message || `${this.name} passed`,
16
+ details,
17
+ severity: 'info',
18
+ };
19
+ }
20
+ failResult(message, details, severity = 'error') {
21
+ return {
22
+ guardName: this.name,
23
+ passed: false,
24
+ message,
25
+ details,
26
+ severity,
27
+ };
28
+ }
29
+ }
30
+ exports.BaseGuard = BaseGuard;
31
+ /**
32
+ * Tool Guard - Blocks dangerous tool calls.
33
+ */
34
+ class ToolGuard extends BaseGuard {
35
+ constructor(options = {}) {
36
+ super();
37
+ this.name = 'ToolGuard';
38
+ this.description = 'Validates tool calls for safety';
39
+ const { blockedTools = [], allowedTools, useDefaultBlocklist = true, dangerousPatterns = [], } = options;
40
+ this.blockedTools = new Set(blockedTools);
41
+ if (useDefaultBlocklist) {
42
+ ToolGuard.DEFAULT_BLOCKED_TOOLS.forEach(t => this.blockedTools.add(t));
43
+ }
44
+ this.allowedTools = allowedTools ? new Set(allowedTools) : null;
45
+ this.dangerousPatterns = [
46
+ ...ToolGuard.DEFAULT_DANGEROUS_PATTERNS,
47
+ ...dangerousPatterns,
48
+ ];
49
+ }
50
+ check(response, context) {
51
+ const toolCalls = this.extractToolCalls(response);
52
+ if (toolCalls.length === 0) {
53
+ return this.passResult('No tool calls to verify');
54
+ }
55
+ for (const call of toolCalls) {
56
+ const toolName = call.toolName || call.tool_name || call.name || 'unknown';
57
+ const args = call.arguments || {};
58
+ // Check blocked list
59
+ if (this.blockedTools.has(toolName)) {
60
+ return this.failResult(`BLOCKED: Tool '${toolName}' is not allowed`, { blockedTool: toolName });
61
+ }
62
+ // Check allowed list (whitelist mode)
63
+ if (this.allowedTools && !this.allowedTools.has(toolName)) {
64
+ return this.failResult(`BLOCKED: Tool '${toolName}' is not in allowed list`, {
65
+ tool: toolName,
66
+ allowed: Array.from(this.allowedTools),
67
+ });
68
+ }
69
+ // Check dangerous patterns
70
+ const argsStr = JSON.stringify(args);
71
+ for (const pattern of this.dangerousPatterns) {
72
+ if (pattern.test(argsStr)) {
73
+ return this.failResult('BLOCKED: Dangerous pattern detected in tool arguments', {
74
+ tool: toolName,
75
+ pattern: pattern.source,
76
+ });
77
+ }
78
+ }
79
+ }
80
+ return this.passResult(`All ${toolCalls.length} tool call(s) verified`);
81
+ }
82
+ extractToolCalls(response) {
83
+ const calls = [];
84
+ if (response.type === 'tool_call') {
85
+ calls.push(response);
86
+ }
87
+ if (response.toolCalls || response.tool_calls) {
88
+ calls.push(...(response.toolCalls || response.tool_calls || []));
89
+ }
90
+ return calls;
91
+ }
92
+ }
93
+ exports.ToolGuard = ToolGuard;
94
+ ToolGuard.DEFAULT_BLOCKED_TOOLS = new Set([
95
+ 'execute_shell', 'shell', 'bash', 'cmd', 'exec', 'eval',
96
+ 'delete_file', 'remove_file', 'write_file', 'modify_file',
97
+ 'send_email', 'transfer_money', 'make_payment',
98
+ ]);
99
+ ToolGuard.DEFAULT_DANGEROUS_PATTERNS = [
100
+ /DROP\s+TABLE/i,
101
+ /DELETE\s+FROM/i,
102
+ /TRUNCATE\s+TABLE/i,
103
+ /rm\s+-rf/i,
104
+ /rmdir\s+\/s/i,
105
+ /eval\s*\(/i,
106
+ /exec\s*\(/i,
107
+ /__import__/i,
108
+ ];
109
+ /**
110
+ * Schema Guard - Validates JSON schema.
111
+ */
112
+ class SchemaGuard extends BaseGuard {
113
+ constructor(schema) {
114
+ super();
115
+ this.name = 'SchemaGuard';
116
+ this.description = 'Validates response against JSON Schema';
117
+ this.schema = schema;
118
+ }
119
+ check(response, context) {
120
+ const data = response.output || response;
121
+ // Basic type checking (full JSON Schema validation would need ajv)
122
+ if (this.schema.type === 'object' && typeof data !== 'object') {
123
+ return this.failResult('Expected object type');
124
+ }
125
+ if (this.schema.required) {
126
+ const missing = this.schema.required.filter((field) => !(field in data));
127
+ if (missing.length > 0) {
128
+ return this.failResult(`Missing required fields: ${missing.join(', ')}`, { missing });
129
+ }
130
+ }
131
+ return this.passResult('Schema validation passed');
132
+ }
133
+ }
134
+ exports.SchemaGuard = SchemaGuard;
135
+ /**
136
+ * Safety Guard - Comprehensive safety checks.
137
+ */
138
+ class SafetyGuard extends BaseGuard {
139
+ constructor(options = {}) {
140
+ super();
141
+ this.name = 'SafetyGuard';
142
+ this.description = 'Comprehensive safety checks';
143
+ this.checkPii = options.checkPii ?? true;
144
+ this.checkInjection = options.checkInjection ?? true;
145
+ }
146
+ check(response, context) {
147
+ const content = this.extractContent(response);
148
+ const issues = [];
149
+ if (this.checkPii) {
150
+ for (const [type, pattern] of Object.entries(SafetyGuard.PII_PATTERNS)) {
151
+ if (pattern.test(content)) {
152
+ issues.push(`PII detected: ${type}`);
153
+ }
154
+ }
155
+ }
156
+ if (this.checkInjection) {
157
+ for (const pattern of SafetyGuard.INJECTION_PATTERNS) {
158
+ if (pattern.test(content)) {
159
+ return this.failResult('BLOCKED: Prompt injection detected', { pattern: pattern.source });
160
+ }
161
+ }
162
+ }
163
+ if (issues.length > 0) {
164
+ return this.failResult(`Safety issues detected: ${issues.join(', ')}`, { issues }, 'warning');
165
+ }
166
+ return this.passResult('All safety checks passed');
167
+ }
168
+ extractContent(response) {
169
+ const parts = [];
170
+ if (typeof response.content === 'string')
171
+ parts.push(response.content);
172
+ if (typeof response.output === 'string')
173
+ parts.push(response.output);
174
+ if (typeof response.text === 'string')
175
+ parts.push(response.text);
176
+ if (typeof response.output === 'object')
177
+ parts.push(JSON.stringify(response.output));
178
+ if (response.arguments)
179
+ parts.push(JSON.stringify(response.arguments));
180
+ return parts.join(' ');
181
+ }
182
+ }
183
+ exports.SafetyGuard = SafetyGuard;
184
+ SafetyGuard.PII_PATTERNS = {
185
+ email: /[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}/,
186
+ phone: /\b\d{3}[-.]?\d{3}[-.]?\d{4}\b/,
187
+ ssn: /\b\d{3}-\d{2}-\d{4}\b/,
188
+ creditCard: /\b\d{4}[\s-]?\d{4}[\s-]?\d{4}[\s-]?\d{4}\b/,
189
+ };
190
+ SafetyGuard.INJECTION_PATTERNS = [
191
+ /ignore\s+(previous|all|above)\s+(instructions?|prompts?)/i,
192
+ /disregard\s+(previous|all|above)/i,
193
+ /forget\s+(everything|all|your\s+instructions)/i,
194
+ /you\s+are\s+now\s+/i,
195
+ /pretend\s+(you|to\s+be)/i,
196
+ ];
197
+ /**
198
+ * Math Guard - Verifies calculations.
199
+ */
200
+ class MathGuard extends BaseGuard {
201
+ constructor(options = {}) {
202
+ super();
203
+ this.name = 'MathGuard';
204
+ this.description = 'Verifies mathematical calculations';
205
+ this.tolerance = options.tolerance ?? 0.01;
206
+ }
207
+ check(response, context) {
208
+ const data = response.output || response;
209
+ if (typeof data !== 'object') {
210
+ return this.passResult('No calculations to verify');
211
+ }
212
+ // Check common total patterns
213
+ if ('total' in data && 'subtotal' in data) {
214
+ const subtotal = Number(data.subtotal) || 0;
215
+ const tax = Number(data.tax) || 0;
216
+ const shipping = Number(data.shipping) || 0;
217
+ const discount = Number(data.discount) || 0;
218
+ const total = Number(data.total);
219
+ const expected = subtotal + tax + shipping - discount;
220
+ if (Math.abs(expected - total) > this.tolerance) {
221
+ return this.failResult(`Total mismatch: expected ${expected}, got ${total}`, { expected, actual: total });
222
+ }
223
+ }
224
+ return this.passResult('Math verification passed');
225
+ }
226
+ }
227
+ exports.MathGuard = MathGuard;
228
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"guards.js","sourceRoot":"","sources":["../src/guards.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * QWED Open Responses - Guards\n */\n\nimport { GuardResult, ParsedResponse } from './types';\n\n/**\n * Base class for all guards.\n */\nexport abstract class BaseGuard {\n    abstract name: string;\n    abstract description: string;\n\n    abstract check(response: ParsedResponse, context?: Record<string, any>): GuardResult;\n\n    protected passResult(message?: string, details?: Record<string, any>): GuardResult {\n        return {\n            guardName: this.name,\n            passed: true,\n            message: message || `${this.name} passed`,\n            details,\n            severity: 'info',\n        };\n    }\n\n    protected failResult(message: string, details?: Record<string, any>, severity: 'error' | 'warning' = 'error'): GuardResult {\n        return {\n            guardName: this.name,\n            passed: false,\n            message,\n            details,\n            severity,\n        };\n    }\n}\n\n/**\n * Tool Guard - Blocks dangerous tool calls.\n */\nexport class ToolGuard extends BaseGuard {\n    name = 'ToolGuard';\n    description = 'Validates tool calls for safety';\n\n    private blockedTools: Set<string>;\n    private allowedTools: Set<string> | null;\n    private dangerousPatterns: RegExp[];\n\n    private static DEFAULT_BLOCKED_TOOLS = new Set([\n        'execute_shell', 'shell', 'bash', 'cmd', 'exec', 'eval',\n        'delete_file', 'remove_file', 'write_file', 'modify_file',\n        'send_email', 'transfer_money', 'make_payment',\n    ]);\n\n    private static DEFAULT_DANGEROUS_PATTERNS = [\n        /DROP\\s+TABLE/i,\n        /DELETE\\s+FROM/i,\n        /TRUNCATE\\s+TABLE/i,\n        /rm\\s+-rf/i,\n        /rmdir\\s+\\/s/i,\n        /eval\\s*\\(/i,\n        /exec\\s*\\(/i,\n        /__import__/i,\n    ];\n\n    constructor(options: {\n        blockedTools?: string[];\n        allowedTools?: string[];\n        useDefaultBlocklist?: boolean;\n        dangerousPatterns?: RegExp[];\n    } = {}) {\n        super();\n\n        const {\n            blockedTools = [],\n            allowedTools,\n            useDefaultBlocklist = true,\n            dangerousPatterns = [],\n        } = options;\n\n        this.blockedTools = new Set(blockedTools);\n        if (useDefaultBlocklist) {\n            ToolGuard.DEFAULT_BLOCKED_TOOLS.forEach(t => this.blockedTools.add(t));\n        }\n\n        this.allowedTools = allowedTools ? new Set(allowedTools) : null;\n        this.dangerousPatterns = [\n            ...ToolGuard.DEFAULT_DANGEROUS_PATTERNS,\n            ...dangerousPatterns,\n        ];\n    }\n\n    check(response: ParsedResponse, context?: Record<string, any>): GuardResult {\n        const toolCalls = this.extractToolCalls(response);\n\n        if (toolCalls.length === 0) {\n            return this.passResult('No tool calls to verify');\n        }\n\n        for (const call of toolCalls) {\n            const toolName = call.toolName || call.tool_name || call.name || 'unknown';\n            const args = call.arguments || {};\n\n            // Check blocked list\n            if (this.blockedTools.has(toolName)) {\n                return this.failResult(`BLOCKED: Tool '${toolName}' is not allowed`, { blockedTool: toolName });\n            }\n\n            // Check allowed list (whitelist mode)\n            if (this.allowedTools && !this.allowedTools.has(toolName)) {\n                return this.failResult(`BLOCKED: Tool '${toolName}' is not in allowed list`, {\n                    tool: toolName,\n                    allowed: Array.from(this.allowedTools),\n                });\n            }\n\n            // Check dangerous patterns\n            const argsStr = JSON.stringify(args);\n            for (const pattern of this.dangerousPatterns) {\n                if (pattern.test(argsStr)) {\n                    return this.failResult('BLOCKED: Dangerous pattern detected in tool arguments', {\n                        tool: toolName,\n                        pattern: pattern.source,\n                    });\n                }\n            }\n        }\n\n        return this.passResult(`All ${toolCalls.length} tool call(s) verified`);\n    }\n\n    private extractToolCalls(response: ParsedResponse): any[] {\n        const calls: any[] = [];\n\n        if (response.type === 'tool_call') {\n            calls.push(response);\n        }\n\n        if (response.toolCalls || response.tool_calls) {\n            calls.push(...(response.toolCalls || response.tool_calls || []));\n        }\n\n        return calls;\n    }\n}\n\n/**\n * Schema Guard - Validates JSON schema.\n */\nexport class SchemaGuard extends BaseGuard {\n    name = 'SchemaGuard';\n    description = 'Validates response against JSON Schema';\n\n    private schema: Record<string, any>;\n\n    constructor(schema: Record<string, any>) {\n        super();\n        this.schema = schema;\n    }\n\n    check(response: ParsedResponse, context?: Record<string, any>): GuardResult {\n        const data = response.output || response;\n\n        // Basic type checking (full JSON Schema validation would need ajv)\n        if (this.schema.type === 'object' && typeof data !== 'object') {\n            return this.failResult('Expected object type');\n        }\n\n        if (this.schema.required) {\n            const missing = this.schema.required.filter((field: string) => !(field in data));\n            if (missing.length > 0) {\n                return this.failResult(`Missing required fields: ${missing.join(', ')}`, { missing });\n            }\n        }\n\n        return this.passResult('Schema validation passed');\n    }\n}\n\n/**\n * Safety Guard - Comprehensive safety checks.\n */\nexport class SafetyGuard extends BaseGuard {\n    name = 'SafetyGuard';\n    description = 'Comprehensive safety checks';\n\n    private checkPii: boolean;\n    private checkInjection: boolean;\n\n    private static PII_PATTERNS = {\n        email: /[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\\.[a-zA-Z]{2,}/,\n        phone: /\\b\\d{3}[-.]?\\d{3}[-.]?\\d{4}\\b/,\n        ssn: /\\b\\d{3}-\\d{2}-\\d{4}\\b/,\n        creditCard: /\\b\\d{4}[\\s-]?\\d{4}[\\s-]?\\d{4}[\\s-]?\\d{4}\\b/,\n    };\n\n    private static INJECTION_PATTERNS = [\n        /ignore\\s+(previous|all|above)\\s+(instructions?|prompts?)/i,\n        /disregard\\s+(previous|all|above)/i,\n        /forget\\s+(everything|all|your\\s+instructions)/i,\n        /you\\s+are\\s+now\\s+/i,\n        /pretend\\s+(you|to\\s+be)/i,\n    ];\n\n    constructor(options: { checkPii?: boolean; checkInjection?: boolean } = {}) {\n        super();\n        this.checkPii = options.checkPii ?? true;\n        this.checkInjection = options.checkInjection ?? true;\n    }\n\n    check(response: ParsedResponse, context?: Record<string, any>): GuardResult {\n        const content = this.extractContent(response);\n        const issues: string[] = [];\n\n        if (this.checkPii) {\n            for (const [type, pattern] of Object.entries(SafetyGuard.PII_PATTERNS)) {\n                if (pattern.test(content)) {\n                    issues.push(`PII detected: ${type}`);\n                }\n            }\n        }\n\n        if (this.checkInjection) {\n            for (const pattern of SafetyGuard.INJECTION_PATTERNS) {\n                if (pattern.test(content)) {\n                    return this.failResult('BLOCKED: Prompt injection detected', { pattern: pattern.source });\n                }\n            }\n        }\n\n        if (issues.length > 0) {\n            return this.failResult(`Safety issues detected: ${issues.join(', ')}`, { issues }, 'warning');\n        }\n\n        return this.passResult('All safety checks passed');\n    }\n\n    private extractContent(response: ParsedResponse): string {\n        const parts: string[] = [];\n\n        if (typeof response.content === 'string') parts.push(response.content);\n        if (typeof response.output === 'string') parts.push(response.output);\n        if (typeof response.text === 'string') parts.push(response.text);\n\n        if (typeof response.output === 'object') parts.push(JSON.stringify(response.output));\n        if (response.arguments) parts.push(JSON.stringify(response.arguments));\n\n        return parts.join(' ');\n    }\n}\n\n/**\n * Math Guard - Verifies calculations.\n */\nexport class MathGuard extends BaseGuard {\n    name = 'MathGuard';\n    description = 'Verifies mathematical calculations';\n\n    private tolerance: number;\n\n    constructor(options: { tolerance?: number } = {}) {\n        super();\n        this.tolerance = options.tolerance ?? 0.01;\n    }\n\n    check(response: ParsedResponse, context?: Record<string, any>): GuardResult {\n        const data = response.output || response;\n\n        if (typeof data !== 'object') {\n            return this.passResult('No calculations to verify');\n        }\n\n        // Check common total patterns\n        if ('total' in data && 'subtotal' in data) {\n            const subtotal = Number(data.subtotal) || 0;\n            const tax = Number(data.tax) || 0;\n            const shipping = Number(data.shipping) || 0;\n            const discount = Number(data.discount) || 0;\n            const total = Number(data.total);\n\n            const expected = subtotal + tax + shipping - discount;\n\n            if (Math.abs(expected - total) > this.tolerance) {\n                return this.failResult(\n                    `Total mismatch: expected ${expected}, got ${total}`,\n                    { expected, actual: total }\n                );\n            }\n        }\n\n        return this.passResult('Math verification passed');\n    }\n}\n"]}
@@ -0,0 +1,58 @@
1
+ /**
2
+ * QWED Open Responses - Express Middleware
3
+ *
4
+ * Verification guards for AI agent outputs in Express.js applications.
5
+ */
6
+ import { Request, Response, RequestHandler } from 'express';
7
+ export * from './guards';
8
+ export * from './types';
9
+ export * from './verifier';
10
+ import { ResponseVerifier, VerificationResult } from './verifier';
11
+ import { BaseGuard, ToolGuard, SchemaGuard, SafetyGuard } from './guards';
12
+ export interface QWEDMiddlewareOptions {
13
+ /** Guards to apply to all requests */
14
+ guards?: BaseGuard[];
15
+ /** Block requests that fail verification */
16
+ blockOnFailure?: boolean;
17
+ /** Custom error handler */
18
+ onError?: (error: VerificationResult, req: Request, res: Response) => void;
19
+ /** Log verification results */
20
+ verbose?: boolean;
21
+ /** Paths to skip verification */
22
+ skipPaths?: string[];
23
+ }
24
+ /**
25
+ * Create QWED verification middleware for Express.
26
+ *
27
+ * @example
28
+ * ```typescript
29
+ * import express from 'express';
30
+ * import { createQWEDMiddleware, ToolGuard, SafetyGuard } from 'qwed-open-responses';
31
+ *
32
+ * const app = express();
33
+ *
34
+ * app.use(createQWEDMiddleware({
35
+ * guards: [new ToolGuard(), new SafetyGuard()],
36
+ * blockOnFailure: true,
37
+ * }));
38
+ * ```
39
+ */
40
+ export declare function createQWEDMiddleware(options?: QWEDMiddlewareOptions): RequestHandler;
41
+ /**
42
+ * Middleware to verify incoming request bodies.
43
+ */
44
+ export declare function verifyRequestBody(options?: QWEDMiddlewareOptions): RequestHandler;
45
+ /**
46
+ * Middleware to verify tool calls in AI agent requests.
47
+ */
48
+ export declare function verifyToolCalls(options?: QWEDMiddlewareOptions): RequestHandler;
49
+ declare const _default: {
50
+ createQWEDMiddleware: typeof createQWEDMiddleware;
51
+ verifyRequestBody: typeof verifyRequestBody;
52
+ verifyToolCalls: typeof verifyToolCalls;
53
+ ResponseVerifier: typeof ResponseVerifier;
54
+ ToolGuard: typeof ToolGuard;
55
+ SchemaGuard: typeof SchemaGuard;
56
+ SafetyGuard: typeof SafetyGuard;
57
+ };
58
+ export default _default;
package/dist/index.js ADDED
@@ -0,0 +1,130 @@
1
+ "use strict";
2
+ /**
3
+ * QWED Open Responses - Express Middleware
4
+ *
5
+ * Verification guards for AI agent outputs in Express.js applications.
6
+ */
7
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
8
+ if (k2 === undefined) k2 = k;
9
+ var desc = Object.getOwnPropertyDescriptor(m, k);
10
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
11
+ desc = { enumerable: true, get: function() { return m[k]; } };
12
+ }
13
+ Object.defineProperty(o, k2, desc);
14
+ }) : (function(o, m, k, k2) {
15
+ if (k2 === undefined) k2 = k;
16
+ o[k2] = m[k];
17
+ }));
18
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
19
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
20
+ };
21
+ Object.defineProperty(exports, "__esModule", { value: true });
22
+ exports.createQWEDMiddleware = createQWEDMiddleware;
23
+ exports.verifyRequestBody = verifyRequestBody;
24
+ exports.verifyToolCalls = verifyToolCalls;
25
+ // Re-export all guards and types
26
+ __exportStar(require("./guards"), exports);
27
+ __exportStar(require("./types"), exports);
28
+ __exportStar(require("./verifier"), exports);
29
+ const verifier_1 = require("./verifier");
30
+ const guards_1 = require("./guards");
31
+ /**
32
+ * Create QWED verification middleware for Express.
33
+ *
34
+ * @example
35
+ * ```typescript
36
+ * import express from 'express';
37
+ * import { createQWEDMiddleware, ToolGuard, SafetyGuard } from 'qwed-open-responses';
38
+ *
39
+ * const app = express();
40
+ *
41
+ * app.use(createQWEDMiddleware({
42
+ * guards: [new ToolGuard(), new SafetyGuard()],
43
+ * blockOnFailure: true,
44
+ * }));
45
+ * ```
46
+ */
47
+ function createQWEDMiddleware(options = {}) {
48
+ const { guards = [], blockOnFailure = true, onError, verbose = false, skipPaths = [], } = options;
49
+ const verifier = new verifier_1.ResponseVerifier(guards);
50
+ return (req, res, next) => {
51
+ // Skip certain paths
52
+ if (skipPaths.some(path => req.path.startsWith(path))) {
53
+ return next();
54
+ }
55
+ // Store original json method
56
+ const originalJson = res.json.bind(res);
57
+ // Override json to verify before sending
58
+ res.json = (body) => {
59
+ const result = verifier.verify(body);
60
+ if (verbose) {
61
+ console.log(`[QWED] ${req.method} ${req.path} -> ${result.verified ? 'PASS' : 'FAIL'}`);
62
+ }
63
+ // Attach verification result to response
64
+ res._qwedVerification = result;
65
+ if (!result.verified && blockOnFailure) {
66
+ if (onError) {
67
+ onError(result, req, res);
68
+ return res;
69
+ }
70
+ return originalJson({
71
+ error: 'Response verification failed',
72
+ code: 'QWED_VERIFICATION_FAILED',
73
+ details: result.guardResults.filter(g => !g.passed).map(g => ({
74
+ guard: g.guardName,
75
+ message: g.message,
76
+ })),
77
+ });
78
+ }
79
+ return originalJson(body);
80
+ };
81
+ next();
82
+ };
83
+ }
84
+ /**
85
+ * Middleware to verify incoming request bodies.
86
+ */
87
+ function verifyRequestBody(options = {}) {
88
+ const { guards = [], blockOnFailure = true, verbose = false, } = options;
89
+ const verifier = new verifier_1.ResponseVerifier(guards);
90
+ return (req, res, next) => {
91
+ if (!req.body) {
92
+ return next();
93
+ }
94
+ const result = verifier.verify(req.body);
95
+ if (verbose) {
96
+ console.log(`[QWED] Request body ${req.method} ${req.path} -> ${result.verified ? 'PASS' : 'FAIL'}`);
97
+ }
98
+ req._qwedVerification = result;
99
+ if (!result.verified && blockOnFailure) {
100
+ return res.status(422).json({
101
+ error: 'Request verification failed',
102
+ code: 'QWED_REQUEST_BLOCKED',
103
+ details: result.guardResults.filter(g => !g.passed).map(g => ({
104
+ guard: g.guardName,
105
+ message: g.message,
106
+ })),
107
+ });
108
+ }
109
+ next();
110
+ };
111
+ }
112
+ /**
113
+ * Middleware to verify tool calls in AI agent requests.
114
+ */
115
+ function verifyToolCalls(options = {}) {
116
+ const toolGuard = new guards_1.ToolGuard();
117
+ const guards = [toolGuard, ...(options.guards || [])];
118
+ return verifyRequestBody({ ...options, guards });
119
+ }
120
+ // Default export
121
+ exports.default = {
122
+ createQWEDMiddleware,
123
+ verifyRequestBody,
124
+ verifyToolCalls,
125
+ ResponseVerifier: verifier_1.ResponseVerifier,
126
+ ToolGuard: guards_1.ToolGuard,
127
+ SchemaGuard: guards_1.SchemaGuard,
128
+ SafetyGuard: guards_1.SafetyGuard,
129
+ };
130
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * QWED Open Responses - Express Middleware\n * \n * Verification guards for AI agent outputs in Express.js applications.\n */\n\nimport { Request, Response, NextFunction, RequestHandler } from 'express';\n\n// Re-export all guards and types\nexport * from './guards';\nexport * from './types';\nexport * from './verifier';\n\nimport { ResponseVerifier, VerificationResult } from './verifier';\nimport { BaseGuard, ToolGuard, SchemaGuard, SafetyGuard } from './guards';\n\nexport interface QWEDMiddlewareOptions {\n    /** Guards to apply to all requests */\n    guards?: BaseGuard[];\n    /** Block requests that fail verification */\n    blockOnFailure?: boolean;\n    /** Custom error handler */\n    onError?: (error: VerificationResult, req: Request, res: Response) => void;\n    /** Log verification results */\n    verbose?: boolean;\n    /** Paths to skip verification */\n    skipPaths?: string[];\n}\n\n/**\n * Create QWED verification middleware for Express.\n * \n * @example\n * ```typescript\n * import express from 'express';\n * import { createQWEDMiddleware, ToolGuard, SafetyGuard } from 'qwed-open-responses';\n * \n * const app = express();\n * \n * app.use(createQWEDMiddleware({\n *   guards: [new ToolGuard(), new SafetyGuard()],\n *   blockOnFailure: true,\n * }));\n * ```\n */\nexport function createQWEDMiddleware(options: QWEDMiddlewareOptions = {}): RequestHandler {\n    const {\n        guards = [],\n        blockOnFailure = true,\n        onError,\n        verbose = false,\n        skipPaths = [],\n    } = options;\n\n    const verifier = new ResponseVerifier(guards);\n\n    return (req: Request, res: Response, next: NextFunction) => {\n        // Skip certain paths\n        if (skipPaths.some(path => req.path.startsWith(path))) {\n            return next();\n        }\n\n        // Store original json method\n        const originalJson = res.json.bind(res);\n\n        // Override json to verify before sending\n        res.json = (body: any) => {\n            const result = verifier.verify(body);\n\n            if (verbose) {\n                console.log(`[QWED] ${req.method} ${req.path} -> ${result.verified ? 'PASS' : 'FAIL'}`);\n            }\n\n            // Attach verification result to response\n            (res as any)._qwedVerification = result;\n\n            if (!result.verified && blockOnFailure) {\n                if (onError) {\n                    onError(result, req, res);\n                    return res;\n                }\n\n                return originalJson({\n                    error: 'Response verification failed',\n                    code: 'QWED_VERIFICATION_FAILED',\n                    details: result.guardResults.filter(g => !g.passed).map(g => ({\n                        guard: g.guardName,\n                        message: g.message,\n                    })),\n                });\n            }\n\n            return originalJson(body);\n        };\n\n        next();\n    };\n}\n\n/**\n * Middleware to verify incoming request bodies.\n */\nexport function verifyRequestBody(options: QWEDMiddlewareOptions = {}): RequestHandler {\n    const {\n        guards = [],\n        blockOnFailure = true,\n        verbose = false,\n    } = options;\n\n    const verifier = new ResponseVerifier(guards);\n\n    return (req: Request, res: Response, next: NextFunction) => {\n        if (!req.body) {\n            return next();\n        }\n\n        const result = verifier.verify(req.body);\n\n        if (verbose) {\n            console.log(`[QWED] Request body ${req.method} ${req.path} -> ${result.verified ? 'PASS' : 'FAIL'}`);\n        }\n\n        (req as any)._qwedVerification = result;\n\n        if (!result.verified && blockOnFailure) {\n            return res.status(422).json({\n                error: 'Request verification failed',\n                code: 'QWED_REQUEST_BLOCKED',\n                details: result.guardResults.filter(g => !g.passed).map(g => ({\n                    guard: g.guardName,\n                    message: g.message,\n                })),\n            });\n        }\n\n        next();\n    };\n}\n\n/**\n * Middleware to verify tool calls in AI agent requests.\n */\nexport function verifyToolCalls(options: QWEDMiddlewareOptions = {}): RequestHandler {\n    const toolGuard = new ToolGuard();\n    const guards = [toolGuard, ...(options.guards || [])];\n\n    return verifyRequestBody({ ...options, guards });\n}\n\n// Default export\nexport default {\n    createQWEDMiddleware,\n    verifyRequestBody,\n    verifyToolCalls,\n    ResponseVerifier,\n    ToolGuard,\n    SchemaGuard,\n    SafetyGuard,\n};\n"]}
@@ -0,0 +1,35 @@
1
+ /**
2
+ * QWED Open Responses - Types
3
+ */
4
+ export interface GuardResult {
5
+ guardName: string;
6
+ passed: boolean;
7
+ message?: string;
8
+ details?: Record<string, any>;
9
+ severity: 'error' | 'warning' | 'info';
10
+ }
11
+ export interface VerificationResult {
12
+ verified: boolean;
13
+ response: any;
14
+ guardsPassed: number;
15
+ guardsFailed: number;
16
+ guardResults: GuardResult[];
17
+ blocked: boolean;
18
+ blockReason?: string;
19
+ timestamp: string;
20
+ }
21
+ export interface ToolCall {
22
+ type?: string;
23
+ toolName?: string;
24
+ tool_name?: string;
25
+ name?: string;
26
+ arguments?: Record<string, any>;
27
+ }
28
+ export interface ParsedResponse {
29
+ type?: string;
30
+ content?: string;
31
+ output?: any;
32
+ toolCalls?: ToolCall[];
33
+ tool_calls?: ToolCall[];
34
+ [key: string]: any;
35
+ }
package/dist/types.js ADDED
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ /**
3
+ * QWED Open Responses - Types
4
+ */
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOztHQUVHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBRV0VEIE9wZW4gUmVzcG9uc2VzIC0gVHlwZXNcbiAqL1xuXG5leHBvcnQgaW50ZXJmYWNlIEd1YXJkUmVzdWx0IHtcbiAgICBndWFyZE5hbWU6IHN0cmluZztcbiAgICBwYXNzZWQ6IGJvb2xlYW47XG4gICAgbWVzc2FnZT86IHN0cmluZztcbiAgICBkZXRhaWxzPzogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgICBzZXZlcml0eTogJ2Vycm9yJyB8ICd3YXJuaW5nJyB8ICdpbmZvJztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBWZXJpZmljYXRpb25SZXN1bHQge1xuICAgIHZlcmlmaWVkOiBib29sZWFuO1xuICAgIHJlc3BvbnNlOiBhbnk7XG4gICAgZ3VhcmRzUGFzc2VkOiBudW1iZXI7XG4gICAgZ3VhcmRzRmFpbGVkOiBudW1iZXI7XG4gICAgZ3VhcmRSZXN1bHRzOiBHdWFyZFJlc3VsdFtdO1xuICAgIGJsb2NrZWQ6IGJvb2xlYW47XG4gICAgYmxvY2tSZWFzb24/OiBzdHJpbmc7XG4gICAgdGltZXN0YW1wOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgVG9vbENhbGwge1xuICAgIHR5cGU/OiBzdHJpbmc7XG4gICAgdG9vbE5hbWU/OiBzdHJpbmc7XG4gICAgdG9vbF9uYW1lPzogc3RyaW5nO1xuICAgIG5hbWU/OiBzdHJpbmc7XG4gICAgYXJndW1lbnRzPzogUmVjb3JkPHN0cmluZywgYW55Pjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQYXJzZWRSZXNwb25zZSB7XG4gICAgdHlwZT86IHN0cmluZztcbiAgICBjb250ZW50Pzogc3RyaW5nO1xuICAgIG91dHB1dD86IGFueTtcbiAgICB0b29sQ2FsbHM/OiBUb29sQ2FsbFtdO1xuICAgIHRvb2xfY2FsbHM/OiBUb29sQ2FsbFtdO1xuICAgIFtrZXk6IHN0cmluZ106IGFueTtcbn1cbiJdfQ==
@@ -0,0 +1,25 @@
1
+ /**
2
+ * QWED Open Responses - Response Verifier
3
+ */
4
+ import { BaseGuard } from './guards';
5
+ import { VerificationResult, GuardResult } from './types';
6
+ export { VerificationResult, GuardResult };
7
+ /**
8
+ * Main verifier for AI responses.
9
+ */
10
+ export declare class ResponseVerifier {
11
+ private defaultGuards;
12
+ private strictMode;
13
+ constructor(guards?: BaseGuard[], options?: {
14
+ strictMode?: boolean;
15
+ });
16
+ /**
17
+ * Verify a response against guards.
18
+ */
19
+ verify(response: any, guards?: BaseGuard[], context?: Record<string, any>): VerificationResult;
20
+ /**
21
+ * Verify a tool call.
22
+ */
23
+ verifyToolCall(toolName: string, args: Record<string, any>, guards?: BaseGuard[]): VerificationResult;
24
+ private parseResponse;
25
+ }
@@ -0,0 +1,89 @@
1
+ "use strict";
2
+ /**
3
+ * QWED Open Responses - Response Verifier
4
+ */
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.ResponseVerifier = void 0;
7
+ /**
8
+ * Main verifier for AI responses.
9
+ */
10
+ class ResponseVerifier {
11
+ constructor(guards = [], options = {}) {
12
+ this.defaultGuards = guards;
13
+ this.strictMode = options.strictMode ?? true;
14
+ }
15
+ /**
16
+ * Verify a response against guards.
17
+ */
18
+ verify(response, guards, context) {
19
+ const guardsToUse = guards ?? this.defaultGuards;
20
+ const parsedResponse = this.parseResponse(response);
21
+ const guardResults = [];
22
+ let guardsPassed = 0;
23
+ let guardsFailed = 0;
24
+ let blocked = false;
25
+ let blockReason;
26
+ for (const guard of guardsToUse) {
27
+ try {
28
+ const result = guard.check(parsedResponse, context);
29
+ guardResults.push(result);
30
+ if (result.passed) {
31
+ guardsPassed++;
32
+ }
33
+ else {
34
+ guardsFailed++;
35
+ if (result.severity === 'error' && this.strictMode) {
36
+ blocked = true;
37
+ blockReason = result.message;
38
+ }
39
+ }
40
+ }
41
+ catch (error) {
42
+ guardResults.push({
43
+ guardName: guard.name,
44
+ passed: false,
45
+ message: `Guard error: ${error instanceof Error ? error.message : String(error)}`,
46
+ severity: 'error',
47
+ });
48
+ guardsFailed++;
49
+ }
50
+ }
51
+ return {
52
+ verified: guardsFailed === 0,
53
+ response: parsedResponse,
54
+ guardsPassed,
55
+ guardsFailed,
56
+ guardResults,
57
+ blocked,
58
+ blockReason,
59
+ timestamp: new Date().toISOString(),
60
+ };
61
+ }
62
+ /**
63
+ * Verify a tool call.
64
+ */
65
+ verifyToolCall(toolName, args, guards) {
66
+ const toolCall = {
67
+ type: 'tool_call',
68
+ toolName,
69
+ arguments: args,
70
+ };
71
+ return this.verify(toolCall, guards);
72
+ }
73
+ parseResponse(response) {
74
+ if (typeof response === 'object' && response !== null) {
75
+ return response;
76
+ }
77
+ if (typeof response === 'string') {
78
+ try {
79
+ return JSON.parse(response);
80
+ }
81
+ catch {
82
+ return { type: 'text', content: response };
83
+ }
84
+ }
85
+ return { type: 'unknown', raw: String(response) };
86
+ }
87
+ }
88
+ exports.ResponseVerifier = ResponseVerifier;
89
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyaWZpZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdmVyaWZpZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOztHQUVHOzs7QUFRSDs7R0FFRztBQUNILE1BQWEsZ0JBQWdCO0lBSXpCLFlBQVksU0FBc0IsRUFBRSxFQUFFLFVBQW9DLEVBQUU7UUFDeEUsSUFBSSxDQUFDLGFBQWEsR0FBRyxNQUFNLENBQUM7UUFDNUIsSUFBSSxDQUFDLFVBQVUsR0FBRyxPQUFPLENBQUMsVUFBVSxJQUFJLElBQUksQ0FBQztJQUNqRCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxNQUFNLENBQ0YsUUFBYSxFQUNiLE1BQW9CLEVBQ3BCLE9BQTZCO1FBRTdCLE1BQU0sV0FBVyxHQUFHLE1BQU0sSUFBSSxJQUFJLENBQUMsYUFBYSxDQUFDO1FBQ2pELE1BQU0sY0FBYyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsUUFBUSxDQUFDLENBQUM7UUFFcEQsTUFBTSxZQUFZLEdBQWtCLEVBQUUsQ0FBQztRQUN2QyxJQUFJLFlBQVksR0FBRyxDQUFDLENBQUM7UUFDckIsSUFBSSxZQUFZLEdBQUcsQ0FBQyxDQUFDO1FBQ3JCLElBQUksT0FBTyxHQUFHLEtBQUssQ0FBQztRQUNwQixJQUFJLFdBQStCLENBQUM7UUFFcEMsS0FBSyxNQUFNLEtBQUssSUFBSSxXQUFXLEVBQUUsQ0FBQztZQUM5QixJQUFJLENBQUM7Z0JBQ0QsTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQyxjQUFjLEVBQUUsT0FBTyxDQUFDLENBQUM7Z0JBQ3BELFlBQVksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7Z0JBRTFCLElBQUksTUFBTSxDQUFDLE1BQU0sRUFBRSxDQUFDO29CQUNoQixZQUFZLEVBQUUsQ0FBQztnQkFDbkIsQ0FBQztxQkFBTSxDQUFDO29CQUNKLFlBQVksRUFBRSxDQUFDO29CQUNmLElBQUksTUFBTSxDQUFDLFFBQVEsS0FBSyxPQUFPLElBQUksSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO3dCQUNqRCxPQUFPLEdBQUcsSUFBSSxDQUFDO3dCQUNmLFdBQVcsR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDO29CQUNqQyxDQUFDO2dCQUNMLENBQUM7WUFDTCxDQUFDO1lBQUMsT0FBTyxLQUFLLEVBQUUsQ0FBQztnQkFDYixZQUFZLENBQUMsSUFBSSxDQUFDO29CQUNkLFNBQVMsRUFBRSxLQUFLLENBQUMsSUFBSTtvQkFDckIsTUFBTSxFQUFFLEtBQUs7b0JBQ2IsT0FBTyxFQUFFLGdCQUFnQixLQUFLLFlBQVksS0FBSyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUU7b0JBQ2pGLFFBQVEsRUFBRSxPQUFPO2lCQUNwQixDQUFDLENBQUM7Z0JBQ0gsWUFBWSxFQUFFLENBQUM7WUFDbkIsQ0FBQztRQUNMLENBQUM7UUFFRCxPQUFPO1lBQ0gsUUFBUSxFQUFFLFlBQVksS0FBSyxDQUFDO1lBQzVCLFFBQVEsRUFBRSxjQUFjO1lBQ3hCLFlBQVk7WUFDWixZQUFZO1lBQ1osWUFBWTtZQUNaLE9BQU87WUFDUCxXQUFXO1lBQ1gsU0FBUyxFQUFFLElBQUksSUFBSSxFQUFFLENBQUMsV0FBVyxFQUFFO1NBQ3RDLENBQUM7SUFDTixDQUFDO0lBRUQ7O09BRUc7SUFDSCxjQUFjLENBQ1YsUUFBZ0IsRUFDaEIsSUFBeUIsRUFDekIsTUFBb0I7UUFFcEIsTUFBTSxRQUFRLEdBQUc7WUFDYixJQUFJLEVBQUUsV0FBVztZQUNqQixRQUFRO1lBQ1IsU0FBUyxFQUFFLElBQUk7U0FDbEIsQ0FBQztRQUNGLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDekMsQ0FBQztJQUVPLGFBQWEsQ0FBQyxRQUFhO1FBQy9CLElBQUksT0FBTyxRQUFRLEtBQUssUUFBUSxJQUFJLFFBQVEsS0FBSyxJQUFJLEVBQUUsQ0FBQztZQUNwRCxPQUFPLFFBQVEsQ0FBQztRQUNwQixDQUFDO1FBRUQsSUFBSSxPQUFPLFFBQVEsS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUMvQixJQUFJLENBQUM7Z0JBQ0QsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ2hDLENBQUM7WUFBQyxNQUFNLENBQUM7Z0JBQ0wsT0FBTyxFQUFFLElBQUksRUFBRSxNQUFNLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxDQUFDO1lBQy9DLENBQUM7UUFDTCxDQUFDO1FBRUQsT0FBTyxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsR0FBRyxFQUFFLE1BQU0sQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDO0lBQ3RELENBQUM7Q0FDSjtBQTlGRCw0Q0E4RkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFFXRUQgT3BlbiBSZXNwb25zZXMgLSBSZXNwb25zZSBWZXJpZmllclxuICovXG5cbmltcG9ydCB7IEJhc2VHdWFyZCB9IGZyb20gJy4vZ3VhcmRzJztcbmltcG9ydCB7IFZlcmlmaWNhdGlvblJlc3VsdCwgR3VhcmRSZXN1bHQsIFBhcnNlZFJlc3BvbnNlIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8vIFJlLWV4cG9ydCB0eXBlc1xuZXhwb3J0IHsgVmVyaWZpY2F0aW9uUmVzdWx0LCBHdWFyZFJlc3VsdCB9O1xuXG4vKipcbiAqIE1haW4gdmVyaWZpZXIgZm9yIEFJIHJlc3BvbnNlcy5cbiAqL1xuZXhwb3J0IGNsYXNzIFJlc3BvbnNlVmVyaWZpZXIge1xuICAgIHByaXZhdGUgZGVmYXVsdEd1YXJkczogQmFzZUd1YXJkW107XG4gICAgcHJpdmF0ZSBzdHJpY3RNb2RlOiBib29sZWFuO1xuXG4gICAgY29uc3RydWN0b3IoZ3VhcmRzOiBCYXNlR3VhcmRbXSA9IFtdLCBvcHRpb25zOiB7IHN0cmljdE1vZGU/OiBib29sZWFuIH0gPSB7fSkge1xuICAgICAgICB0aGlzLmRlZmF1bHRHdWFyZHMgPSBndWFyZHM7XG4gICAgICAgIHRoaXMuc3RyaWN0TW9kZSA9IG9wdGlvbnMuc3RyaWN0TW9kZSA/PyB0cnVlO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFZlcmlmeSBhIHJlc3BvbnNlIGFnYWluc3QgZ3VhcmRzLlxuICAgICAqL1xuICAgIHZlcmlmeShcbiAgICAgICAgcmVzcG9uc2U6IGFueSxcbiAgICAgICAgZ3VhcmRzPzogQmFzZUd1YXJkW10sXG4gICAgICAgIGNvbnRleHQ/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+XG4gICAgKTogVmVyaWZpY2F0aW9uUmVzdWx0IHtcbiAgICAgICAgY29uc3QgZ3VhcmRzVG9Vc2UgPSBndWFyZHMgPz8gdGhpcy5kZWZhdWx0R3VhcmRzO1xuICAgICAgICBjb25zdCBwYXJzZWRSZXNwb25zZSA9IHRoaXMucGFyc2VSZXNwb25zZShyZXNwb25zZSk7XG5cbiAgICAgICAgY29uc3QgZ3VhcmRSZXN1bHRzOiBHdWFyZFJlc3VsdFtdID0gW107XG4gICAgICAgIGxldCBndWFyZHNQYXNzZWQgPSAwO1xuICAgICAgICBsZXQgZ3VhcmRzRmFpbGVkID0gMDtcbiAgICAgICAgbGV0IGJsb2NrZWQgPSBmYWxzZTtcbiAgICAgICAgbGV0IGJsb2NrUmVhc29uOiBzdHJpbmcgfCB1bmRlZmluZWQ7XG5cbiAgICAgICAgZm9yIChjb25zdCBndWFyZCBvZiBndWFyZHNUb1VzZSkge1xuICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgICBjb25zdCByZXN1bHQgPSBndWFyZC5jaGVjayhwYXJzZWRSZXNwb25zZSwgY29udGV4dCk7XG4gICAgICAgICAgICAgICAgZ3VhcmRSZXN1bHRzLnB1c2gocmVzdWx0KTtcblxuICAgICAgICAgICAgICAgIGlmIChyZXN1bHQucGFzc2VkKSB7XG4gICAgICAgICAgICAgICAgICAgIGd1YXJkc1Bhc3NlZCsrO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGd1YXJkc0ZhaWxlZCsrO1xuICAgICAgICAgICAgICAgICAgICBpZiAocmVzdWx0LnNldmVyaXR5ID09PSAnZXJyb3InICYmIHRoaXMuc3RyaWN0TW9kZSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgYmxvY2tlZCA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgICAgICBibG9ja1JlYXNvbiA9IHJlc3VsdC5tZXNzYWdlO1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAgICAgICBndWFyZFJlc3VsdHMucHVzaCh7XG4gICAgICAgICAgICAgICAgICAgIGd1YXJkTmFtZTogZ3VhcmQubmFtZSxcbiAgICAgICAgICAgICAgICAgICAgcGFzc2VkOiBmYWxzZSxcbiAgICAgICAgICAgICAgICAgICAgbWVzc2FnZTogYEd1YXJkIGVycm9yOiAke2Vycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogU3RyaW5nKGVycm9yKX1gLFxuICAgICAgICAgICAgICAgICAgICBzZXZlcml0eTogJ2Vycm9yJyxcbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICBndWFyZHNGYWlsZWQrKztcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICB2ZXJpZmllZDogZ3VhcmRzRmFpbGVkID09PSAwLFxuICAgICAgICAgICAgcmVzcG9uc2U6IHBhcnNlZFJlc3BvbnNlLFxuICAgICAgICAgICAgZ3VhcmRzUGFzc2VkLFxuICAgICAgICAgICAgZ3VhcmRzRmFpbGVkLFxuICAgICAgICAgICAgZ3VhcmRSZXN1bHRzLFxuICAgICAgICAgICAgYmxvY2tlZCxcbiAgICAgICAgICAgIGJsb2NrUmVhc29uLFxuICAgICAgICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICAgIH07XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogVmVyaWZ5IGEgdG9vbCBjYWxsLlxuICAgICAqL1xuICAgIHZlcmlmeVRvb2xDYWxsKFxuICAgICAgICB0b29sTmFtZTogc3RyaW5nLFxuICAgICAgICBhcmdzOiBSZWNvcmQ8c3RyaW5nLCBhbnk+LFxuICAgICAgICBndWFyZHM/OiBCYXNlR3VhcmRbXVxuICAgICk6IFZlcmlmaWNhdGlvblJlc3VsdCB7XG4gICAgICAgIGNvbnN0IHRvb2xDYWxsID0ge1xuICAgICAgICAgICAgdHlwZTogJ3Rvb2xfY2FsbCcsXG4gICAgICAgICAgICB0b29sTmFtZSxcbiAgICAgICAgICAgIGFyZ3VtZW50czogYXJncyxcbiAgICAgICAgfTtcbiAgICAgICAgcmV0dXJuIHRoaXMudmVyaWZ5KHRvb2xDYWxsLCBndWFyZHMpO1xuICAgIH1cblxuICAgIHByaXZhdGUgcGFyc2VSZXNwb25zZShyZXNwb25zZTogYW55KTogUGFyc2VkUmVzcG9uc2Uge1xuICAgICAgICBpZiAodHlwZW9mIHJlc3BvbnNlID09PSAnb2JqZWN0JyAmJiByZXNwb25zZSAhPT0gbnVsbCkge1xuICAgICAgICAgICAgcmV0dXJuIHJlc3BvbnNlO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHR5cGVvZiByZXNwb25zZSA9PT0gJ3N0cmluZycpIHtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIEpTT04ucGFyc2UocmVzcG9uc2UpO1xuICAgICAgICAgICAgfSBjYXRjaCB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIHsgdHlwZTogJ3RleHQnLCBjb250ZW50OiByZXNwb25zZSB9O1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHsgdHlwZTogJ3Vua25vd24nLCByYXc6IFN0cmluZyhyZXNwb25zZSkgfTtcbiAgICB9XG59XG4iXX0=
package/package.json ADDED
@@ -0,0 +1,54 @@
1
+ {
2
+ "name": "qwed-open-responses",
3
+ "version": "0.1.0",
4
+ "description": "Verification guards for AI agent outputs - Works with OpenAI, LangChain, Express.js",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "files": [
8
+ "dist",
9
+ "README.md",
10
+ "LICENSE"
11
+ ],
12
+ "scripts": {
13
+ "build": "tsc",
14
+ "test": "jest",
15
+ "prepublishOnly": "npm run build"
16
+ },
17
+ "keywords": [
18
+ "ai",
19
+ "verification",
20
+ "openai",
21
+ "responses-api",
22
+ "agents",
23
+ "tool-calling",
24
+ "structured-outputs",
25
+ "llm",
26
+ "safety",
27
+ "express",
28
+ "middleware",
29
+ "qwed"
30
+ ],
31
+ "author": "QWED-AI <team@qwedai.com>",
32
+ "license": "Apache-2.0",
33
+ "repository": {
34
+ "type": "git",
35
+ "url": "git+https://github.com/QWED-AI/qwed-open-responses.git"
36
+ },
37
+ "bugs": {
38
+ "url": "https://github.com/QWED-AI/qwed-open-responses/issues"
39
+ },
40
+ "homepage": "https://github.com/QWED-AI/qwed-open-responses#readme",
41
+ "devDependencies": {
42
+ "@types/express": "^4.17.21",
43
+ "@types/node": "^20.10.0",
44
+ "typescript": "^5.3.0",
45
+ "jest": "^29.7.0",
46
+ "@types/jest": "^29.5.11"
47
+ },
48
+ "peerDependencies": {
49
+ "express": "^4.18.0 || ^5.0.0"
50
+ },
51
+ "engines": {
52
+ "node": ">=16.0.0"
53
+ }
54
+ }