UNPKG

llmverify

Version:

AI Output Verification Toolkit — Local-first LLM safety, hallucination detection, PII redaction, prompt injection defense, and runtime monitoring. Zero telemetry. OWASP LLM Top 10 aligned.

433 lines 43.5 kB
"use strict"; /** * PII Detection & Redaction * * Comprehensive PII detection with redaction utilities. * Supports US and international formats. * * @module csm6/security/pii-detection * @author Haiec * @license MIT */ Object.defineProperty(exports, "__esModule", { value: true }); exports.checkPII = checkPII; exports.redactPII = redactPII; exports.containsPII = containsPII; exports.getPIIRiskScore = getPIIRiskScore; const text_1 = require("../../utils/text"); const LIMITATIONS = [ 'Pattern-based detection only', 'May miss obfuscated PII', 'US-centric patterns for SSN/phone (international formats included)', 'Context-dependent false positives possible', 'Cannot detect PII in images or encoded content' ]; const METHODOLOGY = 'Regex-based pattern matching for common PII formats. ' + 'Detects emails, phone numbers, SSNs, credit cards, API keys, and more. ' + 'Accuracy: ~90% for standard formats, lower for variations.'; /** * Comprehensive PII patterns */ const PII_PATTERNS = [ // === PERSONAL IDENTIFIERS === { name: 'EMAIL', pattern: /\b[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\.[A-Z|a-z]{2,}\b/g, severity: 'medium', confidence: 0.95, message: 'Email address detected', category: 'personal' }, { name: 'PHONE_US', pattern: /\b(?:\+1[-.\s]?)?\(?\d{3}\)?[-.\s]?\d{3}[-.\s]?\d{4}\b/g, severity: 'medium', confidence: 0.85, message: 'US phone number detected', category: 'personal' }, { name: 'PHONE_INTL', pattern: /\b\+(?:[0-9][\s.-]?){6,14}[0-9]\b/g, severity: 'medium', confidence: 0.8, message: 'International phone number detected', category: 'personal' }, { name: 'SSN', pattern: /\b\d{3}[-\s]?\d{2}[-\s]?\d{4}\b/g, severity: 'critical', confidence: 0.8, message: 'Potential SSN detected', category: 'personal' }, { name: 'PASSPORT', pattern: /\b[A-Z]{1,2}\d{6,9}\b/g, severity: 'high', confidence: 0.6, message: 'Potential passport number detected', category: 'personal' }, { name: 'DRIVERS_LICENSE', pattern: /\b[A-Z]{1,2}\d{5,8}\b/g, severity: 'high', confidence: 0.5, message: 'Potential drivers license detected', category: 'personal' }, { name: 'DATE_OF_BIRTH', pattern: /\b(?:dob|date\s+of\s+birth|born\s+on|birthday)[:\s]+\d{1,2}[-/]\d{1,2}[-/]\d{2,4}\b/gi, severity: 'medium', confidence: 0.85, message: 'Date of birth detected', category: 'personal' }, // === FINANCIAL === { name: 'CREDIT_CARD', pattern: /\b(?:4[0-9]{12}(?:[0-9]{3})?|5[1-5][0-9]{14}|3[47][0-9]{13}|6(?:011|5[0-9]{2})[0-9]{12})\b/g, severity: 'critical', confidence: 0.9, message: 'Credit card number detected', category: 'financial' }, { name: 'BANK_ACCOUNT', pattern: /\b(?:account|acct)[\s#:]*\d{8,17}\b/gi, severity: 'high', confidence: 0.7, message: 'Bank account number detected', category: 'financial' }, { name: 'ROUTING_NUMBER', pattern: /\b(?:routing|aba)[\s#:]*\d{9}\b/gi, severity: 'high', confidence: 0.75, message: 'Bank routing number detected', category: 'financial' }, { name: 'IBAN', pattern: /\b[A-Z]{2}\d{2}[A-Z0-9]{4}\d{7}([A-Z0-9]?){0,16}\b/g, severity: 'high', confidence: 0.85, message: 'IBAN detected', category: 'financial' }, // === CREDENTIALS & SECRETS === { name: 'API_KEY', pattern: /\b(?:sk|pk|api|key|token|secret|password)[-_]?[a-zA-Z0-9]{20,}\b/gi, severity: 'critical', confidence: 0.75, message: 'Potential API key or secret detected', category: 'credential' }, { name: 'AWS_KEY', pattern: /\bAKIA[0-9A-Z]{16}\b/g, severity: 'critical', confidence: 0.95, message: 'AWS Access Key ID detected', category: 'credential' }, { name: 'AWS_SECRET', pattern: /\b[A-Za-z0-9/+=]{40}\b/g, severity: 'critical', confidence: 0.6, message: 'Potential AWS Secret Key detected', category: 'credential' }, { name: 'GITHUB_TOKEN', pattern: /\b(ghp|gho|ghu|ghs|ghr)_[A-Za-z0-9]{36,}\b/g, severity: 'critical', confidence: 0.95, message: 'GitHub token detected', category: 'credential' }, { name: 'SLACK_TOKEN', pattern: /\bxox[baprs]-[0-9]{10,13}-[0-9]{10,13}-[a-zA-Z0-9]{24}\b/g, severity: 'critical', confidence: 0.95, message: 'Slack token detected', category: 'credential' }, { name: 'STRIPE_KEY', pattern: /\b(sk|pk)_(test|live)_[A-Za-z0-9]{24,}\b/g, severity: 'critical', confidence: 0.95, message: 'Stripe API key detected', category: 'credential' }, { name: 'JWT_TOKEN', pattern: /\beyJ[A-Za-z0-9-_]+\.eyJ[A-Za-z0-9-_]+\.[A-Za-z0-9-_]+\b/g, severity: 'high', confidence: 0.9, message: 'JWT token detected', category: 'credential' }, { name: 'PRIVATE_KEY', pattern: /-----BEGIN\s+(RSA|DSA|EC|OPENSSH|PGP)?\s*PRIVATE\s+KEY-----/gi, severity: 'critical', confidence: 0.98, message: 'Private key detected', category: 'credential' }, { name: 'PASSWORD_INLINE', pattern: /\b(?:password|passwd|pwd)\s*[:=]\s*["']?[^\s"']{8,}["']?\b/gi, severity: 'critical', confidence: 0.85, message: 'Inline password detected', category: 'credential' }, // === LOCATION === { name: 'IP_ADDRESS', pattern: /\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\b/g, severity: 'low', confidence: 0.9, message: 'IP address detected', category: 'location' }, { name: 'IPV6_ADDRESS', pattern: /\b(?:[0-9a-fA-F]{1,4}:){7}[0-9a-fA-F]{1,4}\b/g, severity: 'low', confidence: 0.9, message: 'IPv6 address detected', category: 'location' }, { name: 'MAC_ADDRESS', pattern: /\b([0-9A-Fa-f]{2}[:-]){5}([0-9A-Fa-f]{2})\b/g, severity: 'low', confidence: 0.9, message: 'MAC address detected', category: 'location' }, { name: 'US_ADDRESS', pattern: /\b\d{1,5}\s+[\w\s]+(?:street|st|avenue|ave|road|rd|boulevard|blvd|drive|dr|lane|ln|court|ct)\b/gi, severity: 'medium', confidence: 0.6, message: 'US street address detected', category: 'location' }, { name: 'ZIP_CODE', pattern: /\b\d{5}(?:-\d{4})?\b/g, severity: 'low', confidence: 0.5, message: 'US ZIP code detected', category: 'location' }, // === HEALTH === { name: 'MEDICAL_RECORD', pattern: /\b(?:mrn|medical\s+record)[\s#:]*\d{6,12}\b/gi, severity: 'critical', confidence: 0.8, message: 'Medical record number detected', category: 'health' } ]; /** * Check for PII in content * @param content - The content to scan * @returns Array of findings */ function checkPII(content) { const findings = []; const foundTypes = new Set(); for (const piiPattern of PII_PATTERNS) { // Reset lastIndex piiPattern.pattern.lastIndex = 0; const matches = Array.from(content.matchAll(piiPattern.pattern)); for (const match of matches) { // Skip if we already found this type (avoid spam) if (foundTypes.has(piiPattern.name)) continue; // Skip if likely false positive if (isLikelyFalsePositive(content, match[0], piiPattern.name)) continue; foundTypes.add(piiPattern.name); findings.push({ id: `PII_${piiPattern.name}`, category: 'privacy', severity: piiPattern.severity, surface: 'output', message: piiPattern.message, recommendation: `Remove or redact ${piiPattern.name.toLowerCase().replace(/_/g, ' ')} before use.`, evidence: { textSample: redactSample(match[0]), pattern: piiPattern.name, context: (0, text_1.extractContext)(content, match.index || 0, 30) }, confidence: calculateConfidence(piiPattern.confidence, match[0]), limitations: LIMITATIONS, methodology: METHODOLOGY, metadata: { piiType: piiPattern.name, piiCategory: piiPattern.category } }); } } return findings; } /** * Redact all PII from content * @param content - The content to redact * @param replacement - Replacement string (default: [REDACTED]) * @returns Redacted content and list of redactions */ function redactPII(content, replacement = '[REDACTED]') { let redacted = content; const redactions = []; for (const piiPattern of PII_PATTERNS) { piiPattern.pattern.lastIndex = 0; const matches = Array.from(content.matchAll(piiPattern.pattern)); for (const match of matches) { if (!isLikelyFalsePositive(content, match[0], piiPattern.name)) { redactions.push({ type: piiPattern.name, original: redactSample(match[0]), position: match.index || 0 }); redacted = redacted.replace(match[0], replacement); } } } return { redacted, redactions, piiCount: redactions.length }; } /** * Check if content contains any PII * @param content - The content to check * @returns true if PII detected */ function containsPII(content) { return checkPII(content).length > 0; } /** * Get PII risk score (0-1) * @param content - The content to score * @returns Risk score */ function getPIIRiskScore(content) { const findings = checkPII(content); if (findings.length === 0) return 0; const severityWeights = { low: 0.2, medium: 0.4, high: 0.7, critical: 1.0 }; let maxScore = 0; for (const finding of findings) { const score = severityWeights[finding.severity] * finding.confidence.value; maxScore = Math.max(maxScore, score); } // Add penalty for multiple PII types const typeBonus = Math.min(findings.length * 0.1, 0.3); return Math.min(1, maxScore + typeBonus); } /** * Redact sensitive parts of sample for display */ function redactSample(text) { if (text.length <= 8) { return text.substring(0, 2) + '***' + text.substring(text.length - 2); } const visible = Math.min(4, Math.floor(text.length / 4)); return text.substring(0, visible) + '***' + text.substring(text.length - visible); } /** * Calculate confidence score */ function calculateConfidence(baseConfidence, match) { let value = baseConfidence; // Adjust based on match characteristics if (match.length > 20) value += 0.05; if (/^[A-Z]/.test(match)) value -= 0.05; // Might be a name, not PII value = Math.max(0.5, Math.min(0.95, value)); return { value, interval: [Math.max(0, value - 0.1), Math.min(1, value + 0.1)], method: 'empirical', factors: { patternStrength: baseConfidence } }; } /** * Check for false positives */ function isLikelyFalsePositive(fullText, match, type) { // Example/placeholder indicators const placeholders = [ /example/i, /placeholder/i, /test/i, /sample/i, /dummy/i, /fake/i, /demo/i, /xxx/i ]; const context = fullText.substring(Math.max(0, fullText.indexOf(match) - 50), fullText.indexOf(match) + match.length + 50); if (placeholders.some(p => p.test(context))) { return true; } // SSN false positives if (type === 'SSN') { // Check if it's a date format if (/\d{2}[-/]\d{2}[-/]\d{4}/.test(match)) { return true; } // Check if preceded by date-like context if (/date|time|year|month|day/i.test(context)) { return true; } } // Phone false positives if (type === 'PHONE_US') { // Check if it's a zip code or other number if (/zip|code|id|number/i.test(context) && !/phone|call|mobile|cell/i.test(context)) { return true; } } // ZIP code - too many false positives, require context if (type === 'ZIP_CODE') { if (!/zip|postal|address/i.test(context)) { return true; } } // AWS Secret - high false positive rate if (type === 'AWS_SECRET') { // Only flag if near AWS context if (!/aws|amazon|secret|key/i.test(context)) { return true; } } return false; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pii-detection.js","sourceRoot":"","sources":["../../../src/csm6/security/pii-detection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PII Detection & Redaction\n * \n * Comprehensive PII detection with redaction utilities.\n * Supports US and international formats.\n * \n * @module csm6/security/pii-detection\n * @author Haiec\n * @license MIT\n */\n\nimport { Finding, ConfidenceScore } from '../../types/results';\nimport { truncate, extractContext } from '../../utils/text';\n\nconst LIMITATIONS = [\n  'Pattern-based detection only',\n  'May miss obfuscated PII',\n  'US-centric patterns for SSN/phone (international formats included)',\n  'Context-dependent false positives possible',\n  'Cannot detect PII in images or encoded content'\n];\n\nconst METHODOLOGY =\n  'Regex-based pattern matching for common PII formats. ' +\n  'Detects emails, phone numbers, SSNs, credit cards, API keys, and more. ' +\n  'Accuracy: ~90% for standard formats, lower for variations.';\n\ninterface PIIPattern {\n  name: string;\n  pattern: RegExp;\n  severity: 'low' | 'medium' | 'high' | 'critical';\n  confidence: number;\n  message: string;\n  category: 'personal' | 'financial' | 'credential' | 'location' | 'health';\n}\n\n/**\n * Comprehensive PII patterns\n */\nconst PII_PATTERNS: PIIPattern[] = [\n  // === PERSONAL IDENTIFIERS ===\n  {\n    name: 'EMAIL',\n    pattern: /\\b[A-Za-z0-9._%+-]+@[A-Za-z0-9.-]+\\.[A-Z|a-z]{2,}\\b/g,\n    severity: 'medium',\n    confidence: 0.95,\n    message: 'Email address detected',\n    category: 'personal'\n  },\n  {\n    name: 'PHONE_US',\n    pattern: /\\b(?:\\+1[-.\\s]?)?\\(?\\d{3}\\)?[-.\\s]?\\d{3}[-.\\s]?\\d{4}\\b/g,\n    severity: 'medium',\n    confidence: 0.85,\n    message: 'US phone number detected',\n    category: 'personal'\n  },\n  {\n    name: 'PHONE_INTL',\n    pattern: /\\b\\+(?:[0-9][\\s.-]?){6,14}[0-9]\\b/g,\n    severity: 'medium',\n    confidence: 0.8,\n    message: 'International phone number detected',\n    category: 'personal'\n  },\n  {\n    name: 'SSN',\n    pattern: /\\b\\d{3}[-\\s]?\\d{2}[-\\s]?\\d{4}\\b/g,\n    severity: 'critical',\n    confidence: 0.8,\n    message: 'Potential SSN detected',\n    category: 'personal'\n  },\n  {\n    name: 'PASSPORT',\n    pattern: /\\b[A-Z]{1,2}\\d{6,9}\\b/g,\n    severity: 'high',\n    confidence: 0.6,\n    message: 'Potential passport number detected',\n    category: 'personal'\n  },\n  {\n    name: 'DRIVERS_LICENSE',\n    pattern: /\\b[A-Z]{1,2}\\d{5,8}\\b/g,\n    severity: 'high',\n    confidence: 0.5,\n    message: 'Potential drivers license detected',\n    category: 'personal'\n  },\n  {\n    name: 'DATE_OF_BIRTH',\n    pattern: /\\b(?:dob|date\\s+of\\s+birth|born\\s+on|birthday)[:\\s]+\\d{1,2}[-/]\\d{1,2}[-/]\\d{2,4}\\b/gi,\n    severity: 'medium',\n    confidence: 0.85,\n    message: 'Date of birth detected',\n    category: 'personal'\n  },\n  \n  // === FINANCIAL ===\n  {\n    name: 'CREDIT_CARD',\n    pattern: /\\b(?:4[0-9]{12}(?:[0-9]{3})?|5[1-5][0-9]{14}|3[47][0-9]{13}|6(?:011|5[0-9]{2})[0-9]{12})\\b/g,\n    severity: 'critical',\n    confidence: 0.9,\n    message: 'Credit card number detected',\n    category: 'financial'\n  },\n  {\n    name: 'BANK_ACCOUNT',\n    pattern: /\\b(?:account|acct)[\\s#:]*\\d{8,17}\\b/gi,\n    severity: 'high',\n    confidence: 0.7,\n    message: 'Bank account number detected',\n    category: 'financial'\n  },\n  {\n    name: 'ROUTING_NUMBER',\n    pattern: /\\b(?:routing|aba)[\\s#:]*\\d{9}\\b/gi,\n    severity: 'high',\n    confidence: 0.75,\n    message: 'Bank routing number detected',\n    category: 'financial'\n  },\n  {\n    name: 'IBAN',\n    pattern: /\\b[A-Z]{2}\\d{2}[A-Z0-9]{4}\\d{7}([A-Z0-9]?){0,16}\\b/g,\n    severity: 'high',\n    confidence: 0.85,\n    message: 'IBAN detected',\n    category: 'financial'\n  },\n  \n  // === CREDENTIALS & SECRETS ===\n  {\n    name: 'API_KEY',\n    pattern: /\\b(?:sk|pk|api|key|token|secret|password)[-_]?[a-zA-Z0-9]{20,}\\b/gi,\n    severity: 'critical',\n    confidence: 0.75,\n    message: 'Potential API key or secret detected',\n    category: 'credential'\n  },\n  {\n    name: 'AWS_KEY',\n    pattern: /\\bAKIA[0-9A-Z]{16}\\b/g,\n    severity: 'critical',\n    confidence: 0.95,\n    message: 'AWS Access Key ID detected',\n    category: 'credential'\n  },\n  {\n    name: 'AWS_SECRET',\n    pattern: /\\b[A-Za-z0-9/+=]{40}\\b/g,\n    severity: 'critical',\n    confidence: 0.6,\n    message: 'Potential AWS Secret Key detected',\n    category: 'credential'\n  },\n  {\n    name: 'GITHUB_TOKEN',\n    pattern: /\\b(ghp|gho|ghu|ghs|ghr)_[A-Za-z0-9]{36,}\\b/g,\n    severity: 'critical',\n    confidence: 0.95,\n    message: 'GitHub token detected',\n    category: 'credential'\n  },\n  {\n    name: 'SLACK_TOKEN',\n    pattern: /\\bxox[baprs]-[0-9]{10,13}-[0-9]{10,13}-[a-zA-Z0-9]{24}\\b/g,\n    severity: 'critical',\n    confidence: 0.95,\n    message: 'Slack token detected',\n    category: 'credential'\n  },\n  {\n    name: 'STRIPE_KEY',\n    pattern: /\\b(sk|pk)_(test|live)_[A-Za-z0-9]{24,}\\b/g,\n    severity: 'critical',\n    confidence: 0.95,\n    message: 'Stripe API key detected',\n    category: 'credential'\n  },\n  {\n    name: 'JWT_TOKEN',\n    pattern: /\\beyJ[A-Za-z0-9-_]+\\.eyJ[A-Za-z0-9-_]+\\.[A-Za-z0-9-_]+\\b/g,\n    severity: 'high',\n    confidence: 0.9,\n    message: 'JWT token detected',\n    category: 'credential'\n  },\n  {\n    name: 'PRIVATE_KEY',\n    pattern: /-----BEGIN\\s+(RSA|DSA|EC|OPENSSH|PGP)?\\s*PRIVATE\\s+KEY-----/gi,\n    severity: 'critical',\n    confidence: 0.98,\n    message: 'Private key detected',\n    category: 'credential'\n  },\n  {\n    name: 'PASSWORD_INLINE',\n    pattern: /\\b(?:password|passwd|pwd)\\s*[:=]\\s*[\"']?[^\\s\"']{8,}[\"']?\\b/gi,\n    severity: 'critical',\n    confidence: 0.85,\n    message: 'Inline password detected',\n    category: 'credential'\n  },\n  \n  // === LOCATION ===\n  {\n    name: 'IP_ADDRESS',\n    pattern: /\\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\b/g,\n    severity: 'low',\n    confidence: 0.9,\n    message: 'IP address detected',\n    category: 'location'\n  },\n  {\n    name: 'IPV6_ADDRESS',\n    pattern: /\\b(?:[0-9a-fA-F]{1,4}:){7}[0-9a-fA-F]{1,4}\\b/g,\n    severity: 'low',\n    confidence: 0.9,\n    message: 'IPv6 address detected',\n    category: 'location'\n  },\n  {\n    name: 'MAC_ADDRESS',\n    pattern: /\\b([0-9A-Fa-f]{2}[:-]){5}([0-9A-Fa-f]{2})\\b/g,\n    severity: 'low',\n    confidence: 0.9,\n    message: 'MAC address detected',\n    category: 'location'\n  },\n  {\n    name: 'US_ADDRESS',\n    pattern: /\\b\\d{1,5}\\s+[\\w\\s]+(?:street|st|avenue|ave|road|rd|boulevard|blvd|drive|dr|lane|ln|court|ct)\\b/gi,\n    severity: 'medium',\n    confidence: 0.6,\n    message: 'US street address detected',\n    category: 'location'\n  },\n  {\n    name: 'ZIP_CODE',\n    pattern: /\\b\\d{5}(?:-\\d{4})?\\b/g,\n    severity: 'low',\n    confidence: 0.5,\n    message: 'US ZIP code detected',\n    category: 'location'\n  },\n  \n  // === HEALTH ===\n  {\n    name: 'MEDICAL_RECORD',\n    pattern: /\\b(?:mrn|medical\\s+record)[\\s#:]*\\d{6,12}\\b/gi,\n    severity: 'critical',\n    confidence: 0.8,\n    message: 'Medical record number detected',\n    category: 'health'\n  }\n];\n\n/**\n * Check for PII in content\n * @param content - The content to scan\n * @returns Array of findings\n */\nexport function checkPII(content: string): Finding[] {\n  const findings: Finding[] = [];\n  const foundTypes = new Set<string>();\n  \n  for (const piiPattern of PII_PATTERNS) {\n    // Reset lastIndex\n    piiPattern.pattern.lastIndex = 0;\n    const matches = Array.from(content.matchAll(piiPattern.pattern));\n    \n    for (const match of matches) {\n      // Skip if we already found this type (avoid spam)\n      if (foundTypes.has(piiPattern.name)) continue;\n      \n      // Skip if likely false positive\n      if (isLikelyFalsePositive(content, match[0], piiPattern.name)) continue;\n      \n      foundTypes.add(piiPattern.name);\n      \n      findings.push({\n        id: `PII_${piiPattern.name}`,\n        category: 'privacy',\n        severity: piiPattern.severity,\n        surface: 'output',\n        message: piiPattern.message,\n        recommendation: `Remove or redact ${piiPattern.name.toLowerCase().replace(/_/g, ' ')} before use.`,\n        \n        evidence: {\n          textSample: redactSample(match[0]),\n          pattern: piiPattern.name,\n          context: extractContext(content, match.index || 0, 30)\n        },\n        \n        confidence: calculateConfidence(piiPattern.confidence, match[0]),\n        limitations: LIMITATIONS,\n        methodology: METHODOLOGY,\n        metadata: {\n          piiType: piiPattern.name,\n          piiCategory: piiPattern.category\n        }\n      });\n    }\n  }\n  \n  return findings;\n}\n\n/**\n * Redact all PII from content\n * @param content - The content to redact\n * @param replacement - Replacement string (default: [REDACTED])\n * @returns Redacted content and list of redactions\n */\nexport function redactPII(content: string, replacement = '[REDACTED]'): {\n  redacted: string;\n  redactions: Array<{ type: string; original: string; position: number }>;\n  piiCount: number;\n} {\n  let redacted = content;\n  const redactions: Array<{ type: string; original: string; position: number }> = [];\n  \n  for (const piiPattern of PII_PATTERNS) {\n    piiPattern.pattern.lastIndex = 0;\n    const matches = Array.from(content.matchAll(piiPattern.pattern));\n    \n    for (const match of matches) {\n      if (!isLikelyFalsePositive(content, match[0], piiPattern.name)) {\n        redactions.push({\n          type: piiPattern.name,\n          original: redactSample(match[0]),\n          position: match.index || 0\n        });\n        redacted = redacted.replace(match[0], replacement);\n      }\n    }\n  }\n  \n  return {\n    redacted,\n    redactions,\n    piiCount: redactions.length\n  };\n}\n\n/**\n * Check if content contains any PII\n * @param content - The content to check\n * @returns true if PII detected\n */\nexport function containsPII(content: string): boolean {\n  return checkPII(content).length > 0;\n}\n\n/**\n * Get PII risk score (0-1)\n * @param content - The content to score\n * @returns Risk score\n */\nexport function getPIIRiskScore(content: string): number {\n  const findings = checkPII(content);\n  if (findings.length === 0) return 0;\n  \n  const severityWeights: Record<string, number> = {\n    low: 0.2,\n    medium: 0.4,\n    high: 0.7,\n    critical: 1.0\n  };\n  \n  let maxScore = 0;\n  for (const finding of findings) {\n    const score = severityWeights[finding.severity] * finding.confidence.value;\n    maxScore = Math.max(maxScore, score);\n  }\n  \n  // Add penalty for multiple PII types\n  const typeBonus = Math.min(findings.length * 0.1, 0.3);\n  \n  return Math.min(1, maxScore + typeBonus);\n}\n\n/**\n * Redact sensitive parts of sample for display\n */\nfunction redactSample(text: string): string {\n  if (text.length <= 8) {\n    return text.substring(0, 2) + '***' + text.substring(text.length - 2);\n  }\n  \n  const visible = Math.min(4, Math.floor(text.length / 4));\n  return text.substring(0, visible) + '***' + text.substring(text.length - visible);\n}\n\n/**\n * Calculate confidence score\n */\nfunction calculateConfidence(baseConfidence: number, match: string): ConfidenceScore {\n  let value = baseConfidence;\n  \n  // Adjust based on match characteristics\n  if (match.length > 20) value += 0.05;\n  if (/^[A-Z]/.test(match)) value -= 0.05; // Might be a name, not PII\n  \n  value = Math.max(0.5, Math.min(0.95, value));\n  \n  return {\n    value,\n    interval: [Math.max(0, value - 0.1), Math.min(1, value + 0.1)],\n    method: 'empirical',\n    factors: {\n      patternStrength: baseConfidence\n    }\n  };\n}\n\n/**\n * Check for false positives\n */\nfunction isLikelyFalsePositive(\n  fullText: string, \n  match: string, \n  type: string\n): boolean {\n  // Example/placeholder indicators\n  const placeholders = [\n    /example/i,\n    /placeholder/i,\n    /test/i,\n    /sample/i,\n    /dummy/i,\n    /fake/i,\n    /demo/i,\n    /xxx/i\n  ];\n  \n  const context = fullText.substring(\n    Math.max(0, fullText.indexOf(match) - 50),\n    fullText.indexOf(match) + match.length + 50\n  );\n  \n  if (placeholders.some(p => p.test(context))) {\n    return true;\n  }\n  \n  // SSN false positives\n  if (type === 'SSN') {\n    // Check if it's a date format\n    if (/\\d{2}[-/]\\d{2}[-/]\\d{4}/.test(match)) {\n      return true;\n    }\n    // Check if preceded by date-like context\n    if (/date|time|year|month|day/i.test(context)) {\n      return true;\n    }\n  }\n  \n  // Phone false positives\n  if (type === 'PHONE_US') {\n    // Check if it's a zip code or other number\n    if (/zip|code|id|number/i.test(context) && !/phone|call|mobile|cell/i.test(context)) {\n      return true;\n    }\n  }\n  \n  // ZIP code - too many false positives, require context\n  if (type === 'ZIP_CODE') {\n    if (!/zip|postal|address/i.test(context)) {\n      return true;\n    }\n  }\n  \n  // AWS Secret - high false positive rate\n  if (type === 'AWS_SECRET') {\n    // Only flag if near AWS context\n    if (!/aws|amazon|secret|key/i.test(context)) {\n      return true;\n    }\n  }\n  \n  return false;\n}\n"]}