feat: replace dataclasses-json with Pydantic v2 models#277
Open
feat: replace dataclasses-json with Pydantic v2 models#277
Conversation
Add pydantic dependency and enable_validation parameter on ApiRouter. All 4 routers now have enable_validation=True with RequestValidationError and ResponseValidationError exception handlers.
…s-json Replace dataclasses-json with Pydantic v2 across the entire codebase: - Convert BasicStorageObject, CollectionData, BatchResult to Pydantic BaseModel with DynamoModel base class handling Decimal/datetime coercion - Add to_item() methods on DynamoDB models for complete item serialization - Migrate all 21 route outputs to Pydantic models (BSOOutput, ModifiedOutput, BatchResultOutput, AccountCreateOutput, ProfileOutput, TokenOutput, etc.) - Add BSOOutput.from_bso() factory for BSO→output conversion - Remove dataclasses-json dependency, DecimalEncoder, json_dumps, and all datetime_encoder/decoder/float_to_decimal helpers - Convert OIDCTokenClaims, OIDCProviderConfig, ErrorDetail, TokenResponse to plain dataclasses (no serialization mixin needed) - Convert UserRecord to DynamoModel with to_item() for DynamoDB writes
Diff for stage: DefaultStageWarning 3 Destructive Changes Diff for stack: GitHubOidcStack - 0 to add, 1 to update, 0 to destroy 🟡DetailsResources
[~] Custom::AWSCDKOpenIdConnectProvider GitHubOidcProvider7EBF861F
├─ [~] CodeHash
│ ├─ [-] 62fa02efcaa700e1c247e1d3cc2aa0cd07a0808a9a3e3d2230e51f57a02233fb
│ └─ [+] d75c48c9f82cde63e9bf414df335e84e8ac24f11eb34889be255b702aec71e50
└─ [~] RejectUnauthorized
[~] AWS::Lambda::Function CustomAWSCDKOpenIdConnectProviderCustomResourceProviderHandlerF2C543E0
├─ [~] Code
│ └─ [~] .S3Key:
│ ├─ [-] 62fa02efcaa700e1c247e1d3cc2aa0cd07a0808a9a3e3d2230e51f57a02233fb.zip
│ └─ [+] d75c48c9f82cde63e9bf414df335e84e8ac24f11eb34889be255b702aec71e50.zip
└─ [~] Metadata
└─ [~] .aws:asset:path:
├─ [-] asset.62fa02efcaa700e1c247e1d3cc2aa0cd07a0808a9a3e3d2230e51f57a02233fb
└─ [+] asset.d75c48c9f82cde63e9bf414df335e84e8ac24f11eb34889be255b702aec71e50
Diff for stack: Service-prod - 3 to add, 3 to update, 3 to destroy ❌Details
Resources
[-] AWS::ApiGateway::Deployment AuthApiDeploymentB62B2E46e6df84b588a887363a13ab29adc3623c destroy
[-] AWS::ApiGateway::Deployment TokenApiDeploymentB896C219b16a73c2bacf13dff150a851137e236c destroy
[-] AWS::ApiGateway::Deployment ProfileApiDeployment84A54415e0158c62574d24c31b8dbd21e82873c4 destroy
[+] AWS::ApiGateway::Deployment AuthApiDeploymentB62B2E46b5c04cfbd7bd117a1ee4f8acc7bb74f8
[+] AWS::ApiGateway::Deployment TokenApiDeploymentB896C2197f4fd8f40928d8ff6b65325e6f60c797
[+] AWS::ApiGateway::Deployment ProfileApiDeployment84A54415afa95090416ea0bbd4b1d2aac5a9241c
[~] AWS::Lambda::Function AuthApiHandlerED50ACFA
├─ [~] Code
│ └─ [~] .S3Key:
│ ├─ [-] 1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147.zip
│ └─ [+] 97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a.zip
└─ [~] Metadata
└─ [~] .aws:asset:path:
├─ [-] asset.1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147
└─ [+] asset.97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a
[~] AWS::Lambda::Function TokenApiHandler2E66DB25
├─ [~] Code
│ └─ [~] .S3Key:
│ ├─ [-] 1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147.zip
│ └─ [+] 97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a.zip
└─ [~] Metadata
└─ [~] .aws:asset:path:
├─ [-] asset.1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147
└─ [+] asset.97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a
[~] AWS::Lambda::Function ProfileApiHandler9B65A298
├─ [~] Code
│ └─ [~] .S3Key:
│ ├─ [-] 1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147.zip
│ └─ [+] 97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a.zip
└─ [~] Metadata
└─ [~] .aws:asset:path:
├─ [-] asset.1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147
└─ [+] asset.97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a
[~] AWS::Lambda::Function ApiHandler5E7490E8
├─ [~] Code
│ └─ [~] .S3Key:
│ ├─ [-] 1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147.zip
│ └─ [+] 97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a.zip
└─ [~] Metadata
└─ [~] .aws:asset:path:
├─ [-] asset.1527c534dc3ec24d621d57e297d669e22dc2ace57cbcae6d6da540d42832a147
└─ [+] asset.97638fb87dc714ebe303cdc70246ff3fa200e74abd5b8cee8e2b4f6261ba707a
[~] AWS::ApiGateway::Stage AuthApiDeploymentStageprodB0E4172A
└─ [~] DeploymentId
└─ [~] .Ref:
├─ [-] AuthApiDeploymentB62B2E46e6df84b588a887363a13ab29adc3623c
└─ [+] AuthApiDeploymentB62B2E46b5c04cfbd7bd117a1ee4f8acc7bb74f8
[~] AWS::ApiGateway::Stage TokenApiDeploymentStageprod11035AE4
└─ [~] DeploymentId
└─ [~] .Ref:
├─ [-] TokenApiDeploymentB896C219b16a73c2bacf13dff150a851137e236c
└─ [+] TokenApiDeploymentB896C2197f4fd8f40928d8ff6b65325e6f60c797
[~] AWS::ApiGateway::Stage ProfileApiDeploymentStageprodF609D968
└─ [~] DeploymentId
└─ [~] .Ref:
├─ [-] ProfileApiDeployment84A54415e0158c62574d24c31b8dbd21e82873c4
└─ [+] ProfileApiDeployment84A54415afa95090416ea0bbd4b1d2aac5a9241c
[~] AWS::Lambda::Function ChannelApiHandler02759D57
├─ [~] Code
│ └─ [~] .S3Key:
│ ├─ [-] ca37232fdfd500adbaa06e972c23a68046efd788f7781db21a059050cda4a508.zip
│ └─ [+] ff9d24ca18cd246f688865f3ee12d7134d22cea4e80a7e842eea35b0ba7ce347.zip
└─ [~] Metadata
└─ [~] .aws:asset:path:
├─ [-] asset.ca37232fdfd500adbaa06e972c23a68046efd788f7781db21a059050cda4a508
└─ [+] asset.ff9d24ca18cd246f688865f3ee12d7134d22cea4e80a7e842eea35b0ba7ce347
No Changes for stack: Frontend-prod ✅ Generated for commit b1d465f at 2026-03-05T06:12:27.528Z |
- Use Field(default=None, ...) for BSOInput fields so mypy recognizes defaults - Use model_validate() for dict/Decimal input tests instead of constructor - Add type: ignore for model_validator descriptor proxy call - Add BSOOutput.from_bso() test coverage
Coverage reportClick to see where and how coverage changed
The report is truncated to 25 files out of 38. To see the full report, please visit the workflow summary page. This report was generated by python-coverage-comment-action |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Replace json.dumps([m.model_dump() ...]) with Pydantic TypeAdapter for single-pass list serialization in device, client, and collection routes. Use a _CollectionsResponse wrapper model instead of string concatenation.
Move BSOListAdapter, DeviceListAdapter, ClientListAdapter, and CollectionsResponse from route modules into shared/models.py alongside the models they serialize.
cdd893a to
b1d465f
Compare
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.This suggestion is invalid because no changes were made to the code.Suggestions cannot be applied while the pull request is closed.Suggestions cannot be applied while viewing a subset of changes.Only one suggestion per line can be applied in a batch.Add this suggestion to a batch that can be applied as a single commit.Applying suggestions on deleted lines is not supported.You must change the existing code in this line in order to create a valid suggestion.Outdated suggestions cannot be applied.This suggestion has been applied or marked resolved.Suggestions cannot be applied from pending reviews.Suggestions cannot be applied on multi-line comments.Suggestions cannot be applied while the pull request is queued to merge.Suggestion cannot be applied right now. Please check back later.
Summary
BSOOutput,ModifiedOutput,BatchResultOutput,AccountCreateOutput,ProfileOutput,TokenOutput, etc.) across 21 routesdataclasses-jsonDataClassJsonMixinwith PydanticBaseModelfor DynamoDB models (BasicStorageObject,CollectionData,UserRecord) via aDynamoModelbase class that handles Decimal/datetime coercionto_item()methods on DynamoDB models for complete item serialization (replaces_encode_*methods in StorageManager/UserManager)dataclasses-jsondependency and all dead helpers (DecimalEncoder,json_dumps,datetime_encoder/decoder,float_to_decimal,validate_sortindex,validate_ttl)enable_validation=Trueon all API routers withRequestValidationError/ResponseValidationErrorhandlersTest plan
dataclasses-jsonfully removed from dependencies (+ transitivemarshmallow,typing-inspect)