@buoy-design/core 0.1.3 → 0.1.5

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,250 @@
1
+ /**
2
+ * Fix Generator
3
+ *
4
+ * Generates Fix objects from DriftSignals by matching hardcoded values to design tokens.
5
+ */
6
+ import { createFixId, meetsConfidenceThreshold } from '../models/fix.js';
7
+ import { scoreConfidence } from './confidence.js';
8
+ /**
9
+ * Supported fix types for V1
10
+ */
11
+ const SUPPORTED_FIX_TYPES = [
12
+ 'hardcoded-color',
13
+ 'hardcoded-spacing',
14
+ 'hardcoded-radius',
15
+ 'hardcoded-font-size',
16
+ ];
17
+ /**
18
+ * Generate fixes from drift signals
19
+ */
20
+ export function generateFixes(drifts, tokens, options = {}) {
21
+ const { types = [...SUPPORTED_FIX_TYPES], minConfidence = 'low', includeFiles = [], excludeFiles = [], } = options;
22
+ const fixes = [];
23
+ for (const drift of drifts) {
24
+ // Filter by drift type
25
+ if (!isSupportedFixType(drift.type))
26
+ continue;
27
+ if (!types.includes(drift.type))
28
+ continue;
29
+ // Filter by file patterns
30
+ const file = drift.source.location?.split(':')[0] || '';
31
+ if (!matchesFilePatterns(file, includeFiles, excludeFiles))
32
+ continue;
33
+ // Try to generate a fix
34
+ const fix = generateFixForDrift(drift, tokens);
35
+ if (!fix)
36
+ continue;
37
+ // Filter by confidence
38
+ if (!meetsConfidenceThreshold(fix.confidence, minConfidence))
39
+ continue;
40
+ fixes.push(fix);
41
+ }
42
+ // Sort by confidence (high first) then by file
43
+ return fixes.sort((a, b) => {
44
+ const confidenceOrder = { high: 0, medium: 1, low: 2 };
45
+ const confDiff = confidenceOrder[a.confidence] - confidenceOrder[b.confidence];
46
+ if (confDiff !== 0)
47
+ return confDiff;
48
+ return a.file.localeCompare(b.file);
49
+ });
50
+ }
51
+ /**
52
+ * Generate a fix for a single drift signal
53
+ */
54
+ function generateFixForDrift(drift, tokens) {
55
+ // Extract the hardcoded value from drift details
56
+ const hardcodedValue = getHardcodedValue(drift);
57
+ if (!hardcodedValue)
58
+ return null;
59
+ // Find the best matching token
60
+ const match = findBestTokenMatch(hardcodedValue, drift.type, tokens);
61
+ if (!match)
62
+ return null;
63
+ // Parse location
64
+ const location = parseLocation(drift.source.location);
65
+ if (!location)
66
+ return null;
67
+ // Generate replacement
68
+ const replacement = generateReplacement(match.token, drift.type);
69
+ if (!replacement)
70
+ return null;
71
+ return {
72
+ id: createFixId(location.file, location.line, location.column),
73
+ driftSignalId: drift.id,
74
+ confidence: match.confidence.level,
75
+ confidenceScore: match.confidence.score,
76
+ file: location.file,
77
+ line: location.line,
78
+ column: location.column,
79
+ original: hardcodedValue,
80
+ replacement,
81
+ reason: match.confidence.reason,
82
+ fixType: drift.type,
83
+ tokenName: match.token.name,
84
+ };
85
+ }
86
+ /**
87
+ * Check if drift type is supported for fixing
88
+ */
89
+ function isSupportedFixType(type) {
90
+ return SUPPORTED_FIX_TYPES.includes(type);
91
+ }
92
+ /**
93
+ * Check if file matches include/exclude patterns
94
+ */
95
+ function matchesFilePatterns(file, includePatterns, excludePatterns) {
96
+ // If no include patterns, include all
97
+ if (includePatterns.length > 0) {
98
+ const included = includePatterns.some((pattern) => simpleGlobMatch(file, pattern));
99
+ if (!included)
100
+ return false;
101
+ }
102
+ // Check exclude patterns
103
+ if (excludePatterns.length > 0) {
104
+ const excluded = excludePatterns.some((pattern) => simpleGlobMatch(file, pattern));
105
+ if (excluded)
106
+ return false;
107
+ }
108
+ return true;
109
+ }
110
+ /**
111
+ * Simple glob-like pattern matching
112
+ * Supports * (any chars except /) and ** (any chars including /)
113
+ */
114
+ function simpleGlobMatch(file, pattern) {
115
+ // Convert glob pattern to regex
116
+ const regexStr = pattern
117
+ .replace(/[.+^${}()|[\]\\]/g, '\\$&') // Escape special regex chars
118
+ .replace(/\*\*/g, '{{GLOBSTAR}}') // Placeholder for **
119
+ .replace(/\*/g, '[^/]*') // * matches anything except /
120
+ .replace(/{{GLOBSTAR}}/g, '.*'); // ** matches anything including /
121
+ const regex = new RegExp(`^${regexStr}$|/${regexStr}$|^${regexStr}/`);
122
+ return regex.test(file);
123
+ }
124
+ /**
125
+ * Extract hardcoded value from drift signal
126
+ */
127
+ function getHardcodedValue(drift) {
128
+ // Check details.actual first (the hardcoded value found)
129
+ if (drift.details?.actual && typeof drift.details.actual === 'string') {
130
+ return drift.details.actual;
131
+ }
132
+ // Try to extract from message
133
+ // Common patterns: "Found hardcoded color #ffffff", "value '#3b82f6'"
134
+ const hexMatch = drift.message.match(/#[0-9a-fA-F]{3,8}\b/);
135
+ if (hexMatch)
136
+ return hexMatch[0];
137
+ const rgbMatch = drift.message.match(/rgba?\s*\([^)]+\)/i);
138
+ if (rgbMatch)
139
+ return rgbMatch[0];
140
+ const pxMatch = drift.message.match(/\b\d+(?:\.\d+)?px\b/);
141
+ if (pxMatch)
142
+ return pxMatch[0];
143
+ return null;
144
+ }
145
+ /**
146
+ * Find the best matching token for a hardcoded value
147
+ */
148
+ function findBestTokenMatch(value, driftType, tokens) {
149
+ const relevantTokens = filterTokensByType(tokens, driftType);
150
+ if (relevantTokens.length === 0)
151
+ return null;
152
+ let bestMatch = null;
153
+ for (const token of relevantTokens) {
154
+ const confidence = scoreConfidence(value, token, driftType);
155
+ if (!bestMatch || confidence.score > bestMatch.confidence.score) {
156
+ bestMatch = { token, confidence };
157
+ }
158
+ }
159
+ // Only return if we have a reasonable match
160
+ if (bestMatch && bestMatch.confidence.score >= 40) {
161
+ return bestMatch;
162
+ }
163
+ return null;
164
+ }
165
+ /**
166
+ * Filter tokens by drift type
167
+ */
168
+ function filterTokensByType(tokens, driftType) {
169
+ switch (driftType) {
170
+ case 'hardcoded-color':
171
+ return tokens.filter((t) => t.category === 'color');
172
+ case 'hardcoded-spacing':
173
+ case 'hardcoded-radius':
174
+ return tokens.filter((t) => t.category === 'spacing');
175
+ case 'hardcoded-font-size':
176
+ return tokens.filter((t) => t.category === 'typography' || t.category === 'sizing');
177
+ default:
178
+ return [];
179
+ }
180
+ }
181
+ /**
182
+ * Generate replacement string for a token
183
+ */
184
+ function generateReplacement(token, driftType) {
185
+ // Default to CSS custom property
186
+ const cssVarName = tokenToCssVar(token.name);
187
+ switch (driftType) {
188
+ case 'hardcoded-color':
189
+ return `var(${cssVarName})`;
190
+ case 'hardcoded-spacing':
191
+ case 'hardcoded-radius':
192
+ case 'hardcoded-font-size':
193
+ return `var(${cssVarName})`;
194
+ default:
195
+ return null;
196
+ }
197
+ }
198
+ /**
199
+ * Convert token name to CSS custom property name
200
+ */
201
+ function tokenToCssVar(name) {
202
+ // If already has -- prefix, use as-is
203
+ if (name.startsWith('--'))
204
+ return name;
205
+ // Convert camelCase or dot notation to kebab-case
206
+ const kebab = name
207
+ .replace(/\./g, '-')
208
+ .replace(/([a-z])([A-Z])/g, '$1-$2')
209
+ .toLowerCase();
210
+ return `--${kebab}`;
211
+ }
212
+ /**
213
+ * Parse location string into file, line, column
214
+ */
215
+ function parseLocation(location) {
216
+ if (!location)
217
+ return null;
218
+ // Format: "file/path.tsx:10:5" or "file/path.tsx:10"
219
+ const match = location.match(/^(.+?):(\d+)(?::(\d+))?$/);
220
+ if (!match)
221
+ return null;
222
+ return {
223
+ file: match[1],
224
+ line: parseInt(match[2], 10),
225
+ column: match[3] ? parseInt(match[3], 10) : 1,
226
+ };
227
+ }
228
+ /**
229
+ * Get summary of fixes grouped by type and confidence
230
+ */
231
+ export function summarizeFixes(fixes) {
232
+ const byConfidence = {
233
+ high: 0,
234
+ medium: 0,
235
+ low: 0,
236
+ };
237
+ const byType = {};
238
+ for (const fix of fixes) {
239
+ const conf = fix.confidence;
240
+ byConfidence[conf] = (byConfidence[conf] || 0) + 1;
241
+ byType[fix.fixType] = (byType[fix.fixType] || 0) + 1;
242
+ }
243
+ return {
244
+ total: fixes.length,
245
+ byConfidence,
246
+ byType,
247
+ highConfidenceCount: byConfidence.high || 0,
248
+ };
249
+ }
250
+ //# sourceMappingURL=generator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generator.js","sourceRoot":"","sources":["../../src/fix/generator.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Fix module - generates and applies fixes for drift signals
3
+ */
4
+ export { scoreConfidence, scoreColorConfidence, scoreSpacingConfidence } from './confidence.js';
5
+ export type { ConfidenceResult } from './confidence.js';
6
+ export { generateFixes, summarizeFixes } from './generator.js';
7
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/fix/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,EAAE,eAAe,EAAE,oBAAoB,EAAE,sBAAsB,EAAE,MAAM,iBAAiB,CAAC;AAChG,YAAY,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAGxD,OAAO,EAAE,aAAa,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Fix module - generates and applies fixes for drift signals
3
+ */
4
+ // Re-export confidence scoring
5
+ export { scoreConfidence, scoreColorConfidence, scoreSpacingConfidence } from './confidence.js';
6
+ // Re-export fix generator
7
+ export { generateFixes, summarizeFixes } from './generator.js';
8
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/fix/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,+BAA+B;AAC/B,OAAO,EAAE,eAAe,EAAE,oBAAoB,EAAE,sBAAsB,EAAE,MAAM,iBAAiB,CAAC;AAGhG,0BAA0B;AAC1B,OAAO,EAAE,aAAa,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC"}
package/dist/index.d.ts CHANGED
@@ -5,4 +5,5 @@ export * from './tokenization/index.js';
5
5
  export * from './plugins/index.js';
6
6
  export * from './tokens/index.js';
7
7
  export * from './graph/index.js';
8
+ export * from './fix/index.js';
8
9
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,cAAc,mBAAmB,CAAC;AAGlC,cAAc,qBAAqB,CAAC;AAGpC,cAAc,uBAAuB,CAAC;AAGtC,cAAc,yBAAyB,CAAC;AAGxC,cAAc,oBAAoB,CAAC;AAGnC,cAAc,mBAAmB,CAAC;AAGlC,cAAc,kBAAkB,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AACA,cAAc,mBAAmB,CAAC;AAGlC,cAAc,qBAAqB,CAAC;AAGpC,cAAc,uBAAuB,CAAC;AAGtC,cAAc,yBAAyB,CAAC;AAGxC,cAAc,oBAAoB,CAAC;AAGnC,cAAc,mBAAmB,CAAC;AAGlC,cAAc,kBAAkB,CAAC;AAGjC,cAAc,gBAAgB,CAAC"}
package/dist/index.js CHANGED
@@ -12,4 +12,6 @@ export * from './plugins/index.js';
12
12
  export * from './tokens/index.js';
13
13
  // Re-export graph
14
14
  export * from './graph/index.js';
15
+ // Re-export fix
16
+ export * from './fix/index.js';
15
17
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,uBAAuB;AACvB,cAAc,mBAAmB,CAAC;AAElC,qBAAqB;AACrB,cAAc,qBAAqB,CAAC;AAEpC,uBAAuB;AACvB,cAAc,uBAAuB,CAAC;AAEtC,yBAAyB;AACzB,cAAc,yBAAyB,CAAC;AAExC,oBAAoB;AACpB,cAAc,oBAAoB,CAAC;AAEnC,0BAA0B;AAC1B,cAAc,mBAAmB,CAAC;AAElC,kBAAkB;AAClB,cAAc,kBAAkB,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,uBAAuB;AACvB,cAAc,mBAAmB,CAAC;AAElC,qBAAqB;AACrB,cAAc,qBAAqB,CAAC;AAEpC,uBAAuB;AACvB,cAAc,uBAAuB,CAAC;AAEtC,yBAAyB;AACzB,cAAc,yBAAyB,CAAC;AAExC,oBAAoB;AACpB,cAAc,oBAAoB,CAAC;AAEnC,0BAA0B;AAC1B,cAAc,mBAAmB,CAAC;AAElC,kBAAkB;AAClB,cAAc,kBAAkB,CAAC;AAEjC,gBAAgB;AAChB,cAAc,gBAAgB,CAAC"}
@@ -0,0 +1,270 @@
1
+ import { z } from 'zod';
2
+ /**
3
+ * Confidence level for a fix
4
+ */
5
+ export declare const ConfidenceLevelSchema: z.ZodEnum<["high", "medium", "low"]>;
6
+ export type ConfidenceLevel = z.infer<typeof ConfidenceLevelSchema>;
7
+ /**
8
+ * A proposed fix for a drift signal
9
+ */
10
+ export declare const FixSchema: z.ZodObject<{
11
+ /** Unique identifier for this fix */
12
+ id: z.ZodString;
13
+ /** The drift signal this fix addresses */
14
+ driftSignalId: z.ZodString;
15
+ /** Confidence level (high = 95%+, medium = 70-94%, low = <70%) */
16
+ confidence: z.ZodEnum<["high", "medium", "low"]>;
17
+ /** Numeric confidence score (0-100) */
18
+ confidenceScore: z.ZodNumber;
19
+ /** File path where the fix will be applied */
20
+ file: z.ZodString;
21
+ /** Line number (1-indexed) */
22
+ line: z.ZodNumber;
23
+ /** Column number (1-indexed) */
24
+ column: z.ZodNumber;
25
+ /** Original text to replace */
26
+ original: z.ZodString;
27
+ /** Replacement text */
28
+ replacement: z.ZodString;
29
+ /** Human-readable reason for this fix */
30
+ reason: z.ZodString;
31
+ /** The type of fix */
32
+ fixType: z.ZodEnum<["hardcoded-color", "hardcoded-spacing", "hardcoded-radius", "hardcoded-font-size"]>;
33
+ /** Token name being applied (if applicable) */
34
+ tokenName: z.ZodOptional<z.ZodString>;
35
+ }, "strip", z.ZodTypeAny, {
36
+ line: number;
37
+ id: string;
38
+ reason: string;
39
+ driftSignalId: string;
40
+ confidence: "high" | "medium" | "low";
41
+ confidenceScore: number;
42
+ file: string;
43
+ column: number;
44
+ original: string;
45
+ replacement: string;
46
+ fixType: "hardcoded-color" | "hardcoded-spacing" | "hardcoded-radius" | "hardcoded-font-size";
47
+ tokenName?: string | undefined;
48
+ }, {
49
+ line: number;
50
+ id: string;
51
+ reason: string;
52
+ driftSignalId: string;
53
+ confidence: "high" | "medium" | "low";
54
+ confidenceScore: number;
55
+ file: string;
56
+ column: number;
57
+ original: string;
58
+ replacement: string;
59
+ fixType: "hardcoded-color" | "hardcoded-spacing" | "hardcoded-radius" | "hardcoded-font-size";
60
+ tokenName?: string | undefined;
61
+ }>;
62
+ export type Fix = z.infer<typeof FixSchema>;
63
+ /**
64
+ * Result of applying a single fix
65
+ */
66
+ export declare const FixResultSchema: z.ZodObject<{
67
+ fixId: z.ZodString;
68
+ status: z.ZodEnum<["applied", "skipped", "failed"]>;
69
+ error: z.ZodOptional<z.ZodString>;
70
+ }, "strip", z.ZodTypeAny, {
71
+ status: "applied" | "skipped" | "failed";
72
+ fixId: string;
73
+ error?: string | undefined;
74
+ }, {
75
+ status: "applied" | "skipped" | "failed";
76
+ fixId: string;
77
+ error?: string | undefined;
78
+ }>;
79
+ export type FixResult = z.infer<typeof FixResultSchema>;
80
+ /**
81
+ * A fix session tracks all fixes applied in one run
82
+ */
83
+ export declare const FixSessionSchema: z.ZodObject<{
84
+ /** Unique session identifier */
85
+ id: z.ZodString;
86
+ /** When the session started */
87
+ startedAt: z.ZodDate;
88
+ /** When the session completed */
89
+ completedAt: z.ZodOptional<z.ZodDate>;
90
+ /** All fixes that were considered */
91
+ fixes: z.ZodArray<z.ZodObject<{
92
+ /** Unique identifier for this fix */
93
+ id: z.ZodString;
94
+ /** The drift signal this fix addresses */
95
+ driftSignalId: z.ZodString;
96
+ /** Confidence level (high = 95%+, medium = 70-94%, low = <70%) */
97
+ confidence: z.ZodEnum<["high", "medium", "low"]>;
98
+ /** Numeric confidence score (0-100) */
99
+ confidenceScore: z.ZodNumber;
100
+ /** File path where the fix will be applied */
101
+ file: z.ZodString;
102
+ /** Line number (1-indexed) */
103
+ line: z.ZodNumber;
104
+ /** Column number (1-indexed) */
105
+ column: z.ZodNumber;
106
+ /** Original text to replace */
107
+ original: z.ZodString;
108
+ /** Replacement text */
109
+ replacement: z.ZodString;
110
+ /** Human-readable reason for this fix */
111
+ reason: z.ZodString;
112
+ /** The type of fix */
113
+ fixType: z.ZodEnum<["hardcoded-color", "hardcoded-spacing", "hardcoded-radius", "hardcoded-font-size"]>;
114
+ /** Token name being applied (if applicable) */
115
+ tokenName: z.ZodOptional<z.ZodString>;
116
+ }, "strip", z.ZodTypeAny, {
117
+ line: number;
118
+ id: string;
119
+ reason: string;
120
+ driftSignalId: string;
121
+ confidence: "high" | "medium" | "low";
122
+ confidenceScore: number;
123
+ file: string;
124
+ column: number;
125
+ original: string;
126
+ replacement: string;
127
+ fixType: "hardcoded-color" | "hardcoded-spacing" | "hardcoded-radius" | "hardcoded-font-size";
128
+ tokenName?: string | undefined;
129
+ }, {
130
+ line: number;
131
+ id: string;
132
+ reason: string;
133
+ driftSignalId: string;
134
+ confidence: "high" | "medium" | "low";
135
+ confidenceScore: number;
136
+ file: string;
137
+ column: number;
138
+ original: string;
139
+ replacement: string;
140
+ fixType: "hardcoded-color" | "hardcoded-spacing" | "hardcoded-radius" | "hardcoded-font-size";
141
+ tokenName?: string | undefined;
142
+ }>, "many">;
143
+ /** Results for each fix */
144
+ results: z.ZodArray<z.ZodObject<{
145
+ fixId: z.ZodString;
146
+ status: z.ZodEnum<["applied", "skipped", "failed"]>;
147
+ error: z.ZodOptional<z.ZodString>;
148
+ }, "strip", z.ZodTypeAny, {
149
+ status: "applied" | "skipped" | "failed";
150
+ fixId: string;
151
+ error?: string | undefined;
152
+ }, {
153
+ status: "applied" | "skipped" | "failed";
154
+ fixId: string;
155
+ error?: string | undefined;
156
+ }>, "many">;
157
+ /** Summary statistics */
158
+ summary: z.ZodObject<{
159
+ total: z.ZodNumber;
160
+ applied: z.ZodNumber;
161
+ skipped: z.ZodNumber;
162
+ failed: z.ZodNumber;
163
+ }, "strip", z.ZodTypeAny, {
164
+ applied: number;
165
+ skipped: number;
166
+ failed: number;
167
+ total: number;
168
+ }, {
169
+ applied: number;
170
+ skipped: number;
171
+ failed: number;
172
+ total: number;
173
+ }>;
174
+ }, "strip", z.ZodTypeAny, {
175
+ id: string;
176
+ startedAt: Date;
177
+ fixes: {
178
+ line: number;
179
+ id: string;
180
+ reason: string;
181
+ driftSignalId: string;
182
+ confidence: "high" | "medium" | "low";
183
+ confidenceScore: number;
184
+ file: string;
185
+ column: number;
186
+ original: string;
187
+ replacement: string;
188
+ fixType: "hardcoded-color" | "hardcoded-spacing" | "hardcoded-radius" | "hardcoded-font-size";
189
+ tokenName?: string | undefined;
190
+ }[];
191
+ results: {
192
+ status: "applied" | "skipped" | "failed";
193
+ fixId: string;
194
+ error?: string | undefined;
195
+ }[];
196
+ summary: {
197
+ applied: number;
198
+ skipped: number;
199
+ failed: number;
200
+ total: number;
201
+ };
202
+ completedAt?: Date | undefined;
203
+ }, {
204
+ id: string;
205
+ startedAt: Date;
206
+ fixes: {
207
+ line: number;
208
+ id: string;
209
+ reason: string;
210
+ driftSignalId: string;
211
+ confidence: "high" | "medium" | "low";
212
+ confidenceScore: number;
213
+ file: string;
214
+ column: number;
215
+ original: string;
216
+ replacement: string;
217
+ fixType: "hardcoded-color" | "hardcoded-spacing" | "hardcoded-radius" | "hardcoded-font-size";
218
+ tokenName?: string | undefined;
219
+ }[];
220
+ results: {
221
+ status: "applied" | "skipped" | "failed";
222
+ fixId: string;
223
+ error?: string | undefined;
224
+ }[];
225
+ summary: {
226
+ applied: number;
227
+ skipped: number;
228
+ failed: number;
229
+ total: number;
230
+ };
231
+ completedAt?: Date | undefined;
232
+ }>;
233
+ export type FixSession = z.infer<typeof FixSessionSchema>;
234
+ /**
235
+ * Options for generating fixes
236
+ */
237
+ export interface FixGeneratorOptions {
238
+ /** Filter to specific fix types */
239
+ types?: Array<Fix['fixType']>;
240
+ /** Minimum confidence level to include */
241
+ minConfidence?: ConfidenceLevel;
242
+ /** File glob patterns to include */
243
+ includeFiles?: string[];
244
+ /** File glob patterns to exclude */
245
+ excludeFiles?: string[];
246
+ }
247
+ /**
248
+ * Options for applying fixes
249
+ */
250
+ export interface FixApplyOptions {
251
+ /** Only show what would be done, don't modify files */
252
+ dryRun?: boolean;
253
+ /** Create .bak backup files before modifying */
254
+ backup?: boolean;
255
+ /** Minimum confidence level to apply */
256
+ minConfidence?: ConfidenceLevel;
257
+ }
258
+ /**
259
+ * Create a unique fix ID
260
+ */
261
+ export declare function createFixId(file: string, line: number, column: number): string;
262
+ /**
263
+ * Get confidence level from numeric score
264
+ */
265
+ export declare function getConfidenceLevel(score: number): ConfidenceLevel;
266
+ /**
267
+ * Check if a confidence level meets the minimum threshold
268
+ */
269
+ export declare function meetsConfidenceThreshold(level: ConfidenceLevel, minimum: ConfidenceLevel): boolean;
270
+ //# sourceMappingURL=fix.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fix.d.ts","sourceRoot":"","sources":["../../src/models/fix.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB;;GAEG;AACH,eAAO,MAAM,qBAAqB,sCAAoC,CAAC;AACvE,MAAM,MAAM,eAAe,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,qBAAqB,CAAC,CAAC;AAEpE;;GAEG;AACH,eAAO,MAAM,SAAS;IACpB,qCAAqC;;IAGrC,0CAA0C;;IAG1C,kEAAkE;;IAGlE,uCAAuC;;IAGvC,8CAA8C;;IAG9C,8BAA8B;;IAG9B,gCAAgC;;IAGhC,+BAA+B;;IAG/B,uBAAuB;;IAGvB,yCAAyC;;IAGzC,sBAAsB;;IAQtB,+CAA+C;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAE/C,CAAC;AAEH,MAAM,MAAM,GAAG,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,SAAS,CAAC,CAAC;AAE5C;;GAEG;AACH,eAAO,MAAM,eAAe;;;;;;;;;;;;EAI1B,CAAC;AAEH,MAAM,MAAM,SAAS,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,eAAe,CAAC,CAAC;AAExD;;GAEG;AACH,eAAO,MAAM,gBAAgB;IAC3B,gCAAgC;;IAGhC,+BAA+B;;IAG/B,iCAAiC;;IAGjC,qCAAqC;;QApErC,qCAAqC;;QAGrC,0CAA0C;;QAG1C,kEAAkE;;QAGlE,uCAAuC;;QAGvC,8CAA8C;;QAG9C,8BAA8B;;QAG9B,gCAAgC;;QAGhC,+BAA+B;;QAG/B,uBAAuB;;QAGvB,yCAAyC;;QAGzC,sBAAsB;;QAQtB,+CAA+C;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAiC/C,2BAA2B;;;;;;;;;;;;;;IAG3B,yBAAyB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAOzB,CAAC;AAEH,MAAM,MAAM,UAAU,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,gBAAgB,CAAC,CAAC;AAE1D;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,mCAAmC;IACnC,KAAK,CAAC,EAAE,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC;IAE9B,0CAA0C;IAC1C,aAAa,CAAC,EAAE,eAAe,CAAC;IAEhC,oCAAoC;IACpC,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;IAExB,oCAAoC;IACpC,YAAY,CAAC,EAAE,MAAM,EAAE,CAAC;CACzB;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,uDAAuD;IACvD,MAAM,CAAC,EAAE,OAAO,CAAC;IAEjB,gDAAgD;IAChD,MAAM,CAAC,EAAE,OAAO,CAAC;IAEjB,wCAAwC;IACxC,aAAa,CAAC,EAAE,eAAe,CAAC;CACjC;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,CAE9E;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,MAAM,GAAG,eAAe,CAIjE;AAED;;GAEG;AACH,wBAAgB,wBAAwB,CACtC,KAAK,EAAE,eAAe,EACtB,OAAO,EAAE,eAAe,GACvB,OAAO,CAOT"}