From b6fcd364a78fe05f7d9e27645afebbe35098f96f Mon Sep 17 00:00:00 2001 From: Claude Date: Mon, 19 Jan 2026 02:17:36 +0000 Subject: [PATCH 1/2] Add comprehensive test suite and feature parity documentation - Set up Jest testing framework for TypeScript with mocked native modules - Add unit tests for NMS (Non-Maximum Suppression) algorithm - Add unit tests for YOLO v8 output parser - Add Android unit tests (Kotlin/JUnit) for NMS and YOLO parser - Add iOS unit tests (Swift/XCTest) for NMS and YOLO parser - Create FEATURE_PARITY.md documenting iOS vs Android feature differences - Update package.json with test scripts and dependencies - Update android/build.gradle with test configuration Test coverage includes: - IoU calculation edge cases - NMS suppression behavior - YOLO coordinate transformation - Platform-specific Y-axis handling (iOS CGContext flip) - Real-world NSFW detection scenarios Feature parity gaps documented: - Scene classification (iOS only - VNClassifyImageRequest) - Rectangle detection (iOS only - VNDetectRectanglesRequest) - Sensitive Content Analysis (iOS 17+ only - no Android equivalent) - Live Activity (iOS full, Android stub) --- FEATURE_PARITY.md | 245 + __tests__/__mocks__/react-native.ts | 106 + __tests__/api.test.ts | 322 ++ __tests__/nms.test.ts | 350 ++ __tests__/setup.ts | 41 + __tests__/yolo-parser.test.ts | 581 +++ android/build.gradle | 12 + android/src/test/java/com/visionml/NMSTest.kt | 175 + .../test/java/com/visionml/YOLOParserTest.kt | 306 ++ ios/VisionMLTests/NMSTests.swift | 171 + ios/VisionMLTests/YOLOParserTests.swift | 290 ++ jest.config.js | 25 + package-lock.json | 3926 ++++++++++++++++- package.json | 14 +- 14 files changed, 6496 insertions(+), 68 deletions(-) create mode 100644 FEATURE_PARITY.md create mode 100644 __tests__/__mocks__/react-native.ts create mode 100644 __tests__/api.test.ts create mode 100644 __tests__/nms.test.ts create mode 100644 __tests__/setup.ts create mode 100644 __tests__/yolo-parser.test.ts create mode 100644 android/src/test/java/com/visionml/NMSTest.kt create mode 100644 android/src/test/java/com/visionml/YOLOParserTest.kt create mode 100644 ios/VisionMLTests/NMSTests.swift create mode 100644 ios/VisionMLTests/YOLOParserTests.swift create mode 100644 jest.config.js diff --git a/FEATURE_PARITY.md b/FEATURE_PARITY.md new file mode 100644 index 0000000..e2d0972 --- /dev/null +++ b/FEATURE_PARITY.md @@ -0,0 +1,245 @@ +# Feature Parity: iOS vs Android + +This document tracks the feature parity between iOS and Android implementations of react-native-vision-ml. + +## Summary + +| Category | iOS | Android | Status | +|----------|-----|---------|--------| +| Core ONNX Inference | ✅ | ✅ | **Full Parity** | +| Video Analysis | ✅ | ✅ | **Full Parity** | +| Vision/ML Kit | ✅ | ⚠️ | **Partial** | +| Sensitive Content Analysis | ✅ | ❌ | **iOS Only** | +| Live Activity | ✅ | ⚠️ | **Partial** | + +--- + +## Detailed Feature Comparison + +### Core ONNX Inference ✅ Full Parity + +| Feature | iOS | Android | Notes | +|---------|-----|---------|-------| +| Model Loading | ✅ CoreML | ✅ NNAPI | Both use hardware acceleration | +| Image Decode | ✅ CGImage | ✅ Bitmap | HEIC/JPEG/PNG supported | +| NCHW Conversion | ✅ | ✅ | Identical tensor format | +| YOLO v8 Parsing | ✅ | ✅ | Same output format | +| NMS Algorithm | ✅ | ✅ | Identical implementation | +| Multiple Detectors | ✅ | ✅ | Concurrent instances supported | +| Resource Cleanup | ✅ | ✅ | dispose/disposeAll methods | + +**Performance:** +- iOS: ~120-150ms per image (CoreML acceleration) +- Android: ~120-150ms per image (NNAPI acceleration) + +### Video Analysis ✅ Full Parity + +All 5 video scan modes are implemented identically: + +| Mode | iOS | Android | Description | +|------|-----|---------|-------------| +| `quick_check` | ✅ | ✅ | 3 frames: start, middle, end | +| `sampled` | ✅ | ✅ | Regular interval sampling | +| `thorough` | ✅ | ✅ | Human detection + ONNX on human frames | +| `binary_search` | ✅ | ✅ | Adaptive region expansion | +| `full_short_circuit` | ✅ | ✅ | Progressive scan with early exit | + +**Frame Extraction:** +- iOS: AVAssetImageGenerator +- Android: MediaMetadataRetriever.getFrameAtTime() + +### Vision Framework / ML Kit ⚠️ Partial Parity + +| Feature | iOS (Vision) | Android (ML Kit) | Status | +|---------|--------------|------------------|--------| +| Animal Detection | ✅ VNRecognizeAnimalsRequest | ✅ Image Labeling | **Parity** | +| Human Pose | ✅ VNDetectHumanBodyPoseRequest | ✅ Pose Detection | **Parity** | +| Face Detection | ✅ VNDetectFaceRectanglesRequest | ✅ Face Detection | **Parity** | +| Text Recognition | ✅ VNRecognizeTextRequest | ✅ Text Recognition | **Parity** | +| Scene Classification | ✅ VNClassifyImageRequest | ❌ Not Available | **Gap** | +| Rectangle Detection | ✅ VNDetectRectanglesRequest | ❌ Not Available | **Gap** | + +#### Gap: Scene Classification +- **iOS**: VNClassifyImageRequest provides ~1000 scene categories +- **Android**: ML Kit doesn't have an equivalent API +- **Workaround**: Would require custom TensorFlow Lite model + +#### Gap: Rectangle Detection +- **iOS**: VNDetectRectanglesRequest for screenshot/document detection +- **Android**: Not implemented +- **Impact**: `likelyScreenshot` field always false on Android + +### Sensitive Content Analysis ❌ iOS Only + +| Feature | iOS | Android | Notes | +|---------|-----|---------|-------| +| Single Image SCA | ✅ iOS 17+ | ❌ | Apple proprietary | +| Batch Image SCA | ✅ iOS 17+ | ❌ | Apple proprietary | +| Video SCA | ✅ iOS 17+ | ❌ | Apple proprietary | +| Batch Video SCA | ✅ iOS 17+ | ❌ | Apple proprietary | +| SCA Settings | ✅ | ❌ | Apple proprietary | + +**Why No Android Support:** +Apple's Sensitive Content Analysis framework is a proprietary iOS 17+ feature with no Android equivalent. Google does not provide a similar on-device content analysis API. + +**Workaround for Android:** +Use the ONNX-based detection which works on both platforms and provides detailed bounding boxes and confidence scores. + +### Live Activity ⚠️ Partial Parity + +| Feature | iOS | Android | Notes | +|---------|-----|---------|-------| +| Start Activity | ✅ Dynamic Island | ⚠️ Stub | Foreground service infrastructure ready | +| Update Progress | ✅ Real-time | ⚠️ Stub | Notification UI not implemented | +| End Activity | ✅ Final status | ⚠️ Stub | | + +**Android Implementation Status:** +- Foreground service infrastructure is in place +- Notification progress UI is stubbed (always returns success but no visible UI) +- Full implementation requires custom notification layout + +--- + +## API Compatibility + +### Methods with Full Parity + +```typescript +// These work identically on both platforms +createDetector(modelPath, classLabels, inputSize) +detect(detectorId, imageUri, options) +disposeDetector(detectorId) +disposeAllDetectors() +analyzeVideo(detectorId, assetId, options) +quickCheckVideo(detectorId, assetId, threshold) +analyzeAnimals(assetId) +analyzeHumanPose(assetId) +``` + +### Methods with Platform Differences + +```typescript +// Works but may return partial results on Android +analyzeComprehensive(assetId) +// - Android: scenes=[], rectangles=0, likelyScreenshot=false + +// iOS 17+ only - Android returns {available: false} +getSensitiveContentAnalysisStatus() +analyzeSensitiveContent(assetId) +batchAnalyzeSensitiveContent(assetIds) +analyzeVideoSensitiveContent(assetId) +batchAnalyzeVideosSensitiveContent(assetIds) + +// iOS 16.1+ only - Android returns stub responses +isLiveActivityAvailable() +startVideoScanActivity(name, duration, mode) +updateVideoScanActivity(progress, phase, count, frames) +endVideoScanActivity(count, frames, isNSFW) +``` + +--- + +## Recommended Usage Patterns + +### Cross-Platform Detection + +```typescript +// Works on both iOS and Android +const detector = await createDetector(modelPath, labels, 320); +const result = await detect(detector.detectorId, imageUri, { + confidenceThreshold: 0.6, + iouThreshold: 0.45, +}); +``` + +### Platform-Specific Features + +```typescript +import { Platform } from 'react-native'; + +// Use SCA on iOS 17+, fall back to ONNX on Android +if (Platform.OS === 'ios') { + const status = await getSensitiveContentAnalysisStatus(); + if (status.available && status.enabled) { + const result = await analyzeSensitiveContent(assetId); + if (result.isSensitive) { + // Run ONNX for detailed bounding boxes + const detailed = await detect(detectorId, imageUri); + } + } +} else { + // Android: Use ONNX directly + const result = await detect(detectorId, imageUri); +} +``` + +### Comprehensive Analysis + +```typescript +const analysis = await analyzeComprehensive(assetId); + +// Handle platform differences +const hasSceneData = Platform.OS === 'ios' && analysis.scenes?.length > 0; +const isScreenshot = Platform.OS === 'ios' && analysis.likelyScreenshot; +``` + +--- + +## Test Coverage + +### TypeScript Tests (Jest) +- API integration tests +- NMS algorithm unit tests +- YOLO parser unit tests +- Platform parity verification + +Run with: `npm test` + +### Android Tests (JUnit) +- NMS algorithm tests +- YOLO parser tests +- Coordinate transformation tests + +Run with: `./gradlew test` in android/ + +### iOS Tests (XCTest) +- NMS algorithm tests +- YOLO parser tests +- Coordinate transformation tests + +Run with: Xcode Test Navigator + +--- + +## Roadmap + +### Planned Improvements + +1. **Android Live Activity** (P1) + - Implement notification progress UI + - Match iOS Dynamic Island UX + +2. **Android Scene Classification** (P2) + - Research custom TensorFlow Lite models + - Consider Google Cloud Vision API fallback + +3. **Android Rectangle Detection** (P3) + - Implement ML Kit custom model integration + - Or use OpenCV-based detection + +### Not Planned + +- **Android SCA**: No equivalent API exists; use ONNX detection instead + +--- + +## Version History + +| Version | Changes | +|---------|---------| +| 1.0.5 | Video SCA fixes | +| 1.0.4 | Added video SCA support | +| 1.0.3 | Added batch SCA support | +| 1.0.2 | Full Android implementation | +| 1.0.1 | Initial Android stubs | +| 1.0.0 | Initial iOS-only release | diff --git a/__tests__/__mocks__/react-native.ts b/__tests__/__mocks__/react-native.ts new file mode 100644 index 0000000..8944192 --- /dev/null +++ b/__tests__/__mocks__/react-native.ts @@ -0,0 +1,106 @@ +/** + * Mock React Native NativeModules for testing + */ + +export const NativeModules = { + VisionML: { + createDetector: jest.fn().mockResolvedValue({ + detectorId: 'test-detector-id', + success: true, + message: 'Detector created successfully', + }), + detect: jest.fn().mockResolvedValue({ + detections: [], + inferenceTime: 50, + postProcessTime: 10, + totalTime: 60, + }), + disposeDetector: jest.fn().mockResolvedValue({ success: true }), + disposeAllDetectors: jest.fn().mockResolvedValue({ success: true }), + isLiveActivityAvailable: jest.fn().mockResolvedValue(false), + startVideoScanActivity: jest.fn().mockResolvedValue({ + activityId: null, + success: false, + }), + updateVideoScanActivity: jest.fn().mockResolvedValue(false), + endVideoScanActivity: jest.fn().mockResolvedValue(false), + analyzeVideo: jest.fn().mockResolvedValue({ + isNSFW: false, + nsfwFrameCount: 0, + totalFramesAnalyzed: 3, + firstNSFWTimestamp: null, + nsfwTimestamps: [], + highestConfidence: 0, + totalProcessingTime: 100, + videoDuration: 30, + scanMode: 'sampled', + humanFramesDetected: 0, + }), + quickCheckVideo: jest.fn().mockResolvedValue({ + isNSFW: false, + nsfwFrameCount: 0, + totalFramesAnalyzed: 3, + firstNSFWTimestamp: null, + nsfwTimestamps: [], + highestConfidence: 0, + totalProcessingTime: 50, + videoDuration: 30, + scanMode: 'quick_check', + humanFramesDetected: 0, + }), + analyzeAnimals: jest.fn().mockResolvedValue({ + animals: [], + count: 0, + }), + analyzeHumanPose: jest.fn().mockResolvedValue({ + humans: [], + humanCount: 0, + }), + analyzeComprehensive: jest.fn().mockResolvedValue({ + scenes: [], + faces: [], + faceCount: 0, + animals: [], + animalCount: 0, + humanCount: 0, + hasHumans: false, + hasText: false, + textRegions: 0, + rectangles: 0, + likelyScreenshot: false, + }), + getSensitiveContentAnalysisStatus: jest.fn().mockResolvedValue({ + available: false, + enabled: false, + policy: 'unsupported', + iosVersion: '16.0', + reason: 'Requires iOS 17+', + }), + openSensitiveContentSettings: jest.fn().mockResolvedValue({ + opened: false, + url: null, + }), + analyzeSensitiveContent: jest.fn().mockResolvedValue({ + available: false, + isSensitive: false, + reason: 'ios_version', + }), + batchAnalyzeSensitiveContent: jest.fn().mockResolvedValue({ + available: false, + results: [], + reason: 'ios_version', + }), + analyzeVideoSensitiveContent: jest.fn().mockResolvedValue({ + available: false, + isSensitive: false, + reason: 'ios_version', + }), + batchAnalyzeVideosSensitiveContent: jest.fn().mockResolvedValue({ + available: false, + results: [], + reason: 'ios_version', + }), + }, +}; + +export default { NativeModules }; diff --git a/__tests__/api.test.ts b/__tests__/api.test.ts new file mode 100644 index 0000000..e6cdf41 --- /dev/null +++ b/__tests__/api.test.ts @@ -0,0 +1,322 @@ +/** + * API Integration Tests for react-native-vision-ml + * Tests the TypeScript API interface and native module bridge + */ + +import { + createDetector, + detect, + disposeDetector, + disposeAllDetectors, + analyzeVideo, + quickCheckVideo, + analyzeAnimals, + analyzeHumanPose, + analyzeComprehensive, + getSensitiveContentAnalysisStatus, + analyzeSensitiveContent, + batchAnalyzeSensitiveContent, + analyzeVideoSensitiveContent, + batchAnalyzeVideosSensitiveContent, + isLiveActivityAvailable, + startVideoScanActivity, + updateVideoScanActivity, + endVideoScanActivity, +} from '../src/index'; + +import { NativeModules } from 'react-native'; + +describe('VisionML API', () => { + beforeEach(() => { + jest.clearAllMocks(); + }); + + describe('Detector Lifecycle', () => { + it('should create a detector with default input size', async () => { + const result = await createDetector('/path/to/model.onnx', ['class1', 'class2']); + + expect(result).toEqual({ + detectorId: 'test-detector-id', + success: true, + message: 'Detector created successfully', + }); + expect(NativeModules.VisionML.createDetector).toHaveBeenCalledWith( + '/path/to/model.onnx', + ['class1', 'class2'], + 320 + ); + }); + + it('should create a detector with custom input size', async () => { + await createDetector('/path/to/model.onnx', ['class1'], 640); + + expect(NativeModules.VisionML.createDetector).toHaveBeenCalledWith( + '/path/to/model.onnx', + ['class1'], + 640 + ); + }); + + it('should dispose a specific detector', async () => { + const result = await disposeDetector('test-detector-id'); + + expect(result).toEqual({ success: true }); + expect(NativeModules.VisionML.disposeDetector).toHaveBeenCalledWith('test-detector-id'); + }); + + it('should dispose all detectors', async () => { + const result = await disposeAllDetectors(); + + expect(result).toEqual({ success: true }); + expect(NativeModules.VisionML.disposeAllDetectors).toHaveBeenCalled(); + }); + }); + + describe('Image Detection', () => { + it('should detect with default options', async () => { + const result = await detect('test-detector-id', 'file:///image.jpg'); + + expect(result).toHaveProperty('detections'); + expect(result).toHaveProperty('inferenceTime'); + expect(result).toHaveProperty('postProcessTime'); + expect(result).toHaveProperty('totalTime'); + expect(NativeModules.VisionML.detect).toHaveBeenCalledWith( + 'test-detector-id', + 'file:///image.jpg', + 0.6, // default confidence + 0.45 // default IoU + ); + }); + + it('should detect with custom thresholds', async () => { + await detect('test-detector-id', 'file:///image.jpg', { + confidenceThreshold: 0.8, + iouThreshold: 0.5, + }); + + expect(NativeModules.VisionML.detect).toHaveBeenCalledWith( + 'test-detector-id', + 'file:///image.jpg', + 0.8, + 0.5 + ); + }); + }); + + describe('Video Analysis', () => { + it('should analyze video with default options', async () => { + const result = await analyzeVideo('test-detector-id', 'asset-123'); + + expect(result).toHaveProperty('isNSFW'); + expect(result).toHaveProperty('nsfwFrameCount'); + expect(result).toHaveProperty('totalFramesAnalyzed'); + expect(result).toHaveProperty('scanMode'); + expect(NativeModules.VisionML.analyzeVideo).toHaveBeenCalledWith( + 'test-detector-id', + 'asset-123', + 'sampled', // default mode + 5.0, // default sample interval + 0.6 // default confidence + ); + }); + + it('should analyze video with thorough mode', async () => { + await analyzeVideo('test-detector-id', 'asset-123', { + mode: 'thorough', + sampleInterval: 2.0, + confidenceThreshold: 0.7, + }); + + expect(NativeModules.VisionML.analyzeVideo).toHaveBeenCalledWith( + 'test-detector-id', + 'asset-123', + 'thorough', + 2.0, + 0.7 + ); + }); + + it('should quick check video', async () => { + const result = await quickCheckVideo('test-detector-id', 'asset-123'); + + expect(result.scanMode).toBe('quick_check'); + expect(result.totalFramesAnalyzed).toBe(3); + expect(NativeModules.VisionML.quickCheckVideo).toHaveBeenCalledWith( + 'test-detector-id', + 'asset-123', + 0.6 + ); + }); + + it('should quick check video with custom threshold', async () => { + await quickCheckVideo('test-detector-id', 'asset-123', 0.8); + + expect(NativeModules.VisionML.quickCheckVideo).toHaveBeenCalledWith( + 'test-detector-id', + 'asset-123', + 0.8 + ); + }); + }); + + describe('Vision Framework Analysis', () => { + it('should analyze animals', async () => { + const result = await analyzeAnimals('asset-123'); + + expect(result).toHaveProperty('animals'); + expect(result).toHaveProperty('count'); + expect(NativeModules.VisionML.analyzeAnimals).toHaveBeenCalledWith('asset-123'); + }); + + it('should analyze human pose', async () => { + const result = await analyzeHumanPose('asset-123'); + + expect(result).toHaveProperty('humans'); + expect(result).toHaveProperty('humanCount'); + expect(NativeModules.VisionML.analyzeHumanPose).toHaveBeenCalledWith('asset-123'); + }); + + it('should analyze comprehensive', async () => { + const result = await analyzeComprehensive('asset-123'); + + expect(result).toHaveProperty('scenes'); + expect(result).toHaveProperty('faces'); + expect(result).toHaveProperty('animals'); + expect(result).toHaveProperty('humanCount'); + expect(result).toHaveProperty('hasText'); + expect(result).toHaveProperty('rectangles'); + expect(result).toHaveProperty('likelyScreenshot'); + expect(NativeModules.VisionML.analyzeComprehensive).toHaveBeenCalledWith('asset-123'); + }); + }); + + describe('Sensitive Content Analysis (iOS 17+)', () => { + it('should get SCA status', async () => { + const result = await getSensitiveContentAnalysisStatus(); + + expect(result).toHaveProperty('available'); + expect(result).toHaveProperty('enabled'); + expect(result).toHaveProperty('policy'); + expect(result).toHaveProperty('iosVersion'); + }); + + it('should analyze single image for sensitive content', async () => { + const result = await analyzeSensitiveContent('asset-123'); + + expect(result).toHaveProperty('available'); + expect(result).toHaveProperty('isSensitive'); + expect(NativeModules.VisionML.analyzeSensitiveContent).toHaveBeenCalledWith('asset-123'); + }); + + it('should batch analyze images for sensitive content', async () => { + const assetIds = ['asset-1', 'asset-2', 'asset-3']; + const result = await batchAnalyzeSensitiveContent(assetIds); + + expect(result).toHaveProperty('available'); + expect(result).toHaveProperty('results'); + expect(NativeModules.VisionML.batchAnalyzeSensitiveContent).toHaveBeenCalledWith(assetIds); + }); + + it('should analyze video for sensitive content', async () => { + const result = await analyzeVideoSensitiveContent('asset-123'); + + expect(result).toHaveProperty('available'); + expect(result).toHaveProperty('isSensitive'); + expect(NativeModules.VisionML.analyzeVideoSensitiveContent).toHaveBeenCalledWith('asset-123'); + }); + + it('should batch analyze videos for sensitive content', async () => { + const assetIds = ['video-1', 'video-2']; + const result = await batchAnalyzeVideosSensitiveContent(assetIds); + + expect(result).toHaveProperty('available'); + expect(result).toHaveProperty('results'); + expect(NativeModules.VisionML.batchAnalyzeVideosSensitiveContent).toHaveBeenCalledWith(assetIds); + }); + }); + + describe('Live Activity (iOS 16.1+)', () => { + it('should check live activity availability', async () => { + const result = await isLiveActivityAvailable(); + + expect(typeof result).toBe('boolean'); + expect(NativeModules.VisionML.isLiveActivityAvailable).toHaveBeenCalled(); + }); + + it('should start video scan activity', async () => { + const result = await startVideoScanActivity('test-video.mp4', 120, 'thorough'); + + expect(result).toHaveProperty('activityId'); + expect(result).toHaveProperty('success'); + expect(NativeModules.VisionML.startVideoScanActivity).toHaveBeenCalledWith( + 'test-video.mp4', + 120, + 'thorough' + ); + }); + + it('should update video scan activity', async () => { + const result = await updateVideoScanActivity(0.5, 'Scanning frames', 2, 50); + + expect(typeof result).toBe('boolean'); + expect(NativeModules.VisionML.updateVideoScanActivity).toHaveBeenCalledWith( + 0.5, + 'Scanning frames', + 2, + 50 + ); + }); + + it('should end video scan activity', async () => { + const result = await endVideoScanActivity(5, 100, true); + + expect(typeof result).toBe('boolean'); + expect(NativeModules.VisionML.endVideoScanActivity).toHaveBeenCalledWith(5, 100, true); + }); + }); +}); + +describe('Type Definitions', () => { + it('should export all video scan modes', () => { + const modes: string[] = [ + 'quick_check', + 'sampled', + 'thorough', + 'binary_search', + 'full_short_circuit', + ]; + + // Type check - these should all be valid VideoScanMode values + modes.forEach(mode => { + expect(typeof mode).toBe('string'); + }); + }); + + it('should have correct Detection interface shape', () => { + const detection = { + box: [0, 0, 100, 100] as [number, number, number, number], + score: 0.95, + classIndex: 0, + className: 'test-class', + }; + + expect(detection.box).toHaveLength(4); + expect(typeof detection.score).toBe('number'); + expect(typeof detection.classIndex).toBe('number'); + expect(typeof detection.className).toBe('string'); + }); + + it('should have correct BoundingBox interface shape', () => { + const boundingBox = { + x: 10, + y: 20, + width: 100, + height: 150, + }; + + expect(typeof boundingBox.x).toBe('number'); + expect(typeof boundingBox.y).toBe('number'); + expect(typeof boundingBox.width).toBe('number'); + expect(typeof boundingBox.height).toBe('number'); + }); +}); diff --git a/__tests__/nms.test.ts b/__tests__/nms.test.ts new file mode 100644 index 0000000..99ffc74 --- /dev/null +++ b/__tests__/nms.test.ts @@ -0,0 +1,350 @@ +/** + * Unit Tests for Non-Maximum Suppression (NMS) Algorithm + * + * These tests verify the NMS implementation that should be consistent + * across iOS (Swift) and Android (Kotlin) platforms. + * + * NMS algorithm: + * 1. Sort detections by confidence score (highest first) + * 2. Keep the highest scoring detection + * 3. Remove all other detections that have IoU > threshold with the kept detection + * 4. Repeat until no detections remain + */ + +interface Detection { + box: [number, number, number, number]; // [x1, y1, x2, y2] + score: number; + classIndex: number; + className: string; +} + +/** + * Calculate Intersection over Union (IoU) between two boxes + */ +function calculateIoU(box1: number[], box2: number[]): number { + const x1_1 = box1[0]; + const y1_1 = box1[1]; + const x2_1 = box1[2]; + const y2_1 = box1[3]; + + const x1_2 = box2[0]; + const y1_2 = box2[1]; + const x2_2 = box2[2]; + const y2_2 = box2[3]; + + // Calculate intersection + const x1_i = Math.max(x1_1, x1_2); + const y1_i = Math.max(y1_1, y1_2); + const x2_i = Math.min(x2_1, x2_2); + const y2_i = Math.min(y2_1, y2_2); + + const intersectionWidth = Math.max(0, x2_i - x1_i); + const intersectionHeight = Math.max(0, y2_i - y1_i); + const intersection = intersectionWidth * intersectionHeight; + + // Calculate union + const area1 = (x2_1 - x1_1) * (y2_1 - y1_1); + const area2 = (x2_2 - x1_2) * (y2_2 - y1_2); + const union = area1 + area2 - intersection; + + return union > 0 ? intersection / union : 0; +} + +/** + * Apply NMS to filter overlapping detections + */ +function applyNMS(detections: Detection[], iouThreshold: number = 0.45): Detection[] { + if (detections.length === 0) return []; + + // Sort by confidence score (highest first) + const sorted = [...detections].sort((a, b) => b.score - a.score); + const keep: Detection[] = []; + const suppressed = new Set(); + + for (let i = 0; i < sorted.length; i++) { + if (suppressed.has(i)) continue; + keep.push(sorted[i]); + + // Suppress overlapping detections + for (let j = i + 1; j < sorted.length; j++) { + if (suppressed.has(j)) continue; + + const iou = calculateIoU(sorted[i].box, sorted[j].box); + if (iou > iouThreshold) { + suppressed.add(j); + } + } + } + + return keep; +} + +describe('NMS Algorithm', () => { + describe('IoU Calculation', () => { + it('should return 1.0 for identical boxes', () => { + const box = [0, 0, 100, 100]; + const iou = calculateIoU(box, box); + expect(iou).toBe(1.0); + }); + + it('should return 0.0 for non-overlapping boxes', () => { + const box1 = [0, 0, 50, 50]; + const box2 = [100, 100, 150, 150]; + const iou = calculateIoU(box1, box2); + expect(iou).toBe(0.0); + }); + + it('should calculate correct IoU for partially overlapping boxes', () => { + const box1 = [0, 0, 100, 100]; // Area = 10000 + const box2 = [50, 50, 150, 150]; // Area = 10000 + // Intersection: [50, 50, 100, 100] = 50 * 50 = 2500 + // Union: 10000 + 10000 - 2500 = 17500 + // IoU: 2500 / 17500 = 0.1428... + const iou = calculateIoU(box1, box2); + expect(iou).toBeCloseTo(0.1429, 3); + }); + + it('should handle boxes touching at edge (no overlap)', () => { + const box1 = [0, 0, 100, 100]; + const box2 = [100, 0, 200, 100]; + const iou = calculateIoU(box1, box2); + expect(iou).toBe(0.0); + }); + + it('should handle one box inside another', () => { + const outer = [0, 0, 100, 100]; // Area = 10000 + const inner = [25, 25, 75, 75]; // Area = 2500 + // Intersection = 2500 (inner box entirely inside) + // Union = 10000 + 2500 - 2500 = 10000 + // IoU = 2500 / 10000 = 0.25 + const iou = calculateIoU(outer, inner); + expect(iou).toBeCloseTo(0.25, 3); + }); + + it('should handle zero-area boxes', () => { + const point = [50, 50, 50, 50]; // Zero area + const box = [0, 0, 100, 100]; + const iou = calculateIoU(point, box); + expect(iou).toBe(0); + }); + + it('should be symmetric', () => { + const box1 = [10, 20, 80, 90]; + const box2 = [30, 40, 100, 100]; + const iou1 = calculateIoU(box1, box2); + const iou2 = calculateIoU(box2, box1); + expect(iou1).toBe(iou2); + }); + }); + + describe('NMS Application', () => { + it('should return empty array for empty input', () => { + const result = applyNMS([]); + expect(result).toEqual([]); + }); + + it('should return single detection unchanged', () => { + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'test' }, + ]; + const result = applyNMS(detections); + expect(result).toHaveLength(1); + expect(result[0]).toEqual(detections[0]); + }); + + it('should keep non-overlapping detections', () => { + const detections: Detection[] = [ + { box: [0, 0, 50, 50], score: 0.9, classIndex: 0, className: 'test' }, + { box: [100, 100, 150, 150], score: 0.8, classIndex: 0, className: 'test' }, + ]; + const result = applyNMS(detections); + expect(result).toHaveLength(2); + }); + + it('should suppress overlapping detections with lower scores', () => { + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'test' }, + { box: [10, 10, 110, 110], score: 0.7, classIndex: 0, className: 'test' }, // Highly overlapping + { box: [5, 5, 105, 105], score: 0.8, classIndex: 0, className: 'test' }, // Highly overlapping + ]; + const result = applyNMS(detections, 0.5); + expect(result).toHaveLength(1); + expect(result[0].score).toBe(0.9); // Highest score kept + }); + + it('should keep detections below IoU threshold', () => { + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'test' }, + { box: [50, 50, 150, 150], score: 0.8, classIndex: 0, className: 'test' }, // ~14% overlap + ]; + const result = applyNMS(detections, 0.5); + expect(result).toHaveLength(2); // Both kept because overlap is below threshold + }); + + it('should sort by score before applying NMS', () => { + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.5, classIndex: 0, className: 'test' }, + { box: [5, 5, 105, 105], score: 0.9, classIndex: 0, className: 'test' }, // Higher score, should be kept + { box: [10, 10, 110, 110], score: 0.7, classIndex: 0, className: 'test' }, + ]; + const result = applyNMS(detections, 0.5); + expect(result).toHaveLength(1); + expect(result[0].score).toBe(0.9); // Highest score should be kept + }); + + it('should handle multiple non-overlapping groups', () => { + const detections: Detection[] = [ + // Group 1 - top left + { box: [0, 0, 50, 50], score: 0.9, classIndex: 0, className: 'A' }, + { box: [5, 5, 55, 55], score: 0.8, classIndex: 0, className: 'A' }, + // Group 2 - bottom right (non-overlapping with group 1) + { box: [200, 200, 250, 250], score: 0.85, classIndex: 1, className: 'B' }, + { box: [205, 205, 255, 255], score: 0.7, classIndex: 1, className: 'B' }, + ]; + const result = applyNMS(detections, 0.5); + expect(result).toHaveLength(2); + expect(result.map(d => d.score).sort((a, b) => b - a)).toEqual([0.9, 0.85]); + }); + + it('should use default IoU threshold of 0.45', () => { + // Create boxes with IoU just above and below 0.45 + // IoU = 0.45 means significant overlap + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'test' }, + { box: [35, 35, 135, 135], score: 0.8, classIndex: 0, className: 'test' }, // ~30% overlap, kept + ]; + const result = applyNMS(detections); // Uses default 0.45 + expect(result).toHaveLength(2); + }); + + it('should handle different classes independently (class-agnostic NMS)', () => { + // This tests that NMS considers all classes together (class-agnostic) + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'cat' }, + { box: [5, 5, 105, 105], score: 0.8, classIndex: 1, className: 'dog' }, // Different class but overlapping + ]; + const result = applyNMS(detections, 0.5); + // In class-agnostic NMS, the lower-scoring overlapping detection is suppressed + // regardless of class + expect(result).toHaveLength(1); + }); + + it('should handle real-world confidence scores', () => { + const detections: Detection[] = [ + { box: [156.2, 89.4, 312.8, 445.1], score: 0.873, classIndex: 3, className: 'FEMALE_BREAST_EXPOSED' }, + { box: [158.1, 91.2, 310.5, 442.3], score: 0.821, classIndex: 3, className: 'FEMALE_BREAST_EXPOSED' }, + { box: [155.0, 88.0, 315.0, 448.0], score: 0.756, classIndex: 3, className: 'FEMALE_BREAST_EXPOSED' }, + { box: [450.5, 200.3, 550.7, 380.9], score: 0.945, classIndex: 11, className: 'FACE_FEMALE' }, + ]; + const result = applyNMS(detections, 0.45); + expect(result).toHaveLength(2); // One from breast group, one face (non-overlapping) + expect(result.find(d => d.className === 'FEMALE_BREAST_EXPOSED')?.score).toBe(0.873); + expect(result.find(d => d.className === 'FACE_FEMALE')?.score).toBe(0.945); + }); + }); + + describe('Edge Cases', () => { + it('should handle very small boxes', () => { + const detections: Detection[] = [ + { box: [100, 100, 102, 102], score: 0.9, classIndex: 0, className: 'test' }, // 2x2 pixels + { box: [101, 101, 103, 103], score: 0.8, classIndex: 0, className: 'test' }, // 2x2, 1x1 overlap + ]; + const result = applyNMS(detections, 0.5); + // 1x1 overlap, areas 4+4, union=7, IoU=1/7=0.14 < 0.5, both kept + expect(result).toHaveLength(2); + + // Test with higher overlap + const highOverlap: Detection[] = [ + { box: [100, 100, 110, 110], score: 0.9, classIndex: 0, className: 'test' }, // 10x10 + { box: [102, 102, 112, 112], score: 0.8, classIndex: 0, className: 'test' }, // 10x10, 8x8 overlap + ]; + const highResult = applyNMS(highOverlap, 0.5); + // 64 overlap, 100+100-64=136 union, IoU=64/136=0.47 < 0.5, both kept + // But with threshold 0.45: 64/136=0.47 > 0.45, one suppressed + const suppressedResult = applyNMS(highOverlap, 0.45); + expect(suppressedResult).toHaveLength(1); + }); + + it('should handle boxes at image boundaries', () => { + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'test' }, + { box: [1920 - 100, 1080 - 100, 1920, 1080], score: 0.85, classIndex: 0, className: 'test' }, + ]; + const result = applyNMS(detections, 0.45); + expect(result).toHaveLength(2); // Non-overlapping + }); + + it('should handle many detections efficiently', () => { + const detections: Detection[] = []; + for (let i = 0; i < 100; i++) { + // Create non-overlapping grid of detections + const x = (i % 10) * 100; + const y = Math.floor(i / 10) * 100; + detections.push({ + box: [x, y, x + 50, y + 50], + score: 0.5 + Math.random() * 0.5, + classIndex: 0, + className: 'test', + }); + } + const result = applyNMS(detections, 0.45); + expect(result.length).toBe(100); // All non-overlapping, all kept + }); + + it('should handle equal scores by keeping first encountered', () => { + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: 'first' }, + { box: [5, 5, 105, 105], score: 0.9, classIndex: 0, className: 'second' }, + ]; + const result = applyNMS(detections, 0.5); + expect(result).toHaveLength(1); + // The first one in sorted order (stable sort) should be kept + expect(result[0].className).toBe('first'); + }); + + it('should handle negative coordinates', () => { + // Some coordinate systems allow negative values + const detections: Detection[] = [ + { box: [-50, -50, 50, 50], score: 0.9, classIndex: 0, className: 'test' }, + { box: [-40, -40, 60, 60], score: 0.8, classIndex: 0, className: 'test' }, + ]; + const result = applyNMS(detections, 0.5); + expect(result).toHaveLength(1); + expect(result[0].score).toBe(0.9); + }); + }); +}); + +describe('Platform Parity: NMS Implementation', () => { + it('should match iOS Swift implementation behavior', () => { + // Test case from iOS implementation + const detections: Detection[] = [ + { box: [10, 20, 110, 120], score: 0.95, classIndex: 0, className: 'test' }, + { box: [15, 25, 115, 125], score: 0.85, classIndex: 0, className: 'test' }, + { box: [200, 200, 300, 300], score: 0.90, classIndex: 1, className: 'test2' }, + ]; + + const result = applyNMS(detections, 0.45); + + // iOS behavior: sort by score, suppress high-overlap boxes + expect(result).toHaveLength(2); + expect(result[0].score).toBe(0.95); + expect(result[1].score).toBe(0.90); + }); + + it('should match Android Kotlin implementation behavior', () => { + // Test case from Android implementation + const detections: Detection[] = [ + { box: [0, 0, 100, 100], score: 0.9, classIndex: 2, className: 'BUTTOCKS_EXPOSED' }, + { box: [90, 90, 190, 190], score: 0.8, classIndex: 2, className: 'BUTTOCKS_EXPOSED' }, + { box: [95, 95, 195, 195], score: 0.7, classIndex: 2, className: 'BUTTOCKS_EXPOSED' }, + ]; + + const result = applyNMS(detections, 0.45); + + // First and second have ~1% overlap, both kept + // Second and third have ~90% overlap, third suppressed + expect(result).toHaveLength(2); + expect(result.map(d => d.score)).toEqual([0.9, 0.8]); + }); +}); diff --git a/__tests__/setup.ts b/__tests__/setup.ts new file mode 100644 index 0000000..96d358b --- /dev/null +++ b/__tests__/setup.ts @@ -0,0 +1,41 @@ +/** + * Jest test setup file + */ + +// Extend Jest matchers +expect.extend({ + toBeWithinRange(received: number, floor: number, ceiling: number) { + const pass = received >= floor && received <= ceiling; + if (pass) { + return { + message: () => + `expected ${received} not to be within range ${floor} - ${ceiling}`, + pass: true, + }; + } else { + return { + message: () => + `expected ${received} to be within range ${floor} - ${ceiling}`, + pass: false, + }; + } + }, +}); + +// Silence console.log during tests unless debugging +if (!process.env.DEBUG) { + jest.spyOn(console, 'log').mockImplementation(() => {}); +} + +// Global test timeout +jest.setTimeout(10000); + +declare global { + namespace jest { + interface Matchers { + toBeWithinRange(floor: number, ceiling: number): R; + } + } +} + +export {}; diff --git a/__tests__/yolo-parser.test.ts b/__tests__/yolo-parser.test.ts new file mode 100644 index 0000000..c6d22cd --- /dev/null +++ b/__tests__/yolo-parser.test.ts @@ -0,0 +1,581 @@ +/** + * Unit Tests for YOLO v8 Parser + * + * These tests verify the YOLO output parsing that should be consistent + * across iOS (Swift) and Android (Kotlin) platforms. + * + * YOLOv8 Output Format: + * - Shape: [1, 4+numClasses, numPredictions] + * - Box format: [cx, cy, w, h] (center coordinates) + * - Class scores are direct (no objectness multiplication) + */ + +interface Detection { + box: [number, number, number, number]; // [x1, y1, x2, y2] + score: number; + classIndex: number; + className: string; +} + +// NudeNet class labels (18 classes) +const NUDENET_LABELS = [ + 'FEMALE_GENITALIA_COVERED', // 0 + 'FACE_FEMALE', // 1 + 'BUTTOCKS_EXPOSED', // 2 + 'FEMALE_BREAST_EXPOSED', // 3 + 'FEMALE_GENITALIA_EXPOSED', // 4 + 'MALE_BREAST_EXPOSED', // 5 + 'ANUS_EXPOSED', // 6 + 'FEET_EXPOSED', // 7 + 'BELLY_COVERED', // 8 + 'FEET_COVERED', // 9 + 'ARMPITS_COVERED', // 10 + 'ARMPITS_EXPOSED', // 11 + 'FACE_MALE', // 12 + 'BELLY_EXPOSED', // 13 + 'MALE_GENITALIA_EXPOSED', // 14 + 'ANUS_COVERED', // 15 + 'FEMALE_BREAST_COVERED', // 16 + 'BUTTOCKS_COVERED', // 17 +]; + +// NSFW class indices +const NSFW_CLASS_INDICES = new Set([2, 3, 4, 6, 14]); + +/** + * TypeScript port of YOLO parser for testing + */ +class YOLOParser { + private classLabels: string[]; + private inputSize: number; + private numClasses: number; + + constructor(classLabels: string[], inputSize: number = 320) { + this.classLabels = classLabels; + this.inputSize = inputSize; + this.numClasses = classLabels.length; + } + + /** + * Parse YOLO v8 output tensor + * Note: iOS flips Y-axis due to CGContext origin, Android doesn't + * This version matches Android behavior (standard image coordinates) + */ + parse( + output: number[], + confidenceThreshold: number, + originalWidth: number, + originalHeight: number, + flipY: boolean = false // iOS uses true, Android uses false + ): Detection[] { + const detections: Detection[] = []; + const valuesPerPrediction = 4 + this.numClasses; + const numPredictions = Math.floor(output.length / valuesPerPrediction); + + // Letterboxing: pad to square, resize to inputSize + const maxDim = Math.max(originalWidth, originalHeight); + const scale = maxDim / this.inputSize; + + for (let i = 0; i < numPredictions; i++) { + // Extract center coordinates + const cx = output[0 * numPredictions + i]; + const cy = output[1 * numPredictions + i]; + const w = output[2 * numPredictions + i]; + const h = output[3 * numPredictions + i]; + + // Find best class + let maxScore = 0; + let bestClassIdx = 0; + + for (let c = 0; c < this.numClasses; c++) { + const score = output[(4 + c) * numPredictions + i]; + if (score > maxScore) { + maxScore = score; + bestClassIdx = c; + } + } + + if (maxScore < confidenceThreshold) continue; + + // Convert center to corner format + let x1 = cx - w / 2; + let y1 = cy - h / 2; + let x2 = cx + w / 2; + let y2 = cy + h / 2; + + // Optional Y-flip for iOS CGContext origin + if (flipY) { + const y1_flipped = this.inputSize - y2; + const y2_flipped = this.inputSize - y1; + y1 = y1_flipped; + y2 = y2_flipped; + } + + // Scale to original image coordinates + x1 *= scale; + y1 *= scale; + x2 *= scale; + y2 *= scale; + + // Clip to image boundaries + x1 = Math.max(0, Math.min(x1, originalWidth)); + y1 = Math.max(0, Math.min(y1, originalHeight)); + x2 = Math.max(0, Math.min(x2, originalWidth)); + y2 = Math.max(0, Math.min(y2, originalHeight)); + + // Skip invalid boxes + const boxWidth = x2 - x1; + const boxHeight = y2 - y1; + if (boxWidth <= 0 || boxHeight <= 0) continue; + if (boxWidth < 10 && boxHeight < 10) continue; + + detections.push({ + box: [x1, y1, x2, y2], + score: maxScore, + classIndex: bestClassIdx, + className: this.classLabels[bestClassIdx], + }); + } + + return detections; + } + + /** + * Parse ALL classes above threshold (not just best class) + */ + parseAllClasses( + output: number[], + confidenceThreshold: number, + originalWidth: number, + originalHeight: number, + flipY: boolean = false + ): Detection[] { + const detections: Detection[] = []; + const valuesPerPrediction = 4 + this.numClasses; + const numPredictions = Math.floor(output.length / valuesPerPrediction); + + const maxDim = Math.max(originalWidth, originalHeight); + const scale = maxDim / this.inputSize; + + for (let i = 0; i < numPredictions; i++) { + const cx = output[0 * numPredictions + i]; + const cy = output[1 * numPredictions + i]; + const w = output[2 * numPredictions + i]; + const h = output[3 * numPredictions + i]; + + // Check ALL classes + for (let c = 0; c < this.numClasses; c++) { + const score = output[(4 + c) * numPredictions + i]; + if (score < confidenceThreshold) continue; + + let x1 = (cx - w / 2) * scale; + let y1_raw = cy - h / 2; + let x2 = (cx + w / 2) * scale; + let y2_raw = cy + h / 2; + + let y1: number, y2: number; + if (flipY) { + y1 = (this.inputSize - y2_raw) * scale; + y2 = (this.inputSize - y1_raw) * scale; + } else { + y1 = y1_raw * scale; + y2 = y2_raw * scale; + } + + x1 = Math.max(0, Math.min(x1, originalWidth)); + y1 = Math.max(0, Math.min(y1, originalHeight)); + x2 = Math.max(0, Math.min(x2, originalWidth)); + y2 = Math.max(0, Math.min(y2, originalHeight)); + + const boxWidth = x2 - x1; + const boxHeight = y2 - y1; + if (boxWidth < 10 && boxHeight < 10) continue; + if (boxWidth <= 0 || boxHeight <= 0) continue; + + detections.push({ + box: [x1, y1, x2, y2], + score, + classIndex: c, + className: this.classLabels[c], + }); + } + } + + return detections; + } +} + +/** + * Create mock YOLO output tensor + */ +function createMockYOLOOutput( + predictions: Array<{ + cx: number; + cy: number; + w: number; + h: number; + classScores: number[]; + }>, + numClasses: number = 18 +): number[] { + const numPredictions = predictions.length; + const valuesPerPrediction = 4 + numClasses; + + // Initialize output array (row-major: [values, predictions]) + const output = new Array(valuesPerPrediction * numPredictions).fill(0); + + for (let i = 0; i < numPredictions; i++) { + const pred = predictions[i]; + output[0 * numPredictions + i] = pred.cx; + output[1 * numPredictions + i] = pred.cy; + output[2 * numPredictions + i] = pred.w; + output[3 * numPredictions + i] = pred.h; + + for (let c = 0; c < numClasses; c++) { + output[(4 + c) * numPredictions + i] = pred.classScores[c] || 0; + } + } + + return output; +} + +describe('YOLO Parser', () => { + let parser: YOLOParser; + + beforeEach(() => { + parser = new YOLOParser(NUDENET_LABELS, 320); + }); + + describe('Basic Parsing', () => { + it('should parse empty output', () => { + const result = parser.parse([], 0.5, 640, 480); + expect(result).toEqual([]); + }); + + it('should filter detections below confidence threshold', () => { + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores: new Array(18).fill(0.3) }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + expect(result).toHaveLength(0); + }); + + it('should keep detections above confidence threshold', () => { + const classScores = new Array(18).fill(0); + classScores[1] = 0.9; // FACE_FEMALE + + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + expect(result).toHaveLength(1); + expect(result[0].score).toBe(0.9); + expect(result[0].className).toBe('FACE_FEMALE'); + }); + + it('should select class with highest score', () => { + const classScores = new Array(18).fill(0); + classScores[1] = 0.7; // FACE_FEMALE + classScores[3] = 0.85; // FEMALE_BREAST_EXPOSED (higher) + classScores[11] = 0.6; // ARMPITS_EXPOSED + + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + expect(result).toHaveLength(1); + expect(result[0].classIndex).toBe(3); + expect(result[0].className).toBe('FEMALE_BREAST_EXPOSED'); + }); + }); + + describe('Coordinate Transformation', () => { + it('should convert center to corner format', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + // Center at (160, 160), size 100x100 + // Expected corners: [110, 110, 210, 210] + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + expect(result[0].box[0]).toBeCloseTo(110, 1); + expect(result[0].box[1]).toBeCloseTo(110, 1); + expect(result[0].box[2]).toBeCloseTo(210, 1); + expect(result[0].box[3]).toBeCloseTo(210, 1); + }); + + it('should scale coordinates for non-square images (landscape)', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + // Model coordinates for 320x320 input + // Original image is 640x480 (landscape) + // maxDim = 640, scale = 640/320 = 2 + const output = createMockYOLOOutput([ + { cx: 160, cy: 120, w: 80, h: 60, classScores }, + ]); + + const result = parser.parse(output, 0.5, 640, 480); + // (160-40)*2=240, (120-30)*2=180, (160+40)*2=400, (120+30)*2=300 + expect(result[0].box[0]).toBeCloseTo(240, 1); + expect(result[0].box[1]).toBeCloseTo(180, 1); + expect(result[0].box[2]).toBeCloseTo(400, 1); + expect(result[0].box[3]).toBeCloseTo(300, 1); + }); + + it('should scale coordinates for non-square images (portrait)', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + // Original image is 480x640 (portrait) + // maxDim = 640, scale = 640/320 = 2 + const output = createMockYOLOOutput([ + { cx: 120, cy: 160, w: 60, h: 80, classScores }, + ]); + + const result = parser.parse(output, 0.5, 480, 640); + expect(result[0].box[0]).toBeCloseTo(180, 1); // (120-30)*2 + expect(result[0].box[1]).toBeCloseTo(240, 1); // (160-40)*2 + expect(result[0].box[2]).toBeCloseTo(300, 1); // (120+30)*2 + expect(result[0].box[3]).toBeCloseTo(400, 1); // (160+40)*2 + }); + + it('should clip coordinates to image boundaries', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + // Box extends beyond image (center near corner) + const output = createMockYOLOOutput([ + { cx: 300, cy: 300, w: 100, h: 100, classScores }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + // Box should be clipped to [250, 250, 320, 320] + expect(result[0].box[0]).toBeCloseTo(250, 1); + expect(result[0].box[1]).toBeCloseTo(250, 1); + expect(result[0].box[2]).toBe(320); // Clipped + expect(result[0].box[3]).toBe(320); // Clipped + }); + + it('should filter tiny boxes (< 10px)', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + // Very small box (5x5 after scaling) + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 5, h: 5, classScores }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + expect(result).toHaveLength(0); // Filtered out + }); + }); + + describe('Y-Axis Flipping (iOS vs Android)', () => { + it('should match Android behavior without Y-flip', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + const output = createMockYOLOOutput([ + { cx: 160, cy: 80, w: 100, h: 50, classScores }, // Near top of image + ]); + + const result = parser.parse(output, 0.5, 320, 320, false); + // y1 = 80 - 25 = 55, y2 = 80 + 25 = 105 + expect(result[0].box[1]).toBeCloseTo(55, 1); // Near top + expect(result[0].box[3]).toBeCloseTo(105, 1); + }); + + it('should match iOS behavior with Y-flip', () => { + const classScores = new Array(18).fill(0); + classScores[0] = 0.9; + + const output = createMockYOLOOutput([ + { cx: 160, cy: 80, w: 100, h: 50, classScores }, // Near top in model coords + ]); + + const result = parser.parse(output, 0.5, 320, 320, true); + // After flip: y1 = 320 - 105 = 215, y2 = 320 - 55 = 265 + expect(result[0].box[1]).toBeCloseTo(215, 1); // Now near bottom + expect(result[0].box[3]).toBeCloseTo(265, 1); + }); + }); + + describe('parseAllClasses', () => { + it('should return multiple detections for same box with different classes', () => { + const classScores = new Array(18).fill(0); + classScores[1] = 0.7; // FACE_FEMALE + classScores[3] = 0.8; // FEMALE_BREAST_EXPOSED + classScores[11] = 0.6; // ARMPITS_EXPOSED + + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores }, + ]); + + const result = parser.parseAllClasses(output, 0.55, 320, 320); + // Should return detections for class 1 (0.7) and class 3 (0.8) + // Class 11 (0.6) is above 0.55, so 3 detections total + expect(result.length).toBeGreaterThanOrEqual(2); + + const classIndices = result.map(d => d.classIndex); + expect(classIndices).toContain(1); + expect(classIndices).toContain(3); + }); + + it('should catch NSFW classes even when face scores higher', () => { + const classScores = new Array(18).fill(0); + classScores[1] = 0.95; // FACE_FEMALE (highest) + classScores[3] = 0.7; // FEMALE_BREAST_EXPOSED (NSFW, lower than face) + + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores }, + ]); + + // Regular parse only returns face + const regularResult = parser.parse(output, 0.5, 320, 320); + expect(regularResult).toHaveLength(1); + expect(regularResult[0].classIndex).toBe(1); + + // parseAllClasses catches the NSFW class too + const allClassesResult = parser.parseAllClasses(output, 0.5, 320, 320); + const nsfwDetection = allClassesResult.find(d => d.classIndex === 3); + expect(nsfwDetection).toBeDefined(); + expect(nsfwDetection?.score).toBe(0.7); + }); + }); + + describe('NSFW Detection', () => { + it('should identify NSFW class indices', () => { + // Test that NSFW classes are correctly identified + expect(NSFW_CLASS_INDICES.has(2)).toBe(true); // BUTTOCKS_EXPOSED + expect(NSFW_CLASS_INDICES.has(3)).toBe(true); // FEMALE_BREAST_EXPOSED + expect(NSFW_CLASS_INDICES.has(4)).toBe(true); // FEMALE_GENITALIA_EXPOSED + expect(NSFW_CLASS_INDICES.has(6)).toBe(true); // ANUS_EXPOSED + expect(NSFW_CLASS_INDICES.has(14)).toBe(true); // MALE_GENITALIA_EXPOSED + + // Non-NSFW classes + expect(NSFW_CLASS_INDICES.has(0)).toBe(false); // FEMALE_GENITALIA_COVERED + expect(NSFW_CLASS_INDICES.has(1)).toBe(false); // FACE_FEMALE + expect(NSFW_CLASS_INDICES.has(5)).toBe(false); // MALE_BREAST_EXPOSED + }); + + it('should detect NSFW content', () => { + const classScores = new Array(18).fill(0); + classScores[3] = 0.85; // FEMALE_BREAST_EXPOSED + + const output = createMockYOLOOutput([ + { cx: 160, cy: 160, w: 100, h: 100, classScores }, + ]); + + const result = parser.parse(output, 0.5, 320, 320); + expect(result).toHaveLength(1); + expect(NSFW_CLASS_INDICES.has(result[0].classIndex)).toBe(true); + }); + }); + + describe('Real-World Scenarios', () => { + it('should handle typical NudeNet output size (6300 predictions)', () => { + // NudeNet 320n outputs 6300 predictions + const predictions = []; + for (let i = 0; i < 6300; i++) { + const classScores = new Array(18).fill(0.001); + // Add one high-confidence detection + if (i === 3150) { + classScores[1] = 0.9; + } + predictions.push({ + cx: (i % 79) * 4 + 2, + cy: Math.floor(i / 79) * 4 + 2, + w: 10, + h: 10, + classScores, + }); + } + + const output = createMockYOLOOutput(predictions); + expect(output.length).toBe(22 * 6300); // 22 values * 6300 predictions + + const result = parser.parse(output, 0.5, 320, 320); + expect(result).toHaveLength(1); + expect(result[0].score).toBe(0.9); + }); + + it('should handle multiple detections in same image', () => { + const predictions = [ + // Face in top-left + { cx: 80, cy: 80, w: 60, h: 60, classScores: [0, 0.9, ...new Array(16).fill(0)] }, + // Body in center + { cx: 160, cy: 200, w: 100, h: 150, classScores: [0, 0, 0, 0.8, ...new Array(14).fill(0)] }, + // Feet at bottom + { cx: 160, cy: 290, w: 80, h: 40, classScores: [0, 0, 0, 0, 0, 0, 0, 0.7, ...new Array(10).fill(0)] }, + ]; + + const output = createMockYOLOOutput(predictions); + const result = parser.parse(output, 0.5, 320, 320); + + expect(result).toHaveLength(3); + expect(result.map(d => d.className).sort()).toEqual([ + 'FACE_FEMALE', + 'FEMALE_BREAST_EXPOSED', + 'FEET_EXPOSED', + ].sort()); + }); + }); +}); + +describe('Platform Parity: YOLO Parser', () => { + it('should produce consistent output for same input (Android)', () => { + const parser = new YOLOParser(NUDENET_LABELS, 320); + + const classScores = new Array(18).fill(0); + classScores[3] = 0.85; + + const output = createMockYOLOOutput([ + { cx: 150, cy: 100, w: 80, h: 120, classScores }, + ]); + + // Original image: 640x480 (2x scale) + const result = parser.parse(output, 0.6, 640, 480, false); + + expect(result).toHaveLength(1); + expect(result[0].className).toBe('FEMALE_BREAST_EXPOSED'); + expect(result[0].score).toBe(0.85); + // (150-40)*2=220, (100-60)*2=80, (150+40)*2=380, (100+60)*2=320 + expect(result[0].box[0]).toBeCloseTo(220, 1); + expect(result[0].box[1]).toBeCloseTo(80, 1); + expect(result[0].box[2]).toBeCloseTo(380, 1); + expect(result[0].box[3]).toBeCloseTo(320, 1); + }); + + it('should produce consistent output for same input (iOS with Y-flip)', () => { + const parser = new YOLOParser(NUDENET_LABELS, 320); + + const classScores = new Array(18).fill(0); + classScores[3] = 0.85; + + const output = createMockYOLOOutput([ + { cx: 150, cy: 100, w: 80, h: 120, classScores }, + ]); + + // Original image: 640x480 (2x scale) + const result = parser.parse(output, 0.6, 640, 480, true); + + expect(result).toHaveLength(1); + expect(result[0].className).toBe('FEMALE_BREAST_EXPOSED'); + expect(result[0].score).toBe(0.85); + // X unchanged: 220, 380 + // Y flipped: y1_orig=40, y2_orig=160 + // y1_flipped = 320-160=160, y2_flipped = 320-40=280 + // scaled: 160*2=320, 280*2=560 (but clipped to 480) + expect(result[0].box[0]).toBeCloseTo(220, 1); + expect(result[0].box[2]).toBeCloseTo(380, 1); + expect(result[0].box[1]).toBeCloseTo(320, 1); + expect(result[0].box[3]).toBe(480); // Clipped to height + }); +}); diff --git a/android/build.gradle b/android/build.gradle index 54ae8f0..182750f 100644 --- a/android/build.gradle +++ b/android/build.gradle @@ -43,6 +43,13 @@ android { main { java.srcDirs = ['src/main/java'] } + test { + java.srcDirs = ['src/test/java'] + } + } + + testOptions { + unitTests.returnDefaultValues = true } } @@ -68,4 +75,9 @@ dependencies { implementation 'com.google.mlkit:face-detection:16.1.6' implementation 'com.google.mlkit:image-labeling:17.0.8' implementation 'com.google.mlkit:text-recognition:16.0.0' + + // Testing + testImplementation 'junit:junit:4.13.2' + testImplementation 'org.mockito:mockito-core:5.5.0' + testImplementation 'org.mockito.kotlin:mockito-kotlin:5.1.0' } diff --git a/android/src/test/java/com/visionml/NMSTest.kt b/android/src/test/java/com/visionml/NMSTest.kt new file mode 100644 index 0000000..4063b34 --- /dev/null +++ b/android/src/test/java/com/visionml/NMSTest.kt @@ -0,0 +1,175 @@ +package com.visionml + +import org.junit.Assert.* +import org.junit.Test + +/** + * Unit tests for Non-Maximum Suppression (NMS) algorithm + */ +class NMSTest { + + @Test + fun `calculateIoU returns 1 for identical boxes`() { + val box = floatArrayOf(0f, 0f, 100f, 100f) + val iou = NMS.calculateIoU(box, box) + assertEquals(1.0f, iou, 0.001f) + } + + @Test + fun `calculateIoU returns 0 for non-overlapping boxes`() { + val box1 = floatArrayOf(0f, 0f, 50f, 50f) + val box2 = floatArrayOf(100f, 100f, 150f, 150f) + val iou = NMS.calculateIoU(box1, box2) + assertEquals(0.0f, iou, 0.001f) + } + + @Test + fun `calculateIoU correct for partially overlapping boxes`() { + val box1 = floatArrayOf(0f, 0f, 100f, 100f) // Area = 10000 + val box2 = floatArrayOf(50f, 50f, 150f, 150f) // Area = 10000 + // Intersection: [50,50,100,100] = 2500 + // Union: 10000 + 10000 - 2500 = 17500 + // IoU: 2500/17500 = 0.1428 + val iou = NMS.calculateIoU(box1, box2) + assertEquals(0.1429f, iou, 0.001f) + } + + @Test + fun `calculateIoU handles touching boxes`() { + val box1 = floatArrayOf(0f, 0f, 100f, 100f) + val box2 = floatArrayOf(100f, 0f, 200f, 100f) + val iou = NMS.calculateIoU(box1, box2) + assertEquals(0.0f, iou, 0.001f) + } + + @Test + fun `calculateIoU handles nested boxes`() { + val outer = floatArrayOf(0f, 0f, 100f, 100f) // Area = 10000 + val inner = floatArrayOf(25f, 25f, 75f, 75f) // Area = 2500 + // Intersection = 2500 + // Union = 10000 + // IoU = 0.25 + val iou = NMS.calculateIoU(outer, inner) + assertEquals(0.25f, iou, 0.001f) + } + + @Test + fun `calculateIoU is symmetric`() { + val box1 = floatArrayOf(10f, 20f, 80f, 90f) + val box2 = floatArrayOf(30f, 40f, 100f, 100f) + val iou1 = NMS.calculateIoU(box1, box2) + val iou2 = NMS.calculateIoU(box2, box1) + assertEquals(iou1, iou2, 0.0001f) + } + + @Test + fun `apply returns empty for empty input`() { + val result = NMS.apply(emptyList()) + assertTrue(result.isEmpty()) + } + + @Test + fun `apply returns single detection unchanged`() { + val detection = NMS.Detection( + box = floatArrayOf(0f, 0f, 100f, 100f), + score = 0.9f, + classIndex = 0, + className = "test" + ) + val result = NMS.apply(listOf(detection)) + assertEquals(1, result.size) + assertEquals(detection, result[0]) + } + + @Test + fun `apply keeps non-overlapping detections`() { + val detections = listOf( + NMS.Detection(floatArrayOf(0f, 0f, 50f, 50f), 0.9f, 0, "test"), + NMS.Detection(floatArrayOf(100f, 100f, 150f, 150f), 0.8f, 0, "test") + ) + val result = NMS.apply(detections) + assertEquals(2, result.size) + } + + @Test + fun `apply suppresses overlapping lower-score detections`() { + val detections = listOf( + NMS.Detection(floatArrayOf(0f, 0f, 100f, 100f), 0.9f, 0, "test"), + NMS.Detection(floatArrayOf(10f, 10f, 110f, 110f), 0.7f, 0, "test"), + NMS.Detection(floatArrayOf(5f, 5f, 105f, 105f), 0.8f, 0, "test") + ) + val result = NMS.apply(detections, 0.5f) + assertEquals(1, result.size) + assertEquals(0.9f, result[0].score, 0.001f) + } + + @Test + fun `apply sorts by score before suppression`() { + val detections = listOf( + NMS.Detection(floatArrayOf(0f, 0f, 100f, 100f), 0.5f, 0, "test"), + NMS.Detection(floatArrayOf(5f, 5f, 105f, 105f), 0.9f, 0, "test"), + NMS.Detection(floatArrayOf(10f, 10f, 110f, 110f), 0.7f, 0, "test") + ) + val result = NMS.apply(detections, 0.5f) + assertEquals(1, result.size) + assertEquals(0.9f, result[0].score, 0.001f) + } + + @Test + fun `apply handles multiple non-overlapping groups`() { + val detections = listOf( + // Group 1 + NMS.Detection(floatArrayOf(0f, 0f, 50f, 50f), 0.9f, 0, "A"), + NMS.Detection(floatArrayOf(5f, 5f, 55f, 55f), 0.8f, 0, "A"), + // Group 2 + NMS.Detection(floatArrayOf(200f, 200f, 250f, 250f), 0.85f, 1, "B"), + NMS.Detection(floatArrayOf(205f, 205f, 255f, 255f), 0.7f, 1, "B") + ) + val result = NMS.apply(detections, 0.5f) + assertEquals(2, result.size) + val scores = result.map { it.score }.sortedDescending() + assertEquals(listOf(0.9f, 0.85f), scores) + } + + @Test + fun `apply uses default IoU threshold of 0_45`() { + val detections = listOf( + NMS.Detection(floatArrayOf(0f, 0f, 100f, 100f), 0.9f, 0, "test"), + NMS.Detection(floatArrayOf(35f, 35f, 135f, 135f), 0.8f, 0, "test") + ) + val result = NMS.apply(detections) // default 0.45 + assertEquals(2, result.size) // ~30% overlap, both kept + } + + @Test + fun `apply handles real-world NSFW detections`() { + val detections = listOf( + NMS.Detection( + floatArrayOf(156.2f, 89.4f, 312.8f, 445.1f), + 0.873f, 3, "FEMALE_BREAST_EXPOSED" + ), + NMS.Detection( + floatArrayOf(158.1f, 91.2f, 310.5f, 442.3f), + 0.821f, 3, "FEMALE_BREAST_EXPOSED" + ), + NMS.Detection( + floatArrayOf(155.0f, 88.0f, 315.0f, 448.0f), + 0.756f, 3, "FEMALE_BREAST_EXPOSED" + ), + NMS.Detection( + floatArrayOf(450.5f, 200.3f, 550.7f, 380.9f), + 0.945f, 11, "FACE_FEMALE" + ) + ) + val result = NMS.apply(detections, 0.45f) + assertEquals(2, result.size) + + val breastDetection = result.find { it.className == "FEMALE_BREAST_EXPOSED" } + val faceDetection = result.find { it.className == "FACE_FEMALE" } + + assertNotNull(breastDetection) + assertNotNull(faceDetection) + assertEquals(0.873f, breastDetection!!.score, 0.001f) + assertEquals(0.945f, faceDetection!!.score, 0.001f) + } +} diff --git a/android/src/test/java/com/visionml/YOLOParserTest.kt b/android/src/test/java/com/visionml/YOLOParserTest.kt new file mode 100644 index 0000000..90f21dc --- /dev/null +++ b/android/src/test/java/com/visionml/YOLOParserTest.kt @@ -0,0 +1,306 @@ +package com.visionml + +import org.junit.Assert.* +import org.junit.Before +import org.junit.Test + +/** + * Unit tests for YOLO v8 output parser + */ +class YOLOParserTest { + + companion object { + val NUDENET_LABELS = listOf( + "FEMALE_GENITALIA_COVERED", // 0 + "FACE_FEMALE", // 1 + "BUTTOCKS_EXPOSED", // 2 + "FEMALE_BREAST_EXPOSED", // 3 + "FEMALE_GENITALIA_EXPOSED", // 4 + "MALE_BREAST_EXPOSED", // 5 + "ANUS_EXPOSED", // 6 + "FEET_EXPOSED", // 7 + "BELLY_COVERED", // 8 + "FEET_COVERED", // 9 + "ARMPITS_COVERED", // 10 + "ARMPITS_EXPOSED", // 11 + "FACE_MALE", // 12 + "BELLY_EXPOSED", // 13 + "MALE_GENITALIA_EXPOSED", // 14 + "ANUS_COVERED", // 15 + "FEMALE_BREAST_COVERED", // 16 + "BUTTOCKS_COVERED" // 17 + ) + + val NSFW_CLASS_INDICES = setOf(2, 3, 4, 6, 14) + + /** + * Create mock YOLO output tensor + */ + fun createMockOutput( + predictions: List, + numClasses: Int = 18 + ): FloatArray { + val numPredictions = predictions.size + val valuesPerPrediction = 4 + numClasses + val output = FloatArray(valuesPerPrediction * numPredictions) + + predictions.forEachIndexed { i, pred -> + output[0 * numPredictions + i] = pred.cx + output[1 * numPredictions + i] = pred.cy + output[2 * numPredictions + i] = pred.w + output[3 * numPredictions + i] = pred.h + pred.classScores.forEachIndexed { c, score -> + if (c < numClasses) { + output[(4 + c) * numPredictions + i] = score + } + } + } + return output + } + } + + data class MockPrediction( + val cx: Float, + val cy: Float, + val w: Float, + val h: Float, + val classScores: List + ) + + private lateinit var parser: YOLOParser + + @Before + fun setup() { + parser = YOLOParser(NUDENET_LABELS, 320) + } + + @Test + fun `parse returns empty for empty output`() { + val result = parser.parse(FloatArray(0), 0.5f, 640, 480) + assertTrue(result.isEmpty()) + } + + @Test + fun `parse filters below confidence threshold`() { + val scores = MutableList(18) { 0.3f } + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + val result = parser.parse(output, 0.5f, 320, 320) + assertTrue(result.isEmpty()) + } + + @Test + fun `parse keeps above confidence threshold`() { + val scores = MutableList(18) { 0f } + scores[1] = 0.9f // FACE_FEMALE + + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + val result = parser.parse(output, 0.5f, 320, 320) + assertEquals(1, result.size) + assertEquals(0.9f, result[0].score, 0.001f) + assertEquals("FACE_FEMALE", result[0].className) + } + + @Test + fun `parse selects highest scoring class`() { + val scores = MutableList(18) { 0f } + scores[1] = 0.7f // FACE_FEMALE + scores[3] = 0.85f // FEMALE_BREAST_EXPOSED (higher) + scores[11] = 0.6f // ARMPITS_EXPOSED + + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + val result = parser.parse(output, 0.5f, 320, 320) + assertEquals(1, result.size) + assertEquals(3, result[0].classIndex) + assertEquals("FEMALE_BREAST_EXPOSED", result[0].className) + } + + @Test + fun `parse converts center to corner format`() { + val scores = MutableList(18) { 0f } + scores[0] = 0.9f + + // Center at (160, 160), size 100x100 + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + val result = parser.parse(output, 0.5f, 320, 320) + // Expected: [110, 110, 210, 210] + assertEquals(110f, result[0].box[0], 1f) + assertEquals(110f, result[0].box[1], 1f) + assertEquals(210f, result[0].box[2], 1f) + assertEquals(210f, result[0].box[3], 1f) + } + + @Test + fun `parse scales for landscape image`() { + val scores = MutableList(18) { 0f } + scores[0] = 0.9f + + // Model coords, original 640x480 (scale = 2) + val output = createMockOutput(listOf( + MockPrediction(160f, 120f, 80f, 60f, scores) + )) + + val result = parser.parse(output, 0.5f, 640, 480) + // (160-40)*2=240, (120-30)*2=180, (160+40)*2=400, (120+30)*2=300 + assertEquals(240f, result[0].box[0], 1f) + assertEquals(180f, result[0].box[1], 1f) + assertEquals(400f, result[0].box[2], 1f) + assertEquals(300f, result[0].box[3], 1f) + } + + @Test + fun `parse scales for portrait image`() { + val scores = MutableList(18) { 0f } + scores[0] = 0.9f + + // Original 480x640 (portrait, scale = 2) + val output = createMockOutput(listOf( + MockPrediction(120f, 160f, 60f, 80f, scores) + )) + + val result = parser.parse(output, 0.5f, 480, 640) + assertEquals(180f, result[0].box[0], 1f) // (120-30)*2 + assertEquals(240f, result[0].box[1], 1f) // (160-40)*2 + assertEquals(300f, result[0].box[2], 1f) // (120+30)*2 + assertEquals(400f, result[0].box[3], 1f) // (160+40)*2 + } + + @Test + fun `parse clips to image boundaries`() { + val scores = MutableList(18) { 0f } + scores[0] = 0.9f + + // Box near corner, extends beyond image + val output = createMockOutput(listOf( + MockPrediction(300f, 300f, 100f, 100f, scores) + )) + + val result = parser.parse(output, 0.5f, 320, 320) + assertEquals(250f, result[0].box[0], 1f) + assertEquals(250f, result[0].box[1], 1f) + assertEquals(320f, result[0].box[2], 0.001f) // Clipped + assertEquals(320f, result[0].box[3], 0.001f) // Clipped + } + + @Test + fun `parse filters tiny boxes`() { + val scores = MutableList(18) { 0f } + scores[0] = 0.9f + + // Very small 5x5 box + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 5f, 5f, scores) + )) + + val result = parser.parse(output, 0.5f, 320, 320) + assertTrue(result.isEmpty()) + } + + @Test + fun `parseAllClasses returns multiple detections per box`() { + val scores = MutableList(18) { 0f } + scores[1] = 0.7f // FACE_FEMALE + scores[3] = 0.8f // FEMALE_BREAST_EXPOSED + scores[11] = 0.6f // ARMPITS_EXPOSED + + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + val result = parser.parseAllClasses(output, 0.55f, 320, 320) + assertTrue(result.size >= 2) + + val classIndices = result.map { it.classIndex } + assertTrue(classIndices.contains(1)) + assertTrue(classIndices.contains(3)) + } + + @Test + fun `parseAllClasses catches NSFW when face scores higher`() { + val scores = MutableList(18) { 0f } + scores[1] = 0.95f // FACE_FEMALE (highest) + scores[3] = 0.7f // FEMALE_BREAST_EXPOSED (NSFW) + + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + // Regular parse only returns face + val regularResult = parser.parse(output, 0.5f, 320, 320) + assertEquals(1, regularResult.size) + assertEquals(1, regularResult[0].classIndex) + + // parseAllClasses catches NSFW too + val allResult = parser.parseAllClasses(output, 0.5f, 320, 320) + val nsfwDetection = allResult.find { it.classIndex == 3 } + assertNotNull(nsfwDetection) + assertEquals(0.7f, nsfwDetection!!.score, 0.001f) + } + + @Test + fun `NSFW class indices are correct`() { + assertTrue(NSFW_CLASS_INDICES.contains(2)) // BUTTOCKS_EXPOSED + assertTrue(NSFW_CLASS_INDICES.contains(3)) // FEMALE_BREAST_EXPOSED + assertTrue(NSFW_CLASS_INDICES.contains(4)) // FEMALE_GENITALIA_EXPOSED + assertTrue(NSFW_CLASS_INDICES.contains(6)) // ANUS_EXPOSED + assertTrue(NSFW_CLASS_INDICES.contains(14)) // MALE_GENITALIA_EXPOSED + + // Non-NSFW + assertFalse(NSFW_CLASS_INDICES.contains(0)) // FEMALE_GENITALIA_COVERED + assertFalse(NSFW_CLASS_INDICES.contains(1)) // FACE_FEMALE + assertFalse(NSFW_CLASS_INDICES.contains(5)) // MALE_BREAST_EXPOSED + } + + @Test + fun `parse handles multiple detections`() { + val predictions = listOf( + // Face top-left + MockPrediction(80f, 80f, 60f, 60f, + listOf(0f, 0.9f) + List(16) { 0f }), + // Body center + MockPrediction(160f, 200f, 100f, 150f, + listOf(0f, 0f, 0f, 0.8f) + List(14) { 0f }), + // Feet bottom + MockPrediction(160f, 290f, 80f, 40f, + listOf(0f, 0f, 0f, 0f, 0f, 0f, 0f, 0.7f) + List(10) { 0f }) + ) + + val output = createMockOutput(predictions) + val result = parser.parse(output, 0.5f, 320, 320) + + assertEquals(3, result.size) + val classNames = result.map { it.className }.sorted() + assertEquals( + listOf("FACE_FEMALE", "FEMALE_BREAST_EXPOSED", "FEET_EXPOSED").sorted(), + classNames + ) + } + + @Test + fun `parse stores debug info`() { + val scores = MutableList(18) { 0f } + scores[3] = 0.85f // FEMALE_BREAST_EXPOSED + + val output = createMockOutput(listOf( + MockPrediction(160f, 160f, 100f, 100f, scores) + )) + + parser.parse(output, 0.5f, 640, 480) + + val debugInfo = parser.lastDebugInfo + assertTrue(debugInfo.isNotEmpty()) + assertTrue(debugInfo.containsKey("numPredictions")) + assertTrue(debugInfo.containsKey("nativeScaleFactor")) + } +} diff --git a/ios/VisionMLTests/NMSTests.swift b/ios/VisionMLTests/NMSTests.swift new file mode 100644 index 0000000..d0c68d8 --- /dev/null +++ b/ios/VisionMLTests/NMSTests.swift @@ -0,0 +1,171 @@ +import XCTest +@testable import VisionML + +/** + * Unit tests for Non-Maximum Suppression (NMS) algorithm + */ +class NMSTests: XCTestCase { + + // MARK: - IoU Calculation Tests + + func testCalculateIoU_identicalBoxes_returns1() { + let box: [Float] = [0, 0, 100, 100] + let iou = NMS.calculateIoU(box1: box, box2: box) + XCTAssertEqual(iou, 1.0, accuracy: 0.001) + } + + func testCalculateIoU_nonOverlapping_returns0() { + let box1: [Float] = [0, 0, 50, 50] + let box2: [Float] = [100, 100, 150, 150] + let iou = NMS.calculateIoU(box1: box1, box2: box2) + XCTAssertEqual(iou, 0.0, accuracy: 0.001) + } + + func testCalculateIoU_partialOverlap_correctValue() { + let box1: [Float] = [0, 0, 100, 100] // Area = 10000 + let box2: [Float] = [50, 50, 150, 150] // Area = 10000 + // Intersection: [50,50,100,100] = 2500 + // Union: 10000 + 10000 - 2500 = 17500 + // IoU: 2500/17500 = 0.1428 + let iou = NMS.calculateIoU(box1: box1, box2: box2) + XCTAssertEqual(iou, 0.1429, accuracy: 0.001) + } + + func testCalculateIoU_touchingBoxes_returns0() { + let box1: [Float] = [0, 0, 100, 100] + let box2: [Float] = [100, 0, 200, 100] + let iou = NMS.calculateIoU(box1: box1, box2: box2) + XCTAssertEqual(iou, 0.0, accuracy: 0.001) + } + + func testCalculateIoU_nestedBoxes_correctValue() { + let outer: [Float] = [0, 0, 100, 100] // Area = 10000 + let inner: [Float] = [25, 25, 75, 75] // Area = 2500 + // IoU = 2500/10000 = 0.25 + let iou = NMS.calculateIoU(box1: outer, box2: inner) + XCTAssertEqual(iou, 0.25, accuracy: 0.001) + } + + func testCalculateIoU_isSymmetric() { + let box1: [Float] = [10, 20, 80, 90] + let box2: [Float] = [30, 40, 100, 100] + let iou1 = NMS.calculateIoU(box1: box1, box2: box2) + let iou2 = NMS.calculateIoU(box1: box2, box2: box1) + XCTAssertEqual(iou1, iou2, accuracy: 0.0001) + } + + // MARK: - NMS Apply Tests + + func testApply_emptyInput_returnsEmpty() { + let result = NMS.apply(detections: []) + XCTAssertTrue(result.isEmpty) + } + + func testApply_singleDetection_returnsUnchanged() { + let detection = NMS.Detection( + box: [0, 0, 100, 100], + score: 0.9, + classIndex: 0, + className: "test" + ) + let result = NMS.apply(detections: [detection]) + XCTAssertEqual(result.count, 1) + XCTAssertEqual(result[0].score, detection.score) + } + + func testApply_nonOverlapping_keepsAll() { + let detections = [ + NMS.Detection(box: [0, 0, 50, 50], score: 0.9, classIndex: 0, className: "test"), + NMS.Detection(box: [100, 100, 150, 150], score: 0.8, classIndex: 0, className: "test") + ] + let result = NMS.apply(detections: detections) + XCTAssertEqual(result.count, 2) + } + + func testApply_overlapping_suppressesLowerScore() { + let detections = [ + NMS.Detection(box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: "test"), + NMS.Detection(box: [10, 10, 110, 110], score: 0.7, classIndex: 0, className: "test"), + NMS.Detection(box: [5, 5, 105, 105], score: 0.8, classIndex: 0, className: "test") + ] + let result = NMS.apply(detections: detections, iouThreshold: 0.5) + XCTAssertEqual(result.count, 1) + XCTAssertEqual(result[0].score, 0.9, accuracy: 0.001) + } + + func testApply_sortsByScoreBeforeSuppression() { + let detections = [ + NMS.Detection(box: [0, 0, 100, 100], score: 0.5, classIndex: 0, className: "test"), + NMS.Detection(box: [5, 5, 105, 105], score: 0.9, classIndex: 0, className: "test"), + NMS.Detection(box: [10, 10, 110, 110], score: 0.7, classIndex: 0, className: "test") + ] + let result = NMS.apply(detections: detections, iouThreshold: 0.5) + XCTAssertEqual(result.count, 1) + XCTAssertEqual(result[0].score, 0.9, accuracy: 0.001) + } + + func testApply_multipleGroups_keepsOneFromEach() { + let detections = [ + // Group 1 + NMS.Detection(box: [0, 0, 50, 50], score: 0.9, classIndex: 0, className: "A"), + NMS.Detection(box: [5, 5, 55, 55], score: 0.8, classIndex: 0, className: "A"), + // Group 2 + NMS.Detection(box: [200, 200, 250, 250], score: 0.85, classIndex: 1, className: "B"), + NMS.Detection(box: [205, 205, 255, 255], score: 0.7, classIndex: 1, className: "B") + ] + let result = NMS.apply(detections: detections, iouThreshold: 0.5) + XCTAssertEqual(result.count, 2) + + let scores = result.map { $0.score }.sorted(by: >) + XCTAssertEqual(scores, [0.9, 0.85]) + } + + func testApply_usesDefaultThreshold() { + let detections = [ + NMS.Detection(box: [0, 0, 100, 100], score: 0.9, classIndex: 0, className: "test"), + NMS.Detection(box: [35, 35, 135, 135], score: 0.8, classIndex: 0, className: "test") + ] + let result = NMS.apply(detections: detections) // default 0.45 + XCTAssertEqual(result.count, 2) // ~30% overlap, both kept + } + + func testApply_realWorldNSFWDetections() { + let detections = [ + NMS.Detection( + box: [156.2, 89.4, 312.8, 445.1], + score: 0.873, + classIndex: 3, + className: "FEMALE_BREAST_EXPOSED" + ), + NMS.Detection( + box: [158.1, 91.2, 310.5, 442.3], + score: 0.821, + classIndex: 3, + className: "FEMALE_BREAST_EXPOSED" + ), + NMS.Detection( + box: [155.0, 88.0, 315.0, 448.0], + score: 0.756, + classIndex: 3, + className: "FEMALE_BREAST_EXPOSED" + ), + NMS.Detection( + box: [450.5, 200.3, 550.7, 380.9], + score: 0.945, + classIndex: 11, + className: "FACE_FEMALE" + ) + ] + + let result = NMS.apply(detections: detections, iouThreshold: 0.45) + XCTAssertEqual(result.count, 2) + + let breastDetection = result.first { $0.className == "FEMALE_BREAST_EXPOSED" } + let faceDetection = result.first { $0.className == "FACE_FEMALE" } + + XCTAssertNotNil(breastDetection) + XCTAssertNotNil(faceDetection) + XCTAssertEqual(breastDetection!.score, 0.873, accuracy: 0.001) + XCTAssertEqual(faceDetection!.score, 0.945, accuracy: 0.001) + } +} diff --git a/ios/VisionMLTests/YOLOParserTests.swift b/ios/VisionMLTests/YOLOParserTests.swift new file mode 100644 index 0000000..5c29d20 --- /dev/null +++ b/ios/VisionMLTests/YOLOParserTests.swift @@ -0,0 +1,290 @@ +import XCTest +@testable import VisionML + +/** + * Unit tests for YOLO v8 output parser + */ +class YOLOParserTests: XCTestCase { + + // MARK: - Test Data + + static let nudeNetLabels = [ + "FEMALE_GENITALIA_COVERED", // 0 + "FACE_FEMALE", // 1 + "BUTTOCKS_EXPOSED", // 2 + "FEMALE_BREAST_EXPOSED", // 3 + "FEMALE_GENITALIA_EXPOSED", // 4 + "MALE_BREAST_EXPOSED", // 5 + "ANUS_EXPOSED", // 6 + "FEET_EXPOSED", // 7 + "BELLY_COVERED", // 8 + "FEET_COVERED", // 9 + "ARMPITS_COVERED", // 10 + "ARMPITS_EXPOSED", // 11 + "FACE_MALE", // 12 + "BELLY_EXPOSED", // 13 + "MALE_GENITALIA_EXPOSED", // 14 + "ANUS_COVERED", // 15 + "FEMALE_BREAST_COVERED", // 16 + "BUTTOCKS_COVERED" // 17 + ] + + static let nsfwClassIndices: Set = [2, 3, 4, 6, 14] + + var parser: YOLOParser! + + override func setUp() { + super.setUp() + parser = YOLOParser(classLabels: YOLOParserTests.nudeNetLabels, inputSize: 320) + } + + // MARK: - Helper Functions + + struct MockPrediction { + let cx: Float + let cy: Float + let w: Float + let h: Float + let classScores: [Float] + } + + func createMockOutput(predictions: [MockPrediction], numClasses: Int = 18) -> [Float] { + let numPredictions = predictions.count + let valuesPerPrediction = 4 + numClasses + var output = [Float](repeating: 0, count: valuesPerPrediction * numPredictions) + + for (i, pred) in predictions.enumerated() { + output[0 * numPredictions + i] = pred.cx + output[1 * numPredictions + i] = pred.cy + output[2 * numPredictions + i] = pred.w + output[3 * numPredictions + i] = pred.h + + for (c, score) in pred.classScores.enumerated() where c < numClasses { + output[(4 + c) * numPredictions + i] = score + } + } + + return output + } + + // MARK: - Basic Parsing Tests + + func testParse_emptyOutput_returnsEmpty() { + let result = parser.parse(output: [], confidenceThreshold: 0.5, originalWidth: 640, originalHeight: 480) + XCTAssertTrue(result.isEmpty) + } + + func testParse_belowThreshold_filtered() { + var scores = [Float](repeating: 0.3, count: 18) + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + XCTAssertTrue(result.isEmpty) + } + + func testParse_aboveThreshold_kept() { + var scores = [Float](repeating: 0, count: 18) + scores[1] = 0.9 // FACE_FEMALE + + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + XCTAssertEqual(result.count, 1) + XCTAssertEqual(result[0].score, 0.9, accuracy: 0.001) + XCTAssertEqual(result[0].className, "FACE_FEMALE") + } + + func testParse_selectsHighestScoringClass() { + var scores = [Float](repeating: 0, count: 18) + scores[1] = 0.7 // FACE_FEMALE + scores[3] = 0.85 // FEMALE_BREAST_EXPOSED (higher) + scores[11] = 0.6 // ARMPITS_EXPOSED + + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + XCTAssertEqual(result.count, 1) + XCTAssertEqual(result[0].classIndex, 3) + XCTAssertEqual(result[0].className, "FEMALE_BREAST_EXPOSED") + } + + // MARK: - Coordinate Transformation Tests + + func testParse_convertsCenterToCorner() { + var scores = [Float](repeating: 0, count: 18) + scores[0] = 0.9 + + // Center at (160, 160), size 100x100 + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + + // iOS uses Y-flip, so Y coordinates will be different + // x1 = 160 - 50 = 110 + // x2 = 160 + 50 = 210 + XCTAssertEqual(result[0].box[0], 110, accuracy: 1) + XCTAssertEqual(result[0].box[2], 210, accuracy: 1) + } + + func testParse_scalesForLandscapeImage() { + var scores = [Float](repeating: 0, count: 18) + scores[0] = 0.9 + + // Model coords, original 640x480 (scale = 2) + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 120, w: 80, h: 60, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 640, originalHeight: 480) + + // x: (160-40)*2=240, (160+40)*2=400 + XCTAssertEqual(result[0].box[0], 240, accuracy: 1) + XCTAssertEqual(result[0].box[2], 400, accuracy: 1) + } + + func testParse_clipsToImageBoundaries() { + var scores = [Float](repeating: 0, count: 18) + scores[0] = 0.9 + + // Box near corner, extends beyond + let output = createMockOutput(predictions: [ + MockPrediction(cx: 300, cy: 300, w: 100, h: 100, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + + // Should be clipped to image boundaries + XCTAssertEqual(result[0].box[2], 320, accuracy: 0.001) // Clipped to width + } + + func testParse_filtersTinyBoxes() { + var scores = [Float](repeating: 0, count: 18) + scores[0] = 0.9 + + // Very small 5x5 box + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 5, h: 5, classScores: scores) + ]) + + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + XCTAssertTrue(result.isEmpty) + } + + // MARK: - parseAllClasses Tests + + func testParseAllClasses_returnsMultipleDetectionsPerBox() { + var scores = [Float](repeating: 0, count: 18) + scores[1] = 0.7 // FACE_FEMALE + scores[3] = 0.8 // FEMALE_BREAST_EXPOSED + scores[11] = 0.6 // ARMPITS_EXPOSED + + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + let result = parser.parseAllClasses( + output: output, + confidenceThreshold: 0.55, + originalWidth: 320, + originalHeight: 320 + ) + + XCTAssertGreaterThanOrEqual(result.count, 2) + + let classIndices = result.map { $0.classIndex } + XCTAssertTrue(classIndices.contains(1)) + XCTAssertTrue(classIndices.contains(3)) + } + + func testParseAllClasses_catchesNSFWWhenFaceScoresHigher() { + var scores = [Float](repeating: 0, count: 18) + scores[1] = 0.95 // FACE_FEMALE (highest) + scores[3] = 0.7 // FEMALE_BREAST_EXPOSED (NSFW) + + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + // Regular parse only returns face + let regularResult = parser.parse( + output: output, + confidenceThreshold: 0.5, + originalWidth: 320, + originalHeight: 320 + ) + XCTAssertEqual(regularResult.count, 1) + XCTAssertEqual(regularResult[0].classIndex, 1) + + // parseAllClasses catches NSFW too + let allResult = parser.parseAllClasses( + output: output, + confidenceThreshold: 0.5, + originalWidth: 320, + originalHeight: 320 + ) + let nsfwDetection = allResult.first { $0.classIndex == 3 } + XCTAssertNotNil(nsfwDetection) + XCTAssertEqual(nsfwDetection!.score, 0.7, accuracy: 0.001) + } + + // MARK: - NSFW Classification Tests + + func testNSFWClassIndices_areCorrect() { + XCTAssertTrue(YOLOParserTests.nsfwClassIndices.contains(2)) // BUTTOCKS_EXPOSED + XCTAssertTrue(YOLOParserTests.nsfwClassIndices.contains(3)) // FEMALE_BREAST_EXPOSED + XCTAssertTrue(YOLOParserTests.nsfwClassIndices.contains(4)) // FEMALE_GENITALIA_EXPOSED + XCTAssertTrue(YOLOParserTests.nsfwClassIndices.contains(6)) // ANUS_EXPOSED + XCTAssertTrue(YOLOParserTests.nsfwClassIndices.contains(14)) // MALE_GENITALIA_EXPOSED + + // Non-NSFW + XCTAssertFalse(YOLOParserTests.nsfwClassIndices.contains(0)) // FEMALE_GENITALIA_COVERED + XCTAssertFalse(YOLOParserTests.nsfwClassIndices.contains(1)) // FACE_FEMALE + XCTAssertFalse(YOLOParserTests.nsfwClassIndices.contains(5)) // MALE_BREAST_EXPOSED + } + + // MARK: - Multiple Detections Test + + func testParse_handlesMultipleDetections() { + let predictions = [ + // Face top-left + MockPrediction(cx: 80, cy: 80, w: 60, h: 60, classScores: [0, 0.9] + [Float](repeating: 0, count: 16)), + // Body center + MockPrediction(cx: 160, cy: 200, w: 100, h: 150, classScores: [0, 0, 0, 0.8] + [Float](repeating: 0, count: 14)), + // Feet bottom + MockPrediction(cx: 160, cy: 290, w: 80, h: 40, classScores: [0, 0, 0, 0, 0, 0, 0, 0.7] + [Float](repeating: 0, count: 10)) + ] + + let output = createMockOutput(predictions: predictions) + let result = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 320, originalHeight: 320) + + XCTAssertEqual(result.count, 3) + let classNames = result.map { $0.className }.sorted() + XCTAssertEqual(classNames, ["FACE_FEMALE", "FEMALE_BREAST_EXPOSED", "FEET_EXPOSED"].sorted()) + } + + // MARK: - Debug Info Test + + func testParse_storesDebugInfo() { + var scores = [Float](repeating: 0, count: 18) + scores[3] = 0.85 // FEMALE_BREAST_EXPOSED + + let output = createMockOutput(predictions: [ + MockPrediction(cx: 160, cy: 160, w: 100, h: 100, classScores: scores) + ]) + + _ = parser.parse(output: output, confidenceThreshold: 0.5, originalWidth: 640, originalHeight: 480) + + let debugInfo = parser.lastDebugInfo + XCTAssertFalse(debugInfo.isEmpty) + XCTAssertNotNil(debugInfo["numPredictions"]) + XCTAssertNotNil(debugInfo["nativeScaleFactor"]) + } +} diff --git a/jest.config.js b/jest.config.js new file mode 100644 index 0000000..c72004c --- /dev/null +++ b/jest.config.js @@ -0,0 +1,25 @@ +module.exports = { + testEnvironment: 'node', + transform: { + '^.+\\.(ts|tsx)$': ['ts-jest', { + tsconfig: { + module: 'commonjs', + esModuleInterop: true, + allowSyntheticDefaultImports: true, + }, + }], + }, + testMatch: [ + '**/__tests__/**/*.test.ts', + '**/__tests__/**/*.test.tsx', + ], + moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'], + collectCoverageFrom: [ + 'src/**/*.{ts,tsx}', + '!src/**/*.d.ts', + ], + setupFilesAfterEnv: ['/__tests__/setup.ts'], + moduleNameMapper: { + '^react-native$': '/__tests__/__mocks__/react-native.ts', + }, +}; diff --git a/package-lock.json b/package-lock.json index 4e7323a..793cea0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,122 +1,3716 @@ { "name": "react-native-vision-ml", - "version": "1.0.0", + "version": "1.0.5", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "react-native-vision-ml", + "version": "1.0.5", + "license": "MIT", + "devDependencies": { + "@types/jest": "^29.5.0", + "@types/react": "^18.0.0", + "@types/react-native": "^0.72.0", + "jest": "^29.7.0", + "ts-jest": "^29.1.0", + "typescript": "^5.0.0" + }, + "peerDependencies": { + "react": "*", + "react-native": "*" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.28.6.tgz", + "integrity": "sha512-JYgintcMjRiCvS8mMECzaEn+m3PfoQiyqukOMCCVQtoJGYJw8j/8LBJEiqkHLkfwCcs74E3pbAUFNg7d9VNJ+Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-validator-identifier": "^7.28.5", + "js-tokens": "^4.0.0", + "picocolors": "^1.1.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.28.6.tgz", + "integrity": "sha512-2lfu57JtzctfIrcGMz992hyLlByuzgIk58+hhGCxjKZ3rWI82NnVLjXcaTqkI2NvlcvOskZaiZ5kjUALo3Lpxg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.28.6.tgz", + "integrity": "sha512-H3mcG6ZDLTlYfaSNi0iOKkigqMFvkTKlGUYlD8GW7nNOYRrevuA46iTypPyv+06V3fEmvvazfntkBU34L0azAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/generator": "^7.28.6", + "@babel/helper-compilation-targets": "^7.28.6", + "@babel/helper-module-transforms": "^7.28.6", + "@babel/helpers": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/template": "^7.28.6", + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6", + "@jridgewell/remapping": "^2.3.5", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.28.6.tgz", + "integrity": "sha512-lOoVRwADj8hjf7al89tvQ2a1lf53Z+7tiXMgpZJL3maQPDxh0DgLMN62B2MKUOFcoodBHLMbDM6WAbKgNy5Suw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6", + "@jridgewell/gen-mapping": "^0.3.12", + "@jridgewell/trace-mapping": "^0.3.28", + "jsesc": "^3.0.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz", + "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/compat-data": "^7.28.6", + "@babel/helper-validator-option": "^7.27.1", + "browserslist": "^4.24.0", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-globals": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz", + "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz", + "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/traverse": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz", + "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.28.6", + "@babel/helper-validator-identifier": "^7.28.5", + "@babel/traverse": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.28.6.tgz", + "integrity": "sha512-S9gzZ/bz83GRysI7gAD4wPT/AI3uCnY+9xn+Mx/KPs2JwHJIz1W8PZkg2cqyt3RNOBM8ejcXhV6y8Og7ly/Dug==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz", + "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.28.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz", + "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.27.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz", + "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.28.6.tgz", + "integrity": "sha512-xOBvwq86HHdB7WUDTfKfT/Vuxh7gElQ+Sfti2Cy6yIWNW05P8iUslOVcZ4/sKbE+/jQaukQAdz/gf3724kYdqw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.28.6.tgz", + "integrity": "sha512-TeR9zWR18BvbfPmGbLampPMW+uW1NZnJlRuuHso8i87QZNq2JRF9i6RgxRqtEq+wQGsS19NNTWr2duhnE49mfQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.6" + }, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-static-block": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-static-block/-/plugin-syntax-class-static-block-7.14.5.tgz", + "integrity": "sha512-b+YyPmr6ldyNnM6sqYeMWE+bgJcJpO6yS4QD7ymxgH34GBPNDM/THBh8iunyvKIZztiwLH4CJZ0RxTk9emgpjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-attributes": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.28.6.tgz", + "integrity": "sha512-jiLC0ma9XkQT3TKJ9uYvlakm66Pamywo+qwL+oL8HJOvc6TWdZXVfhqJr8CCzbSGUAbDOzlGHJC1U+vRfLQDvw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.28.6.tgz", + "integrity": "sha512-wgEmr06G6sIpqr8YDwA2dSRTE3bJ+V0IfpzfSY3Lfgd7YWOaAdlykvJi13ZKBt8cZHfgH1IXN+CL656W3uUa4w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-private-property-in-object": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz", + "integrity": "sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.28.6.tgz", + "integrity": "sha512-+nDNmQye7nlnuuHDboPbGm00Vqg3oO8niRRL27/4LYHUsHYh0zJ1xWOz0uRwNFmM1Avzk8wZbc6rdiYhomzv/A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-plugin-utils": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz", + "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/parser": "^7.28.6", + "@babel/types": "^7.28.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.28.6.tgz", + "integrity": "sha512-fgWX62k02qtjqdSNTAGxmKYY/7FSL9WAS1o2Hu5+I5m9T0yxZzr4cnrfXQ/MX0rIifthCSs6FKTlzYbJcPtMNg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.28.6", + "@babel/generator": "^7.28.6", + "@babel/helper-globals": "^7.28.0", + "@babel/parser": "^7.28.6", + "@babel/template": "^7.28.6", + "@babel/types": "^7.28.6", + "debug": "^4.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.28.6", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.28.6.tgz", + "integrity": "sha512-0ZrskXVEHSWIqZM/sQZ4EV3jZJXRkio/WCxaqKZP1g//CEWEPSfeZFcms4XeKBCHU0ZKnIkdJeU/kF+eRp5lBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-string-parser": "^7.27.1", + "@babel/helper-validator-identifier": "^7.28.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", + "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-29.7.0.tgz", + "integrity": "sha512-5Ni4CU7XHQi32IJ398EEP4RrB8eV09sXP2ROqD4bksHrnTree52PsxvX8tpL8LvTZ3pFzXyPbNQReSN41CAhOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/core": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-29.7.0.tgz", + "integrity": "sha512-n7aeXWKMnGtDA48y8TLWJPJmLmmZ642Ceo78cYWEpiD7FzDgmNDV/GCVRorPABdXLJZ/9wzzgZAlHjXjxDHGsg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/reporters": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-changed-files": "^29.7.0", + "jest-config": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-resolve-dependencies": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "jest-watcher": "^29.7.0", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/environment": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-29.7.0.tgz", + "integrity": "sha512-aQIfHDq33ExsN4jP1NWGXhxgQ/wixs60gDiKO+XVMd8Mn0NWPWgc34ZQDTb2jKaUWQ7MuwoitXAsN2XVXNMpAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-8uMeAMycttpva3P1lBHB8VciS9V0XAr3GymPpipdyQXbBcuhkLQOSe8E/p92RyAdToS6ZD1tFkX+CkhoECE0dQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "^29.7.0", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.7.0.tgz", + "integrity": "sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/fake-timers": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-29.7.0.tgz", + "integrity": "sha512-q4DH1Ha4TTFPdxLsqDXK1d3+ioSL7yL5oCMJZgDYm6i+6CygW5E5xVr/D1HdsGxjt1ZWSfUAs9OxSB/BNelWrQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@sinonjs/fake-timers": "^10.0.2", + "@types/node": "*", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/globals": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-29.7.0.tgz", + "integrity": "sha512-mpiz3dutLbkW2MNFubUGUEVLkTGiqW6yLVTA+JbP6fI6J5iL9Y0Nlg8k95pcF8ctKwCS7WVxteBs29hhfAotzQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/types": "^29.6.3", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/reporters": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-29.7.0.tgz", + "integrity": "sha512-DApq0KJbJOEzAFYjHADNNxAE3KbhxQB1y5Kplb5Waqw6zVbuWatSnMjE5gs8FUgEPmNsnZA3NCWl9NG0ia04Pg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "@types/node": "*", + "chalk": "^4.0.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^6.0.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^4.0.0", + "istanbul-reports": "^3.1.3", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "slash": "^3.0.0", + "string-length": "^4.0.1", + "strip-ansi": "^6.0.0", + "v8-to-istanbul": "^9.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/schemas": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", + "integrity": "sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@sinclair/typebox": "^0.27.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/source-map": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-29.6.3.tgz", + "integrity": "sha512-MHjT95QuipcPrpLM+8JMSzFx6eHp5Bm+4XeFDJlwsvVBjmKNiIAvasGK2fxz2WbGRlnvqehFbh07MMa7n3YJnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.18", + "callsites": "^3.0.0", + "graceful-fs": "^4.2.9" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-result": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-29.7.0.tgz", + "integrity": "sha512-Fdx+tv6x1zlkJPcWXmMDAG2HBnaR9XPSd5aDWQVsfrZmLVT3lU1cwyxLgRmXR9yrq4NBoEm9BMsfgFzTQAbJYA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-sequencer": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-29.7.0.tgz", + "integrity": "sha512-GQwJ5WZVrKnOJuiYiAF52UNUJXgTZx1NHjFSEB0qEMmSZKAkdMoIzw/Cj6x6NF4AvV23AUqDpFzQkN/eYCYTxw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/transform": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-29.7.0.tgz", + "integrity": "sha512-ok/BTPFzFKVMwO5eOHRrvnBVHdRy9IrsrW1GpMaQ9MCnilNLXQKmAX8s1YXDFaai9xJpac2ySzV0YeRRECr2Vw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "babel-plugin-istanbul": "^6.1.1", + "chalk": "^4.0.0", + "convert-source-map": "^2.0.0", + "fast-json-stable-stringify": "^2.1.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "micromatch": "^4.0.4", + "pirates": "^4.0.4", + "slash": "^3.0.0", + "write-file-atomic": "^4.0.2" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/types": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", + "integrity": "sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^17.0.8", + "chalk": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.13", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz", + "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/sourcemap-codec": "^1.5.0", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/remapping": { + "version": "2.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz", + "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.5", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz", + "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.31", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz", + "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@react-native/virtualized-lists": { + "version": "0.72.8", + "resolved": "https://registry.npmjs.org/@react-native/virtualized-lists/-/virtualized-lists-0.72.8.tgz", + "integrity": "sha512-J3Q4Bkuo99k7mu+jPS9gSUSgq+lLRSI/+ahXNwV92XgJ/8UgOTxu2LPwhJnBk/sQKxq7E8WkZBnBiozukQMqrw==", + "dev": true, + "license": "MIT", + "dependencies": { + "invariant": "^2.2.4", + "nullthrows": "^1.1.1" + }, + "peerDependencies": { + "react-native": "*" + } + }, + "node_modules/@sinclair/typebox": { + "version": "0.27.8", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", + "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@sinonjs/commons": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-3.0.1.tgz", + "integrity": "sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-10.3.0.tgz", + "integrity": "sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@sinonjs/commons": "^3.0.0" + } + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.27.0", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.27.0.tgz", + "integrity": "sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.28.0", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.28.0.tgz", + "integrity": "sha512-8PvcXf70gTDZBgt9ptxJ8elBeBjcLOAcOtoO/mPJjtji1+CdGbHgm77om1GrsPxsiE+uXIpNSK64UYaIwQXd4Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/types": "^7.28.2" + } + }, + "node_modules/@types/graceful-fs": { + "version": "4.1.9", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", + "integrity": "sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/istanbul-lib-coverage": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz", + "integrity": "sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/istanbul-lib-report": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.3.tgz", + "integrity": "sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-coverage": "*" + } + }, + "node_modules/@types/istanbul-reports": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.4.tgz", + "integrity": "sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/istanbul-lib-report": "*" + } + }, + "node_modules/@types/jest": { + "version": "29.5.14", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.14.tgz", + "integrity": "sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "expect": "^29.0.0", + "pretty-format": "^29.0.0" + } + }, + "node_modules/@types/node": { + "version": "25.0.9", + "resolved": "https://registry.npmjs.org/@types/node/-/node-25.0.9.tgz", + "integrity": "sha512-/rpCXHlCWeqClNBwUhDcusJxXYDjZTyE8v5oTO7WbL8eij2nKhUeU89/6xgjU7N4/Vh3He0BtyhJdQbDyhiXAw==", + "dev": true, + "license": "MIT", + "dependencies": { + "undici-types": "~7.16.0" + } + }, + "node_modules/@types/prop-types": { + "version": "15.7.15", + "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz", + "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/react": { + "version": "18.3.27", + "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.27.tgz", + "integrity": "sha512-cisd7gxkzjBKU2GgdYrTdtQx1SORymWyaAFhaxQPK9bYO9ot3Y5OikQRvY0VYQtvwjeQnizCINJAenh/V7MK2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/prop-types": "*", + "csstype": "^3.2.2" + } + }, + "node_modules/@types/react-native": { + "version": "0.72.8", + "resolved": "https://registry.npmjs.org/@types/react-native/-/react-native-0.72.8.tgz", + "integrity": "sha512-St6xA7+EoHN5mEYfdWnfYt0e8u6k2FR0P9s2arYgakQGFgU1f9FlPrIEcj0X24pLCF5c5i3WVuLCUdiCYHmOoA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@react-native/virtualized-lists": "^0.72.4", + "@types/react": "*" + } + }, + "node_modules/@types/stack-utils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.3.tgz", + "integrity": "sha512-9aEbYZ3TbYMznPdcdr3SmIrLXwC/AKZXQeCf9Pgao5CKb8CyHuEX5jzWPTkvregvhRJHcpRO6BFoGW9ycaOkYw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/yargs": { + "version": "17.0.35", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.35.tgz", + "integrity": "sha512-qUHkeCyQFxMXg79wQfTtfndEC+N9ZZg76HJftDJp+qH2tV7Gj4OJi7l+PiWwJ+pWtW8GwSmqsDj/oymhrTWXjg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/yargs-parser": { + "version": "21.0.3", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.3.tgz", + "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "license": "ISC", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "license": "MIT", + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/babel-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", + "integrity": "sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/transform": "^29.7.0", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^29.6.3", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.8.0" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/node_modules/istanbul-lib-instrument": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", + "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-jest-hoist": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", + "integrity": "sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.1.14", + "@types/babel__traverse": "^7.0.6" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.2.0.tgz", + "integrity": "sha512-E/VlAEzRrsLEb2+dv8yp3bo4scof3l9nR4lrld+Iy5NyVqgVYUJnDAmunkhPMisRI32Qc4iRiz425d8vM++2fg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.12.13", + "@babel/plugin-syntax-class-static-block": "^7.14.5", + "@babel/plugin-syntax-import-attributes": "^7.24.7", + "@babel/plugin-syntax-import-meta": "^7.10.4", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-private-property-in-object": "^7.14.5", + "@babel/plugin-syntax-top-level-await": "^7.14.5" + }, + "peerDependencies": { + "@babel/core": "^7.0.0 || ^8.0.0-0" + } + }, + "node_modules/babel-preset-jest": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-29.6.3.tgz", + "integrity": "sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==", + "dev": true, + "license": "MIT", + "dependencies": { + "babel-plugin-jest-hoist": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/baseline-browser-mapping": { + "version": "2.9.15", + "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.9.15.tgz", + "integrity": "sha512-kX8h7K2srmDyYnXRIppo4AH/wYgzWVCs+eKr3RusRSQ5PvRYoEFmR/I0PbdTjKFAoKqp5+kbxnNTFO9jOfSVJg==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "baseline-browser-mapping": "dist/cli.js" + } + }, + "node_modules/brace-expansion": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz", + "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.28.1", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.1.tgz", + "integrity": "sha512-ZC5Bd0LgJXgwGqUknZY/vkUQ04r8NXnJZ3yYi4vDmSiZmC/pdSN0NbNRPxZpbtO4uAfDUAFffO8IZoM3Gj8IkA==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "baseline-browser-mapping": "^2.9.0", + "caniuse-lite": "^1.0.30001759", + "electron-to-chromium": "^1.5.263", + "node-releases": "^2.0.27", + "update-browserslist-db": "^1.2.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bs-logger": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", + "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-json-stable-stringify": "2.x" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001765", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001765.tgz", + "integrity": "sha512-LWcNtSyZrakjECqmpP4qdg0MMGdN368D7X8XvvAqOcqMv0RxnlqVKZl2V6/mBR68oYMxOZPLw/gO7DuisMHUvQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "CC-BY-4.0" + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/ci-info": { + "version": "3.9.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", + "integrity": "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/sibiraj-s" + } + ], + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/cjs-module-lexer": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.4.3.tgz", + "integrity": "sha512-9z8TZaGM1pfswYeXrUpzPrkx8UnWYdhJclsiYMm6x/w5+nN+8Tf/LnAgfLGQCm59qAOxU8WwHEq2vNwF6i4j+Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "license": "ISC", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.3.tgz", + "integrity": "sha512-1L5aqIkwPfiodaMgQunkF1zRhNqifHBmtbbbxcr6yVxxBnliw4TDOW6NxpO8DJLgJ16OT+Y4ztZqP6p/FtXnAw==", + "dev": true, + "license": "MIT" + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true, + "license": "MIT" + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true, + "license": "MIT" + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "prompts": "^2.0.1" + }, + "bin": { + "create-jest": "bin/create-jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/csstype": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", + "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/debug": { + "version": "4.4.3", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz", + "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/dedent": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.7.1.tgz", + "integrity": "sha512-9JmrhGZpOlEgOLdQgSm0zxFaYoQon408V1v49aqTWuXENVlnCuY9JBZcXZiCsZQWDjTm5Qf/nIvAy77mXDAjEg==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "babel-plugin-macros": "^3.1.0" + }, + "peerDependenciesMeta": { + "babel-plugin-macros": { + "optional": true + } + } + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.5.267", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.267.tgz", + "integrity": "sha512-0Drusm6MVRXSOJpGbaSVgcQsuB4hEkMpHXaVstcPmhu5LIedxs1xNK/nIxmQIU/RPC0+1/o0AVZfBTkTNJOdUw==", + "dev": true, + "license": "ISC" + }, + "node_modules/emittery": { + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.13.1.tgz", + "integrity": "sha512-DeWwawk6r5yR9jFgnDKYt4sLS0LmHJJi3ZOnb5/JdbYwj3nW+FxQnHIjhBKz8YLC7oRNPVM9NQ47I3CVx34eqQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true, + "license": "MIT" + }, + "node_modules/error-ex": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.4.tgz", + "integrity": "sha512-sqQamAnR14VgCr1A618A3sGrygcpK+HEbenA/HiEAkkUwcZIIB/tgWqHFxWgOyDh4nB4JCRimh79dR5Ywc9MDQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/escalade": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", + "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "license": "BSD-2-Clause", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "license": "MIT", + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/expect-utils": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fb-watchman": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", + "integrity": "sha512-p5161BqbuCaSnB8jIbzQHOlpgsPmK5rJVDfDKO91Axs5NC1uu3HRQm6wt9cd9/+GtQQIO53JdGXXoyDpTAsgYA==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "bser": "2.1.1" + } + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "license": "MIT", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/fs.realpath": { "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true, + "license": "ISC" + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "license": "ISC", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "license": "ISC", + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/handlebars": { + "version": "4.7.8", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.8.tgz", + "integrity": "sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "minimist": "^1.2.5", + "neo-async": "^2.6.2", + "source-map": "^0.6.1", + "wordwrap": "^1.0.0" + }, + "bin": { + "handlebars": "bin/handlebars" + }, + "engines": { + "node": ">=0.4.7" + }, + "optionalDependencies": { + "uglify-js": "^3.1.4" + } + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true, + "license": "MIT" + }, + "node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/import-local": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", + "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", + "dev": true, + "license": "MIT", + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "license": "ISC", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/invariant": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", + "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "dev": true, + "license": "MIT", + "dependencies": { + "loose-envify": "^1.0.0" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "dev": true, + "license": "MIT" + }, + "node_modules/is-core-module": { + "version": "2.16.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", + "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", + "dev": true, + "license": "MIT", + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", + "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-6.0.3.tgz", + "integrity": "sha512-Vtgk7L/R2JHyyGW07spoFlB8/lpjiOLTjMdms6AFMraYt3BaJauod/NGrfnVG/y4Ix1JEuMRPDPEj2ua+zz1/Q==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "@babel/core": "^7.23.9", + "@babel/parser": "^7.23.9", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-instrument/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-reports": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.2.0.tgz", + "integrity": "sha512-HGYWWS/ehqTV3xN10i23tkPkpH46MLCIMFNCaaKNavAXTF1RkqxawEPtnjnGZ6XKSInBKkiOA5BKS+aZiY3AvA==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", + "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/types": "^29.6.3", + "import-local": "^3.0.2", + "jest-cli": "^29.7.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-29.7.0.tgz", + "integrity": "sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==", + "dev": true, + "license": "MIT", + "dependencies": { + "execa": "^5.0.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-29.7.0.tgz", + "integrity": "sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^1.0.0", + "is-generator-fn": "^2.0.0", + "jest-each": "^29.7.0", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0", + "pretty-format": "^29.7.0", + "pure-rand": "^6.0.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-cli": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", + "integrity": "sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "create-jest": "^29.7.0", + "exit": "^0.1.2", + "import-local": "^3.0.2", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "yargs": "^17.3.1" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-config": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-29.7.0.tgz", + "integrity": "sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/test-sequencer": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-jest": "^29.7.0", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-circus": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@types/node": "*", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-diff": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", + "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-docblock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", + "integrity": "sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "detect-newline": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-29.7.0.tgz", + "integrity": "sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "jest-util": "^29.7.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-environment-node": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", + "integrity": "sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-29.7.0.tgz", + "integrity": "sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/graceful-fs": "^4.1.3", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "micromatch": "^4.0.4", + "walker": "^1.0.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + } + }, + "node_modules/jest-leak-detector": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-29.7.0.tgz", + "integrity": "sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", + "integrity": "sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", + "integrity": "sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.6.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-mock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", + "integrity": "sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.3.tgz", + "integrity": "sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", + "integrity": "sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-29.7.0.tgz", + "integrity": "sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==", + "dev": true, + "license": "MIT", + "dependencies": { + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "resolve": "^1.20.0", + "resolve.exports": "^2.0.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-29.7.0.tgz", + "integrity": "sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==", + "dev": true, + "license": "MIT", + "dependencies": { + "jest-regex-util": "^29.6.3", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runner": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-29.7.0.tgz", + "integrity": "sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/environment": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-leak-detector": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-resolve": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-util": "^29.7.0", + "jest-watcher": "^29.7.0", + "jest-worker": "^29.7.0", + "p-limit": "^3.1.0", + "source-map-support": "0.5.13" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runtime": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-29.7.0.tgz", + "integrity": "sha512-gUnLjgwdGqW7B4LvOIkbKs9WGbn+QLqRQQ9juC6HndeDiezIwhDP+mhMwHWCEcfQ5RUXa6OPnFF8BJh5xegwwQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/globals": "^29.7.0", + "@jest/source-map": "^29.6.3", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "cjs-module-lexer": "^1.0.0", + "collect-v8-coverage": "^1.0.0", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-29.7.0.tgz", + "integrity": "sha512-Rm0BMWtxBcioHr1/OX5YCP8Uov4riHvKPknOGs804Zg9JGZgmIBkbtlxJC/7Z4msKYVbIJtfU+tKb8xlYNfdkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/core": "^7.11.6", + "@babel/generator": "^7.7.2", + "@babel/plugin-syntax-jsx": "^7.7.2", + "@babel/plugin-syntax-typescript": "^7.7.2", + "@babel/types": "^7.3.3", + "@jest/expect-utils": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0", + "chalk": "^4.0.0", + "expect": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "natural-compare": "^1.4.0", + "pretty-format": "^29.7.0", + "semver": "^7.5.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jest-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", + "integrity": "sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-29.7.0.tgz", + "integrity": "sha512-ZB7wHqaRGVw/9hST/OuFUReG7M8vKeq0/J2egIGLdvjHCmYqGARhzXmtgi+gVeZ5uXFF219aOc3Ls2yLg27tkw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/types": "^29.6.3", + "camelcase": "^6.2.0", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "leven": "^3.1.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-watcher": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", + "integrity": "sha512-49Fg7WXkU3Vl2h6LbLtMQ/HyB6rXSIX7SqvBLQmssRBGN9I0PNvPmAmCWSOY6SOvrjhI/F7/bGAv9RtnsPA03g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "jest-util": "^29.7.0", + "string-length": "^4.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-worker": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-29.7.0.tgz", + "integrity": "sha512-eIz2msL/EzL9UFTFFx7jBTkeZfku0yUAyZZZmJ93H2TYEiroIx2PQjEXcwYtYl8zXCxb+PAmA2hLIt/6ZEkPHw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*", + "jest-util": "^29.7.0", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/js-yaml": { + "version": "3.14.2", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.2.tgz", + "integrity": "sha512-PMSmkqxr106Xa156c2M265Z+FTrPl+oxd/rgOQy2tijQeK5TxQ43psO1ZCwhVOSdnn+RzkzlRz/eY4BgJBYVpg==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsesc": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz", + "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==", + "dev": true, + "license": "MIT", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true, + "license": "MIT" + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "license": "MIT", + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true, + "license": "MIT" + }, + "node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", + "dev": true, + "license": "MIT" + }, + "node_modules/loose-envify": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", + "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "license": "ISC", + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "license": "MIT", + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-dir/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true, + "license": "ISC" + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true, + "license": "MIT" + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "dev": true, + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/node-releases": { + "version": "2.0.27", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.27.tgz", + "integrity": "sha512-nmh3lCkYZ3grZvqcCH+fjmQ7X+H0OeZgP40OierEaAptX4XofMh5kwNbWh7lBduUzCcV/8kZ+NDLCwm2iorIlA==", + "dev": true, + "license": "MIT" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/nullthrows": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/nullthrows/-/nullthrows-1.1.1.tgz", + "integrity": "sha512-2vPPEi+Z7WqML2jZYddDIfy5Dqb0r2fze2zTxNNknZaFpVHU3mFB3R+DWeJWGVx0ecvttSGlJTI+WG+8Z4cDWw==", + "dev": true, + "license": "MIT" + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-locate/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true, + "license": "MIT" + }, + "node_modules/picocolors": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", + "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", + "dev": true, + "license": "ISC" + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz", + "integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/pure-rand": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-6.1.0.tgz", + "integrity": "sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/dubzzz" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fast-check" + } + ], + "license": "MIT" + }, + "node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true, + "license": "MIT" + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.11", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.11.tgz", + "integrity": "sha512-RfqAvLnMl313r7c9oclB1HhUEAezcpLjz95wFH4LVuhk9JF/r22qmVP9AMmOU4vMX7Q8pN8jwNg/CSpdFnMjTQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-core-module": "^2.16.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, "license": "MIT", "dependencies": { - "react-native-vision-ml": "^0.1.1" - }, - "devDependencies": { - "@types/react": "^18.0.0", - "@types/react-native": "^0.72.0", - "typescript": "^5.0.0" + "resolve-from": "^5.0.0" }, - "peerDependencies": { - "react": "*", - "react-native": "*" + "engines": { + "node": ">=8" } }, - "node_modules/@react-native/virtualized-lists": { - "version": "0.72.8", - "resolved": "https://registry.npmjs.org/@react-native/virtualized-lists/-/virtualized-lists-0.72.8.tgz", - "integrity": "sha512-J3Q4Bkuo99k7mu+jPS9gSUSgq+lLRSI/+ahXNwV92XgJ/8UgOTxu2LPwhJnBk/sQKxq7E8WkZBnBiozukQMqrw==", + "node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve.exports": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.3.tgz", + "integrity": "sha512-OcXjMsGdhL4XnbShKpAcSqPMzQoYkYyhbEaeSko47MjRP9NfEQMhZkXL1DoFlt9LWQn4YttrdnV6X2OiyzBi+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", "dev": true, "license": "MIT", "dependencies": { - "invariant": "^2.2.4", - "nullthrows": "^1.1.1" + "shebang-regex": "^3.0.0" }, - "peerDependencies": { - "react-native": "*" + "engines": { + "node": ">=8" } }, - "node_modules/@types/prop-types": { - "version": "15.7.15", - "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz", - "integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==", + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", "dev": true, "license": "MIT" }, - "node_modules/@types/react": { - "version": "18.3.27", - "resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.27.tgz", - "integrity": "sha512-cisd7gxkzjBKU2GgdYrTdtQx1SORymWyaAFhaxQPK9bYO9ot3Y5OikQRvY0VYQtvwjeQnizCINJAenh/V7MK2w==", + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.13", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.13.tgz", + "integrity": "sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w==", "dev": true, "license": "MIT", "dependencies": { - "@types/prop-types": "*", - "csstype": "^3.2.2" + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" } }, - "node_modules/@types/react-native": { - "version": "0.72.8", - "resolved": "https://registry.npmjs.org/@types/react-native/-/react-native-0.72.8.tgz", - "integrity": "sha512-St6xA7+EoHN5mEYfdWnfYt0e8u6k2FR0P9s2arYgakQGFgU1f9FlPrIEcj0X24pLCF5c5i3WVuLCUdiCYHmOoA==", + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/stack-utils": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.6.tgz", + "integrity": "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==", "dev": true, "license": "MIT", "dependencies": { - "@react-native/virtualized-lists": "^0.72.4", - "@types/react": "*" + "escape-string-regexp": "^2.0.0" + }, + "engines": { + "node": ">=10" } }, - "node_modules/csstype": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz", - "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==", + "node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", "dev": true, - "license": "MIT" + "license": "MIT", + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } }, - "node_modules/invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", + "node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", "dev": true, "license": "MIT", "dependencies": { - "loose-envify": "^1.0.0" + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" } }, - "node_modules/js-tokens": { + "node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", "dev": true, - "license": "MIT" + "license": "MIT", + "engines": { + "node": ">=8" + } }, - "node_modules/loose-envify": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", - "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", "dev": true, "license": "MIT", "dependencies": { - "js-tokens": "^3.0.0 || ^4.0.0" + "has-flag": "^4.0.0" }, - "bin": { - "loose-envify": "cli.js" + "engines": { + "node": ">=8" } }, - "node_modules/nullthrows": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/nullthrows/-/nullthrows-1.1.1.tgz", - "integrity": "sha512-2vPPEi+Z7WqML2jZYddDIfy5Dqb0r2fze2zTxNNknZaFpVHU3mFB3R+DWeJWGVx0ecvttSGlJTI+WG+8Z4cDWw==", + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", "dev": true, - "license": "MIT" + "license": "MIT", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dev": true, + "license": "ISC", + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tmpl": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", + "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==", + "dev": true, + "license": "BSD-3-Clause" + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } }, - "node_modules/react-native-vision-ml": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/react-native-vision-ml/-/react-native-vision-ml-0.1.1.tgz", - "integrity": "sha512-Ywqkixzx/NvSHN+Ez4aGnBiOpzDPwQFHO8GvZxozfPQWgpzd32BoGHCAq3hKBUSOa7UAhEVjAa1fV/gwnYh5+g==", + "node_modules/ts-jest": { + "version": "29.4.6", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.4.6.tgz", + "integrity": "sha512-fSpWtOO/1AjSNQguk43hb/JCo16oJDnMJf3CdEGNkqsEX3t0KX96xvyX1D7PfLCpVoKu4MfVrqUkFyblYoY4lA==", + "dev": true, "license": "MIT", + "dependencies": { + "bs-logger": "^0.2.6", + "fast-json-stable-stringify": "^2.1.0", + "handlebars": "^4.7.8", + "json5": "^2.2.3", + "lodash.memoize": "^4.1.2", + "make-error": "^1.3.6", + "semver": "^7.7.3", + "type-fest": "^4.41.0", + "yargs-parser": "^21.1.1" + }, + "bin": { + "ts-jest": "cli.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || ^18.0.0 || >=20.0.0" + }, "peerDependencies": { - "expo": "*", - "react": "*", - "react-native": "*" + "@babel/core": ">=7.0.0-beta.0 <8", + "@jest/transform": "^29.0.0 || ^30.0.0", + "@jest/types": "^29.0.0 || ^30.0.0", + "babel-jest": "^29.0.0 || ^30.0.0", + "jest": "^29.0.0 || ^30.0.0", + "jest-util": "^29.0.0 || ^30.0.0", + "typescript": ">=4.3 <6" + }, + "peerDependenciesMeta": { + "@babel/core": { + "optional": true + }, + "@jest/transform": { + "optional": true + }, + "@jest/types": { + "optional": true + }, + "babel-jest": { + "optional": true + }, + "esbuild": { + "optional": true + }, + "jest-util": { + "optional": true + } + } + }, + "node_modules/ts-jest/node_modules/semver": { + "version": "7.7.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.3.tgz", + "integrity": "sha512-SdsKMrI9TdgjdweUSR9MweHA4EJ8YxHn8DFaDisvhVlUOe4BF1tLD7GAj0lIqWVl+dPb/rExr0Btby5loQm20Q==", + "dev": true, + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/ts-jest/node_modules/type-fest": { + "version": "4.41.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.41.0.tgz", + "integrity": "sha512-TeTSQ6H5YHvpqVwBRcnLDCBnDOHWYu7IvGbHT6N8AOymcr9PJGjc1GTtiWZTYg0NCgYwvnYWEkVChQAr9bjfwA==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=16" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "license": "(MIT OR CC0-1.0)", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, "node_modules/typescript": { @@ -132,6 +3726,204 @@ "engines": { "node": ">=14.17" } + }, + "node_modules/uglify-js": { + "version": "3.19.3", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.19.3.tgz", + "integrity": "sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==", + "dev": true, + "license": "BSD-2-Clause", + "optional": true, + "bin": { + "uglifyjs": "bin/uglifyjs" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/undici-types": { + "version": "7.16.0", + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-7.16.0.tgz", + "integrity": "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw==", + "dev": true, + "license": "MIT" + }, + "node_modules/update-browserslist-db": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz", + "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "license": "MIT", + "dependencies": { + "escalade": "^3.2.0", + "picocolors": "^1.1.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/v8-to-istanbul": { + "version": "9.3.0", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.3.0.tgz", + "integrity": "sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA==", + "dev": true, + "license": "ISC", + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.12", + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^2.0.0" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/walker": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", + "integrity": "sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "makeerror": "1.0.12" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/wordwrap": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", + "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true, + "license": "ISC" + }, + "node_modules/write-file-atomic": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-4.0.2.tgz", + "integrity": "sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==", + "dev": true, + "license": "ISC", + "dependencies": { + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.7" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + } + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true, + "license": "ISC" + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dev": true, + "license": "MIT", + "dependencies": { + "cliui": "^8.0.1", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.3", + "y18n": "^5.0.5", + "yargs-parser": "^21.1.1" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dev": true, + "license": "ISC", + "engines": { + "node": ">=12" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } } } } diff --git a/package.json b/package.json index 995e631..fe4e70c 100644 --- a/package.json +++ b/package.json @@ -9,18 +9,26 @@ "nativeModulesDir": "./ios" } }, + "scripts": { + "test": "jest", + "test:watch": "jest --watch", + "test:coverage": "jest --coverage", + "typecheck": "tsc --noEmit" + }, "author": "Neher Data", "license": "MIT", "keywords": [ "react-native", "ios", + "android", "onnx", "yolo", "vision", "machine-learning", "nsfw-detection", "object-detection", - "coreml" + "coreml", + "mlkit" ], "repository": { "type": "git", @@ -35,13 +43,17 @@ "react-native": "*" }, "devDependencies": { + "@types/jest": "^29.5.0", "@types/react": "^18.0.0", "@types/react-native": "^0.72.0", + "jest": "^29.7.0", + "ts-jest": "^29.1.0", "typescript": "^5.0.0" }, "files": [ "src/", "ios/", + "android/", "react-native-vision-ml.podspec", "README.md" ] From 54b4a387d7843646d53e38ef59f586e58fb3b13b Mon Sep 17 00:00:00 2001 From: Claude Date: Mon, 19 Jan 2026 12:50:13 +0000 Subject: [PATCH 2/2] Add GitHub Actions workflow for self-hosted runners - TypeScript/Jest tests on every push/PR - Android unit tests with Gradle - iOS tests commented out (ready for macOS runner) Runs on self-hosted Linux runners. Uncomment iOS job when MacBook Air is configured as a runner. --- .github/workflows/tests.yml | 71 +++++++++++++++++++++++++++++++++++++ 1 file changed, 71 insertions(+) create mode 100644 .github/workflows/tests.yml diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml new file mode 100644 index 0000000..a195960 --- /dev/null +++ b/.github/workflows/tests.yml @@ -0,0 +1,71 @@ +name: Tests + +on: + push: + branches: [main, master] + pull_request: + branches: [main, master] + +jobs: + typescript-tests: + name: TypeScript Tests + runs-on: self-hosted + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Setup Node.js + uses: actions/setup-node@v4 + with: + node-version: '20' + cache: 'npm' + + - name: Install dependencies + run: npm ci --legacy-peer-deps + + - name: Run TypeScript tests + run: npm test + + - name: Run TypeScript tests with coverage + run: npm run test:coverage + continue-on-error: true + + android-tests: + name: Android Unit Tests + runs-on: self-hosted + + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Setup Java + uses: actions/setup-java@v4 + with: + distribution: 'temurin' + java-version: '17' + + - name: Setup Android SDK + uses: android-actions/setup-android@v3 + + - name: Run Android unit tests + working-directory: android + run: ./gradlew test --no-daemon + + # Uncomment when MacBook Air is set up as a self-hosted runner + # ios-tests: + # name: iOS Unit Tests + # runs-on: self-hosted-macos # Update with your macOS runner label + # + # steps: + # - name: Checkout + # uses: actions/checkout@v4 + # + # - name: Run iOS tests + # working-directory: ios + # run: | + # xcodebuild test \ + # -scheme VisionML \ + # -destination 'platform=iOS Simulator,name=iPhone 15' \ + # -only-testing:VisionMLTests \ + # | xcpretty