feat: add VIN photo OCR pipeline (refs #67)
All checks were successful
Deploy to Staging / Build Images (pull_request) Successful in 31s
Deploy to Staging / Deploy to Staging (pull_request) Successful in 31s
Deploy to Staging / Verify Staging (pull_request) Successful in 2m19s
Deploy to Staging / Notify Staging Ready (pull_request) Successful in 8s
Deploy to Staging / Notify Staging Failure (pull_request) Has been skipped

Implement VIN-specific OCR extraction with optimized preprocessing:

- Add POST /extract/vin endpoint for VIN extraction
- VIN preprocessor: CLAHE, deskew, denoise, adaptive threshold
- VIN validator: check digit validation, OCR error correction (I->1, O->0)
- VIN extractor: PSM modes 6/7/8, character whitelist, alternatives
- Response includes confidence, bounding box, and alternatives
- Unit tests for validator and preprocessor
- Integration tests for VIN extraction endpoint

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
Eric Gullickson
2026-02-01 19:31:36 -06:00
parent 004940b013
commit 54cbd49171
14 changed files with 1694 additions and 1 deletions

View File

@@ -21,6 +21,36 @@ class ExtractedField(BaseModel):
confidence: float = Field(ge=0.0, le=1.0)
class BoundingBox(BaseModel):
"""Bounding box for detected region."""
x: int
y: int
width: int
height: int
class VinAlternative(BaseModel):
"""Alternative VIN candidate."""
vin: str
confidence: float = Field(ge=0.0, le=1.0)
class VinExtractionResponse(BaseModel):
"""Response from VIN extraction endpoint."""
success: bool
vin: Optional[str] = None
confidence: float = Field(ge=0.0, le=1.0)
bounding_box: Optional[BoundingBox] = Field(default=None, alias="boundingBox")
alternatives: list[VinAlternative] = Field(default_factory=list)
processing_time_ms: int = Field(alias="processingTimeMs")
error: Optional[str] = None
model_config = {"populate_by_name": True}
class OcrResponse(BaseModel):
"""Response from OCR extraction."""