feat: create ownership_costs backend feature capsule (refs #29)
Milestone 1: Complete backend feature with: - Migration with CHECK (amount > 0) constraint - Repository with mapRow() for snake_case -> camelCase - Service with CRUD and vehicle authorization - Controller with HTTP handlers - Routes registered at /api/ownership-costs - Validation with Zod schemas - README with endpoint documentation 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
153
backend/src/features/ownership-costs/README.md
Normal file
153
backend/src/features/ownership-costs/README.md
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
# Ownership Costs Feature
|
||||||
|
|
||||||
|
Tracks vehicle ownership costs including insurance, registration, tax, inspection, parking, and other costs.
|
||||||
|
|
||||||
|
## Database Schema
|
||||||
|
|
||||||
|
```sql
|
||||||
|
ownership_costs (
|
||||||
|
id UUID PRIMARY KEY,
|
||||||
|
user_id VARCHAR(255) NOT NULL,
|
||||||
|
vehicle_id UUID NOT NULL REFERENCES vehicles(id) ON DELETE CASCADE,
|
||||||
|
document_id UUID NULL REFERENCES documents(id) ON DELETE CASCADE,
|
||||||
|
cost_type VARCHAR(32) NOT NULL CHECK (cost_type IN ('insurance', 'registration', 'tax', 'inspection', 'parking', 'other')),
|
||||||
|
amount DECIMAL(10, 2) NOT NULL CHECK (amount > 0),
|
||||||
|
description VARCHAR(200),
|
||||||
|
period_start DATE,
|
||||||
|
period_end DATE,
|
||||||
|
notes TEXT,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP
|
||||||
|
)
|
||||||
|
```
|
||||||
|
|
||||||
|
## API Endpoints
|
||||||
|
|
||||||
|
### POST /ownership-costs
|
||||||
|
Create a new ownership cost record.
|
||||||
|
|
||||||
|
**Auth**: Required (JWT)
|
||||||
|
|
||||||
|
**Request Body**:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"vehicleId": "uuid",
|
||||||
|
"documentId": "uuid (optional)",
|
||||||
|
"costType": "insurance | registration | tax | inspection | parking | other",
|
||||||
|
"amount": 150.00,
|
||||||
|
"description": "Auto insurance premium (optional)",
|
||||||
|
"periodStart": "2024-01-01 (optional)",
|
||||||
|
"periodEnd": "2024-12-31 (optional)",
|
||||||
|
"notes": "Additional notes (optional)"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Response**: 201 Created
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"id": "uuid",
|
||||||
|
"userId": "string",
|
||||||
|
"vehicleId": "uuid",
|
||||||
|
"documentId": "uuid",
|
||||||
|
"costType": "insurance",
|
||||||
|
"amount": 150.00,
|
||||||
|
"description": "Auto insurance premium",
|
||||||
|
"periodStart": "2024-01-01",
|
||||||
|
"periodEnd": "2024-12-31",
|
||||||
|
"notes": "Additional notes",
|
||||||
|
"createdAt": "2024-01-01T00:00:00.000Z",
|
||||||
|
"updatedAt": "2024-01-01T00:00:00.000Z"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### GET /ownership-costs
|
||||||
|
List all ownership costs for the authenticated user.
|
||||||
|
|
||||||
|
**Auth**: Required (JWT)
|
||||||
|
|
||||||
|
**Query Parameters**:
|
||||||
|
- `vehicleId` (optional): Filter by vehicle UUID
|
||||||
|
- `costType` (optional): Filter by cost type
|
||||||
|
- `documentId` (optional): Filter by document UUID
|
||||||
|
|
||||||
|
**Response**: 200 OK
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"id": "uuid",
|
||||||
|
"userId": "string",
|
||||||
|
"vehicleId": "uuid",
|
||||||
|
"costType": "insurance",
|
||||||
|
"amount": 150.00,
|
||||||
|
"description": "Auto insurance premium",
|
||||||
|
"periodStart": "2024-01-01",
|
||||||
|
"periodEnd": "2024-12-31",
|
||||||
|
"createdAt": "2024-01-01T00:00:00.000Z",
|
||||||
|
"updatedAt": "2024-01-01T00:00:00.000Z"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
### GET /ownership-costs/:id
|
||||||
|
Get a specific ownership cost record.
|
||||||
|
|
||||||
|
**Auth**: Required (JWT)
|
||||||
|
|
||||||
|
**Response**: 200 OK (same as POST response)
|
||||||
|
|
||||||
|
**Errors**:
|
||||||
|
- 404 Not Found: Cost not found or not owned by user
|
||||||
|
|
||||||
|
### PUT /ownership-costs/:id
|
||||||
|
Update an ownership cost record.
|
||||||
|
|
||||||
|
**Auth**: Required (JWT)
|
||||||
|
|
||||||
|
**Request Body**: (all fields optional)
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"documentId": "uuid",
|
||||||
|
"costType": "insurance",
|
||||||
|
"amount": 160.00,
|
||||||
|
"description": "Updated description",
|
||||||
|
"periodStart": "2024-01-01",
|
||||||
|
"periodEnd": "2024-12-31",
|
||||||
|
"notes": "Updated notes"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Response**: 200 OK (same as POST response)
|
||||||
|
|
||||||
|
**Errors**:
|
||||||
|
- 404 Not Found: Cost not found or not owned by user
|
||||||
|
|
||||||
|
### DELETE /ownership-costs/:id
|
||||||
|
Delete an ownership cost record.
|
||||||
|
|
||||||
|
**Auth**: Required (JWT)
|
||||||
|
|
||||||
|
**Response**: 204 No Content
|
||||||
|
|
||||||
|
**Errors**:
|
||||||
|
- 404 Not Found: Cost not found or not owned by user
|
||||||
|
|
||||||
|
## Cost Types
|
||||||
|
|
||||||
|
| Type | Description |
|
||||||
|
|------|-------------|
|
||||||
|
| insurance | Auto insurance premiums |
|
||||||
|
| registration | Vehicle registration fees |
|
||||||
|
| tax | Property or excise taxes |
|
||||||
|
| inspection | State inspection fees |
|
||||||
|
| parking | Parking permits, monthly fees |
|
||||||
|
| other | Other ownership costs |
|
||||||
|
|
||||||
|
## Authorization
|
||||||
|
|
||||||
|
All endpoints enforce user scoping - users can only access their own ownership cost records. Vehicle ownership is verified before creating records.
|
||||||
|
|
||||||
|
## Integration with Other Features
|
||||||
|
|
||||||
|
- **Documents**: Optional document_id links costs to supporting documents
|
||||||
|
- **Vehicles**: vehicle_id foreign key with CASCADE delete
|
||||||
|
- **TCO Calculation**: Service provides `getVehicleOwnershipCosts()` for total cost of ownership aggregation
|
||||||
@@ -1,225 +1,145 @@
|
|||||||
/**
|
import { FastifyReply, FastifyRequest } from 'fastify';
|
||||||
* @ai-summary Fastify route handlers for ownership costs API
|
|
||||||
* @ai-context HTTP request/response handling with Fastify reply methods
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { FastifyRequest, FastifyReply } from 'fastify';
|
|
||||||
import { OwnershipCostsService } from '../domain/ownership-costs.service';
|
import { OwnershipCostsService } from '../domain/ownership-costs.service';
|
||||||
import { pool } from '../../../core/config/database';
|
import type { CreateBody, UpdateBody, IdParams, ListQuery } from './ownership-costs.validation';
|
||||||
import { logger } from '../../../core/logging/logger';
|
import { logger } from '../../../core/logging/logger';
|
||||||
import {
|
|
||||||
OwnershipCostParams,
|
|
||||||
VehicleParams,
|
|
||||||
CreateOwnershipCostBody,
|
|
||||||
UpdateOwnershipCostBody
|
|
||||||
} from '../domain/ownership-costs.types';
|
|
||||||
|
|
||||||
export class OwnershipCostsController {
|
export class OwnershipCostsController {
|
||||||
private service: OwnershipCostsService;
|
private readonly service = new OwnershipCostsService();
|
||||||
|
|
||||||
constructor() {
|
async list(request: FastifyRequest<{ Querystring: ListQuery }>, reply: FastifyReply) {
|
||||||
this.service = new OwnershipCostsService(pool);
|
const userId = (request as any).user?.sub as string;
|
||||||
}
|
|
||||||
|
|
||||||
async create(request: FastifyRequest<{ Body: CreateOwnershipCostBody }>, reply: FastifyReply) {
|
logger.info('Ownership costs list requested', {
|
||||||
try {
|
operation: 'ownership-costs.list',
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
userId,
|
||||||
const userId = (request as any).user.sub;
|
filters: {
|
||||||
const cost = await this.service.create(request.body, userId);
|
vehicleId: request.query.vehicleId,
|
||||||
|
costType: request.query.costType,
|
||||||
return reply.code(201).send(cost);
|
documentId: request.query.documentId,
|
||||||
} catch (error: unknown) {
|
},
|
||||||
const err = error as Error;
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
logger.error('Error creating ownership cost', { error: err, userId: (request as any).user?.sub });
|
|
||||||
|
|
||||||
if (err.message.includes('not found')) {
|
|
||||||
return reply.code(404).send({
|
|
||||||
error: 'Not Found',
|
|
||||||
message: err.message
|
|
||||||
});
|
});
|
||||||
}
|
|
||||||
if (err.message.includes('Unauthorized')) {
|
const costs = await this.service.getCosts(userId, {
|
||||||
return reply.code(403).send({
|
vehicleId: request.query.vehicleId,
|
||||||
error: 'Forbidden',
|
costType: request.query.costType,
|
||||||
message: err.message
|
documentId: request.query.documentId,
|
||||||
});
|
});
|
||||||
}
|
|
||||||
|
|
||||||
return reply.code(500).send({
|
logger.info('Ownership costs list retrieved', {
|
||||||
error: 'Internal server error',
|
operation: 'ownership-costs.list.success',
|
||||||
message: 'Failed to create ownership cost'
|
userId,
|
||||||
|
costCount: costs.length,
|
||||||
});
|
});
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getByVehicle(request: FastifyRequest<{ Params: VehicleParams }>, reply: FastifyReply) {
|
|
||||||
try {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
const userId = (request as any).user.sub;
|
|
||||||
const { vehicleId } = request.params;
|
|
||||||
|
|
||||||
const costs = await this.service.getByVehicleId(vehicleId, userId);
|
|
||||||
|
|
||||||
return reply.code(200).send(costs);
|
return reply.code(200).send(costs);
|
||||||
} catch (error: unknown) {
|
}
|
||||||
const err = error as Error;
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
logger.error('Error listing ownership costs', { error: err, vehicleId: request.params.vehicleId, userId: (request as any).user?.sub });
|
|
||||||
|
|
||||||
if (err.message.includes('not found')) {
|
async get(request: FastifyRequest<{ Params: IdParams }>, reply: FastifyReply) {
|
||||||
return reply.code(404).send({
|
const userId = (request as any).user?.sub as string;
|
||||||
error: 'Not Found',
|
const costId = request.params.id;
|
||||||
message: err.message
|
|
||||||
|
logger.info('Ownership cost get requested', {
|
||||||
|
operation: 'ownership-costs.get',
|
||||||
|
userId,
|
||||||
|
costId,
|
||||||
});
|
});
|
||||||
}
|
|
||||||
if (err.message.includes('Unauthorized')) {
|
const cost = await this.service.getCost(userId, costId);
|
||||||
return reply.code(403).send({
|
if (!cost) {
|
||||||
error: 'Forbidden',
|
logger.warn('Ownership cost not found', {
|
||||||
message: err.message
|
operation: 'ownership-costs.get.not_found',
|
||||||
|
userId,
|
||||||
|
costId,
|
||||||
});
|
});
|
||||||
|
return reply.code(404).send({ error: 'Not Found' });
|
||||||
}
|
}
|
||||||
|
|
||||||
return reply.code(500).send({
|
logger.info('Ownership cost retrieved', {
|
||||||
error: 'Internal server error',
|
operation: 'ownership-costs.get.success',
|
||||||
message: 'Failed to get ownership costs'
|
userId,
|
||||||
|
costId,
|
||||||
|
vehicleId: cost.vehicleId,
|
||||||
|
costType: cost.costType,
|
||||||
});
|
});
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getById(request: FastifyRequest<{ Params: OwnershipCostParams }>, reply: FastifyReply) {
|
|
||||||
try {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
const userId = (request as any).user.sub;
|
|
||||||
const { id } = request.params;
|
|
||||||
|
|
||||||
const cost = await this.service.getById(id, userId);
|
|
||||||
|
|
||||||
return reply.code(200).send(cost);
|
return reply.code(200).send(cost);
|
||||||
} catch (error: unknown) {
|
}
|
||||||
const err = error as Error;
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
logger.error('Error getting ownership cost', { error: err, costId: request.params.id, userId: (request as any).user?.sub });
|
|
||||||
|
|
||||||
if (err.message.includes('not found')) {
|
async create(request: FastifyRequest<{ Body: CreateBody }>, reply: FastifyReply) {
|
||||||
return reply.code(404).send({
|
const userId = (request as any).user?.sub as string;
|
||||||
error: 'Not Found',
|
|
||||||
message: err.message
|
logger.info('Ownership cost create requested', {
|
||||||
|
operation: 'ownership-costs.create',
|
||||||
|
userId,
|
||||||
|
vehicleId: request.body.vehicleId,
|
||||||
|
costType: request.body.costType,
|
||||||
|
amount: request.body.amount,
|
||||||
});
|
});
|
||||||
}
|
|
||||||
if (err.message.includes('Unauthorized')) {
|
const created = await this.service.createCost(userId, request.body);
|
||||||
return reply.code(403).send({
|
|
||||||
error: 'Forbidden',
|
logger.info('Ownership cost created', {
|
||||||
message: err.message
|
operation: 'ownership-costs.create.success',
|
||||||
|
userId,
|
||||||
|
costId: created.id,
|
||||||
|
vehicleId: created.vehicleId,
|
||||||
|
costType: created.costType,
|
||||||
|
amount: created.amount,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
return reply.code(201).send(created);
|
||||||
}
|
}
|
||||||
|
|
||||||
return reply.code(500).send({
|
async update(request: FastifyRequest<{ Params: IdParams; Body: UpdateBody }>, reply: FastifyReply) {
|
||||||
error: 'Internal server error',
|
const userId = (request as any).user?.sub as string;
|
||||||
message: 'Failed to get ownership cost'
|
const costId = request.params.id;
|
||||||
|
|
||||||
|
logger.info('Ownership cost update requested', {
|
||||||
|
operation: 'ownership-costs.update',
|
||||||
|
userId,
|
||||||
|
costId,
|
||||||
|
updateFields: Object.keys(request.body),
|
||||||
});
|
});
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async update(request: FastifyRequest<{ Params: OwnershipCostParams; Body: UpdateOwnershipCostBody }>, reply: FastifyReply) {
|
const updated = await this.service.updateCost(userId, costId, request.body);
|
||||||
try {
|
if (!updated) {
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
logger.warn('Ownership cost not found for update', {
|
||||||
const userId = (request as any).user.sub;
|
operation: 'ownership-costs.update.not_found',
|
||||||
const { id } = request.params;
|
userId,
|
||||||
|
costId,
|
||||||
const updatedCost = await this.service.update(id, request.body, userId);
|
|
||||||
|
|
||||||
return reply.code(200).send(updatedCost);
|
|
||||||
} catch (error: unknown) {
|
|
||||||
const err = error as Error;
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
logger.error('Error updating ownership cost', { error: err, costId: request.params.id, userId: (request as any).user?.sub });
|
|
||||||
|
|
||||||
if (err.message.includes('not found')) {
|
|
||||||
return reply.code(404).send({
|
|
||||||
error: 'Not Found',
|
|
||||||
message: err.message
|
|
||||||
});
|
});
|
||||||
|
return reply.code(404).send({ error: 'Not Found' });
|
||||||
}
|
}
|
||||||
if (err.message.includes('Unauthorized')) {
|
|
||||||
return reply.code(403).send({
|
logger.info('Ownership cost updated', {
|
||||||
error: 'Forbidden',
|
operation: 'ownership-costs.update.success',
|
||||||
message: err.message
|
userId,
|
||||||
|
costId,
|
||||||
|
vehicleId: updated.vehicleId,
|
||||||
|
costType: updated.costType,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
return reply.code(200).send(updated);
|
||||||
}
|
}
|
||||||
|
|
||||||
return reply.code(500).send({
|
async remove(request: FastifyRequest<{ Params: IdParams }>, reply: FastifyReply) {
|
||||||
error: 'Internal server error',
|
const userId = (request as any).user?.sub as string;
|
||||||
message: 'Failed to update ownership cost'
|
const costId = request.params.id;
|
||||||
|
|
||||||
|
logger.info('Ownership cost delete requested', {
|
||||||
|
operation: 'ownership-costs.delete',
|
||||||
|
userId,
|
||||||
|
costId,
|
||||||
});
|
});
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async delete(request: FastifyRequest<{ Params: OwnershipCostParams }>, reply: FastifyReply) {
|
await this.service.deleteCost(userId, costId);
|
||||||
try {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
const userId = (request as any).user.sub;
|
|
||||||
const { id } = request.params;
|
|
||||||
|
|
||||||
await this.service.delete(id, userId);
|
logger.info('Ownership cost deleted', {
|
||||||
|
operation: 'ownership-costs.delete.success',
|
||||||
|
userId,
|
||||||
|
costId,
|
||||||
|
});
|
||||||
|
|
||||||
return reply.code(204).send();
|
return reply.code(204).send();
|
||||||
} catch (error: unknown) {
|
|
||||||
const err = error as Error;
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
logger.error('Error deleting ownership cost', { error: err, costId: request.params.id, userId: (request as any).user?.sub });
|
|
||||||
|
|
||||||
if (err.message.includes('not found')) {
|
|
||||||
return reply.code(404).send({
|
|
||||||
error: 'Not Found',
|
|
||||||
message: err.message
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (err.message.includes('Unauthorized')) {
|
|
||||||
return reply.code(403).send({
|
|
||||||
error: 'Forbidden',
|
|
||||||
message: err.message
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return reply.code(500).send({
|
|
||||||
error: 'Internal server error',
|
|
||||||
message: 'Failed to delete ownership cost'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getVehicleStats(request: FastifyRequest<{ Params: VehicleParams }>, reply: FastifyReply) {
|
|
||||||
try {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
const userId = (request as any).user.sub;
|
|
||||||
const { vehicleId } = request.params;
|
|
||||||
|
|
||||||
const stats = await this.service.getVehicleCostStats(vehicleId, userId);
|
|
||||||
|
|
||||||
return reply.code(200).send(stats);
|
|
||||||
} catch (error: unknown) {
|
|
||||||
const err = error as Error;
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
||||||
logger.error('Error getting ownership cost stats', { error: err, vehicleId: request.params.vehicleId, userId: (request as any).user?.sub });
|
|
||||||
|
|
||||||
if (err.message.includes('not found')) {
|
|
||||||
return reply.code(404).send({
|
|
||||||
error: 'Not Found',
|
|
||||||
message: err.message
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if (err.message.includes('Unauthorized')) {
|
|
||||||
return reply.code(403).send({
|
|
||||||
error: 'Forbidden',
|
|
||||||
message: err.message
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return reply.code(500).send({
|
|
||||||
error: 'Internal server error',
|
|
||||||
message: 'Failed to get ownership cost stats'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,56 +1,38 @@
|
|||||||
/**
|
/**
|
||||||
* @ai-summary Fastify routes for ownership costs API
|
* @ai-summary Fastify routes for ownership costs API
|
||||||
* @ai-context Route definitions with Fastify plugin pattern and authentication
|
|
||||||
*/
|
*/
|
||||||
|
import { FastifyInstance, FastifyPluginAsync, FastifyPluginOptions } from 'fastify';
|
||||||
import { FastifyInstance, FastifyPluginOptions } from 'fastify';
|
|
||||||
import { FastifyPluginAsync } from 'fastify';
|
|
||||||
import { OwnershipCostsController } from './ownership-costs.controller';
|
import { OwnershipCostsController } from './ownership-costs.controller';
|
||||||
|
|
||||||
export const ownershipCostsRoutes: FastifyPluginAsync = async (
|
export const ownershipCostsRoutes: FastifyPluginAsync = async (
|
||||||
fastify: FastifyInstance,
|
fastify: FastifyInstance,
|
||||||
_opts: FastifyPluginOptions
|
_opts: FastifyPluginOptions
|
||||||
) => {
|
) => {
|
||||||
const controller = new OwnershipCostsController();
|
const ctrl = new OwnershipCostsController();
|
||||||
|
const requireAuth = fastify.authenticate.bind(fastify);
|
||||||
|
|
||||||
// POST /api/ownership-costs - Create new ownership cost
|
fastify.get('/ownership-costs', {
|
||||||
fastify.post('/ownership-costs', {
|
preHandler: [requireAuth],
|
||||||
preHandler: [fastify.authenticate],
|
handler: ctrl.list.bind(ctrl)
|
||||||
handler: controller.create.bind(controller)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// GET /api/ownership-costs/:id - Get specific ownership cost
|
fastify.get<{ Params: any }>('/ownership-costs/:id', {
|
||||||
fastify.get('/ownership-costs/:id', {
|
preHandler: [requireAuth],
|
||||||
preHandler: [fastify.authenticate],
|
handler: ctrl.get.bind(ctrl)
|
||||||
handler: controller.getById.bind(controller)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// PUT /api/ownership-costs/:id - Update ownership cost
|
fastify.post<{ Body: any }>('/ownership-costs', {
|
||||||
fastify.put('/ownership-costs/:id', {
|
preHandler: [requireAuth],
|
||||||
preHandler: [fastify.authenticate],
|
handler: ctrl.create.bind(ctrl)
|
||||||
handler: controller.update.bind(controller)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// DELETE /api/ownership-costs/:id - Delete ownership cost
|
fastify.put<{ Params: any; Body: any }>('/ownership-costs/:id', {
|
||||||
fastify.delete('/ownership-costs/:id', {
|
preHandler: [requireAuth],
|
||||||
preHandler: [fastify.authenticate],
|
handler: ctrl.update.bind(ctrl)
|
||||||
handler: controller.delete.bind(controller)
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// GET /api/ownership-costs/vehicle/:vehicleId - Get costs for a vehicle
|
fastify.delete<{ Params: any }>('/ownership-costs/:id', {
|
||||||
fastify.get('/ownership-costs/vehicle/:vehicleId', {
|
preHandler: [requireAuth],
|
||||||
preHandler: [fastify.authenticate],
|
handler: ctrl.remove.bind(ctrl)
|
||||||
handler: controller.getByVehicle.bind(controller)
|
|
||||||
});
|
|
||||||
|
|
||||||
// GET /api/ownership-costs/vehicle/:vehicleId/stats - Get aggregated cost stats
|
|
||||||
fastify.get('/ownership-costs/vehicle/:vehicleId/stats', {
|
|
||||||
preHandler: [fastify.authenticate],
|
|
||||||
handler: controller.getVehicleStats.bind(controller)
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
// For backward compatibility during migration
|
|
||||||
export function registerOwnershipCostsRoutes() {
|
|
||||||
throw new Error('registerOwnershipCostsRoutes is deprecated - use ownershipCostsRoutes Fastify plugin instead');
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
import { CreateOwnershipCostSchema, UpdateOwnershipCostSchema, OwnershipCostTypeSchema } from '../domain/ownership-costs.types';
|
||||||
|
|
||||||
|
export const ListQuerySchema = z.object({
|
||||||
|
vehicleId: z.string().uuid().optional(),
|
||||||
|
costType: OwnershipCostTypeSchema.optional(),
|
||||||
|
documentId: z.string().uuid().optional(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const IdParamsSchema = z.object({ id: z.string().uuid() });
|
||||||
|
|
||||||
|
export const CreateBodySchema = CreateOwnershipCostSchema;
|
||||||
|
export const UpdateBodySchema = UpdateOwnershipCostSchema;
|
||||||
|
|
||||||
|
export type ListQuery = z.infer<typeof ListQuerySchema>;
|
||||||
|
export type IdParams = z.infer<typeof IdParamsSchema>;
|
||||||
|
export type CreateBody = z.infer<typeof CreateBodySchema>;
|
||||||
|
export type UpdateBody = z.infer<typeof UpdateBodySchema>;
|
||||||
@@ -1,210 +1,160 @@
|
|||||||
/**
|
|
||||||
* @ai-summary Data access layer for ownership costs
|
|
||||||
* @ai-context Handles database operations for vehicle ownership costs
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { Pool } from 'pg';
|
import { Pool } from 'pg';
|
||||||
import {
|
import pool from '../../../core/config/database';
|
||||||
OwnershipCost,
|
import type { OwnershipCost, OwnershipCostType } from '../domain/ownership-costs.types';
|
||||||
CreateOwnershipCostRequest,
|
|
||||||
UpdateOwnershipCostRequest,
|
|
||||||
CostInterval,
|
|
||||||
OwnershipCostType
|
|
||||||
} from '../domain/ownership-costs.types';
|
|
||||||
|
|
||||||
export class OwnershipCostsRepository {
|
export class OwnershipCostsRepository {
|
||||||
constructor(private pool: Pool) {}
|
constructor(private readonly db: Pool = pool) {}
|
||||||
|
|
||||||
async create(data: CreateOwnershipCostRequest & { userId: string }): Promise<OwnershipCost> {
|
// ========================
|
||||||
const query = `
|
// Row Mappers
|
||||||
INSERT INTO ownership_costs (
|
// ========================
|
||||||
user_id, vehicle_id, document_id, cost_type, description,
|
|
||||||
amount, interval, start_date, end_date
|
|
||||||
)
|
|
||||||
VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9)
|
|
||||||
RETURNING *
|
|
||||||
`;
|
|
||||||
|
|
||||||
const values = [
|
private mapRow(row: any): OwnershipCost {
|
||||||
data.userId,
|
return {
|
||||||
data.vehicleId,
|
id: row.id,
|
||||||
data.documentId ?? null,
|
userId: row.user_id,
|
||||||
data.costType,
|
vehicleId: row.vehicle_id,
|
||||||
data.description ?? null,
|
documentId: row.document_id,
|
||||||
data.amount,
|
costType: row.cost_type,
|
||||||
data.interval,
|
amount: row.amount,
|
||||||
data.startDate,
|
description: row.description,
|
||||||
data.endDate ?? null
|
periodStart: row.period_start,
|
||||||
];
|
periodEnd: row.period_end,
|
||||||
|
notes: row.notes,
|
||||||
const result = await this.pool.query(query, values);
|
createdAt: row.created_at,
|
||||||
return this.mapRow(result.rows[0]);
|
updatedAt: row.updated_at
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async findByVehicleId(vehicleId: string, userId: string): Promise<OwnershipCost[]> {
|
// ========================
|
||||||
const query = `
|
// CRUD Operations
|
||||||
SELECT * FROM ownership_costs
|
// ========================
|
||||||
WHERE vehicle_id = $1 AND user_id = $2
|
|
||||||
ORDER BY start_date DESC, created_at DESC
|
|
||||||
`;
|
|
||||||
|
|
||||||
const result = await this.pool.query(query, [vehicleId, userId]);
|
async insert(cost: {
|
||||||
return result.rows.map(row => this.mapRow(row));
|
id: string;
|
||||||
}
|
userId: string;
|
||||||
|
vehicleId: string;
|
||||||
async findByUserId(userId: string): Promise<OwnershipCost[]> {
|
documentId?: string | null;
|
||||||
const query = `
|
costType: OwnershipCostType;
|
||||||
SELECT * FROM ownership_costs
|
amount: number;
|
||||||
WHERE user_id = $1
|
description?: string | null;
|
||||||
ORDER BY start_date DESC, created_at DESC
|
periodStart?: string | null;
|
||||||
`;
|
periodEnd?: string | null;
|
||||||
|
notes?: string | null;
|
||||||
const result = await this.pool.query(query, [userId]);
|
}): Promise<OwnershipCost> {
|
||||||
return result.rows.map(row => this.mapRow(row));
|
const res = await this.db.query(
|
||||||
}
|
`INSERT INTO ownership_costs (
|
||||||
|
id, user_id, vehicle_id, document_id, cost_type, amount, description, period_start, period_end, notes
|
||||||
async findById(id: string): Promise<OwnershipCost | null> {
|
) VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10)
|
||||||
const query = 'SELECT * FROM ownership_costs WHERE id = $1';
|
RETURNING *`,
|
||||||
const result = await this.pool.query(query, [id]);
|
[
|
||||||
|
cost.id,
|
||||||
if (result.rows.length === 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.mapRow(result.rows[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
async findByDocumentId(documentId: string): Promise<OwnershipCost[]> {
|
|
||||||
const query = `
|
|
||||||
SELECT * FROM ownership_costs
|
|
||||||
WHERE document_id = $1
|
|
||||||
ORDER BY start_date DESC
|
|
||||||
`;
|
|
||||||
|
|
||||||
const result = await this.pool.query(query, [documentId]);
|
|
||||||
return result.rows.map(row => this.mapRow(row));
|
|
||||||
}
|
|
||||||
|
|
||||||
async update(id: string, data: UpdateOwnershipCostRequest): Promise<OwnershipCost | null> {
|
|
||||||
const fields = [];
|
|
||||||
const values = [];
|
|
||||||
let paramCount = 1;
|
|
||||||
|
|
||||||
// Build dynamic update query
|
|
||||||
if (data.documentId !== undefined) {
|
|
||||||
fields.push(`document_id = $${paramCount++}`);
|
|
||||||
values.push(data.documentId);
|
|
||||||
}
|
|
||||||
if (data.costType !== undefined) {
|
|
||||||
fields.push(`cost_type = $${paramCount++}`);
|
|
||||||
values.push(data.costType);
|
|
||||||
}
|
|
||||||
if (data.description !== undefined) {
|
|
||||||
fields.push(`description = $${paramCount++}`);
|
|
||||||
values.push(data.description);
|
|
||||||
}
|
|
||||||
if (data.amount !== undefined) {
|
|
||||||
fields.push(`amount = $${paramCount++}`);
|
|
||||||
values.push(data.amount);
|
|
||||||
}
|
|
||||||
if (data.interval !== undefined) {
|
|
||||||
fields.push(`interval = $${paramCount++}`);
|
|
||||||
values.push(data.interval);
|
|
||||||
}
|
|
||||||
if (data.startDate !== undefined) {
|
|
||||||
fields.push(`start_date = $${paramCount++}`);
|
|
||||||
values.push(data.startDate);
|
|
||||||
}
|
|
||||||
if (data.endDate !== undefined) {
|
|
||||||
fields.push(`end_date = $${paramCount++}`);
|
|
||||||
values.push(data.endDate);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fields.length === 0) {
|
|
||||||
return this.findById(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
values.push(id);
|
|
||||||
const query = `
|
|
||||||
UPDATE ownership_costs
|
|
||||||
SET ${fields.join(', ')}
|
|
||||||
WHERE id = $${paramCount}
|
|
||||||
RETURNING *
|
|
||||||
`;
|
|
||||||
|
|
||||||
const result = await this.pool.query(query, values);
|
|
||||||
|
|
||||||
if (result.rows.length === 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.mapRow(result.rows[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
async delete(id: string): Promise<boolean> {
|
|
||||||
const query = 'DELETE FROM ownership_costs WHERE id = $1';
|
|
||||||
const result = await this.pool.query(query, [id]);
|
|
||||||
return (result.rowCount ?? 0) > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
async batchInsert(
|
|
||||||
costs: Array<CreateOwnershipCostRequest & { userId: string }>,
|
|
||||||
client?: Pool
|
|
||||||
): Promise<OwnershipCost[]> {
|
|
||||||
if (costs.length === 0) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
const queryClient = client || this.pool;
|
|
||||||
const placeholders: string[] = [];
|
|
||||||
const values: unknown[] = [];
|
|
||||||
let paramCount = 1;
|
|
||||||
|
|
||||||
costs.forEach((cost) => {
|
|
||||||
const costParams = [
|
|
||||||
cost.userId,
|
cost.userId,
|
||||||
cost.vehicleId,
|
cost.vehicleId,
|
||||||
cost.documentId ?? null,
|
cost.documentId ?? null,
|
||||||
cost.costType,
|
cost.costType,
|
||||||
cost.description ?? null,
|
|
||||||
cost.amount,
|
cost.amount,
|
||||||
cost.interval,
|
cost.description ?? null,
|
||||||
cost.startDate,
|
cost.periodStart ?? null,
|
||||||
cost.endDate ?? null
|
cost.periodEnd ?? null,
|
||||||
];
|
cost.notes ?? null,
|
||||||
|
]
|
||||||
const placeholder = `($${paramCount++}, $${paramCount++}, $${paramCount++}, $${paramCount++}, $${paramCount++}, $${paramCount++}, $${paramCount++}, $${paramCount++}, $${paramCount++})`;
|
);
|
||||||
placeholders.push(placeholder);
|
return this.mapRow(res.rows[0]);
|
||||||
values.push(...costParams);
|
|
||||||
});
|
|
||||||
|
|
||||||
const query = `
|
|
||||||
INSERT INTO ownership_costs (
|
|
||||||
user_id, vehicle_id, document_id, cost_type, description,
|
|
||||||
amount, interval, start_date, end_date
|
|
||||||
)
|
|
||||||
VALUES ${placeholders.join(', ')}
|
|
||||||
RETURNING *
|
|
||||||
`;
|
|
||||||
|
|
||||||
const result = await queryClient.query(query, values);
|
|
||||||
return result.rows.map((row: Record<string, unknown>) => this.mapRow(row));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private mapRow(row: Record<string, unknown>): OwnershipCost {
|
async findById(id: string, userId: string): Promise<OwnershipCost | null> {
|
||||||
return {
|
const res = await this.db.query(
|
||||||
id: row.id as string,
|
`SELECT * FROM ownership_costs WHERE id = $1 AND user_id = $2`,
|
||||||
userId: row.user_id as string,
|
[id, userId]
|
||||||
vehicleId: row.vehicle_id as string,
|
);
|
||||||
documentId: row.document_id as string | undefined,
|
return res.rows[0] ? this.mapRow(res.rows[0]) : null;
|
||||||
costType: row.cost_type as OwnershipCostType,
|
}
|
||||||
description: row.description as string | undefined,
|
|
||||||
amount: Number(row.amount),
|
async findByUserId(
|
||||||
interval: row.interval as CostInterval,
|
userId: string,
|
||||||
startDate: row.start_date as Date,
|
filters?: { vehicleId?: string; costType?: OwnershipCostType; documentId?: string }
|
||||||
endDate: row.end_date as Date | undefined,
|
): Promise<OwnershipCost[]> {
|
||||||
createdAt: row.created_at as Date,
|
const conds: string[] = ['user_id = $1'];
|
||||||
updatedAt: row.updated_at as Date,
|
const params: any[] = [userId];
|
||||||
};
|
let i = 2;
|
||||||
|
|
||||||
|
if (filters?.vehicleId) {
|
||||||
|
conds.push(`vehicle_id = $${i++}`);
|
||||||
|
params.push(filters.vehicleId);
|
||||||
|
}
|
||||||
|
if (filters?.costType) {
|
||||||
|
conds.push(`cost_type = $${i++}`);
|
||||||
|
params.push(filters.costType);
|
||||||
|
}
|
||||||
|
if (filters?.documentId) {
|
||||||
|
conds.push(`document_id = $${i++}`);
|
||||||
|
params.push(filters.documentId);
|
||||||
|
}
|
||||||
|
|
||||||
|
const sql = `SELECT * FROM ownership_costs WHERE ${conds.join(' AND ')} ORDER BY period_start DESC, created_at DESC`;
|
||||||
|
const res = await this.db.query(sql, params);
|
||||||
|
return res.rows.map(row => this.mapRow(row));
|
||||||
|
}
|
||||||
|
|
||||||
|
async findByVehicleId(vehicleId: string, userId: string): Promise<OwnershipCost[]> {
|
||||||
|
const res = await this.db.query(
|
||||||
|
`SELECT * FROM ownership_costs WHERE vehicle_id = $1 AND user_id = $2 ORDER BY period_start DESC, created_at DESC`,
|
||||||
|
[vehicleId, userId]
|
||||||
|
);
|
||||||
|
return res.rows.map(row => this.mapRow(row));
|
||||||
|
}
|
||||||
|
|
||||||
|
async update(
|
||||||
|
id: string,
|
||||||
|
userId: string,
|
||||||
|
patch: Partial<Pick<OwnershipCost, 'documentId' | 'costType' | 'amount' | 'description' | 'periodStart' | 'periodEnd' | 'notes'>>
|
||||||
|
): Promise<OwnershipCost | null> {
|
||||||
|
const fields: string[] = [];
|
||||||
|
const params: any[] = [];
|
||||||
|
let i = 1;
|
||||||
|
|
||||||
|
if (patch.documentId !== undefined) {
|
||||||
|
fields.push(`document_id = $${i++}`);
|
||||||
|
params.push(patch.documentId);
|
||||||
|
}
|
||||||
|
if (patch.costType !== undefined) {
|
||||||
|
fields.push(`cost_type = $${i++}`);
|
||||||
|
params.push(patch.costType);
|
||||||
|
}
|
||||||
|
if (patch.amount !== undefined) {
|
||||||
|
fields.push(`amount = $${i++}`);
|
||||||
|
params.push(patch.amount);
|
||||||
|
}
|
||||||
|
if (patch.description !== undefined) {
|
||||||
|
fields.push(`description = $${i++}`);
|
||||||
|
params.push(patch.description);
|
||||||
|
}
|
||||||
|
if (patch.periodStart !== undefined) {
|
||||||
|
fields.push(`period_start = $${i++}`);
|
||||||
|
params.push(patch.periodStart);
|
||||||
|
}
|
||||||
|
if (patch.periodEnd !== undefined) {
|
||||||
|
fields.push(`period_end = $${i++}`);
|
||||||
|
params.push(patch.periodEnd);
|
||||||
|
}
|
||||||
|
if (patch.notes !== undefined) {
|
||||||
|
fields.push(`notes = $${i++}`);
|
||||||
|
params.push(patch.notes);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fields.length) return this.findById(id, userId);
|
||||||
|
|
||||||
|
params.push(id, userId);
|
||||||
|
const sql = `UPDATE ownership_costs SET ${fields.join(', ')} WHERE id = $${i++} AND user_id = $${i++} RETURNING *`;
|
||||||
|
const res = await this.db.query(sql, params);
|
||||||
|
return res.rows[0] ? this.mapRow(res.rows[0]) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
async delete(id: string, userId: string): Promise<void> {
|
||||||
|
await this.db.query(
|
||||||
|
`DELETE FROM ownership_costs WHERE id = $1 AND user_id = $2`,
|
||||||
|
[id, userId]
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,199 +1,141 @@
|
|||||||
/**
|
import { randomUUID } from 'crypto';
|
||||||
* @ai-summary Business logic for ownership costs feature
|
import type { Pool } from 'pg';
|
||||||
* @ai-context Handles ownership cost operations and TCO aggregation
|
import type {
|
||||||
*/
|
|
||||||
|
|
||||||
import { Pool } from 'pg';
|
|
||||||
import { OwnershipCostsRepository } from '../data/ownership-costs.repository';
|
|
||||||
import {
|
|
||||||
OwnershipCost,
|
|
||||||
CreateOwnershipCostRequest,
|
CreateOwnershipCostRequest,
|
||||||
UpdateOwnershipCostRequest,
|
UpdateOwnershipCostRequest,
|
||||||
|
OwnershipCost,
|
||||||
OwnershipCostResponse,
|
OwnershipCostResponse,
|
||||||
OwnershipCostStats,
|
OwnershipCostType,
|
||||||
PAYMENTS_PER_YEAR,
|
OwnershipCostStats
|
||||||
OwnershipCostType
|
|
||||||
} from './ownership-costs.types';
|
} from './ownership-costs.types';
|
||||||
import { logger } from '../../../core/logging/logger';
|
import { OwnershipCostsRepository } from '../data/ownership-costs.repository';
|
||||||
import { VehiclesRepository } from '../../vehicles/data/vehicles.repository';
|
import pool from '../../../core/config/database';
|
||||||
|
|
||||||
export class OwnershipCostsService {
|
export class OwnershipCostsService {
|
||||||
private repository: OwnershipCostsRepository;
|
private readonly repo: OwnershipCostsRepository;
|
||||||
private vehiclesRepository: VehiclesRepository;
|
private readonly db: Pool;
|
||||||
|
|
||||||
constructor(pool: Pool) {
|
constructor(dbPool?: Pool) {
|
||||||
this.repository = new OwnershipCostsRepository(pool);
|
this.db = dbPool || pool;
|
||||||
this.vehiclesRepository = new VehiclesRepository(pool);
|
this.repo = new OwnershipCostsRepository(this.db);
|
||||||
}
|
}
|
||||||
|
|
||||||
async create(data: CreateOwnershipCostRequest, userId: string): Promise<OwnershipCostResponse> {
|
async createCost(userId: string, body: CreateOwnershipCostRequest): Promise<OwnershipCost> {
|
||||||
logger.info('Creating ownership cost', { userId, vehicleId: data.vehicleId, costType: data.costType });
|
await this.assertVehicleOwnership(userId, body.vehicleId);
|
||||||
|
|
||||||
// Verify vehicle ownership
|
const id = randomUUID();
|
||||||
const vehicle = await this.vehiclesRepository.findById(data.vehicleId);
|
const cost = await this.repo.insert({
|
||||||
if (!vehicle) {
|
id,
|
||||||
throw new Error('Vehicle not found');
|
userId,
|
||||||
}
|
vehicleId: body.vehicleId,
|
||||||
if (vehicle.userId !== userId) {
|
documentId: body.documentId,
|
||||||
throw new Error('Unauthorized');
|
costType: body.costType,
|
||||||
|
amount: body.amount,
|
||||||
|
description: body.description,
|
||||||
|
periodStart: body.periodStart,
|
||||||
|
periodEnd: body.periodEnd,
|
||||||
|
notes: body.notes,
|
||||||
|
});
|
||||||
|
|
||||||
|
return cost;
|
||||||
}
|
}
|
||||||
|
|
||||||
const cost = await this.repository.create({ ...data, userId });
|
async getCost(userId: string, id: string): Promise<OwnershipCostResponse | null> {
|
||||||
|
const cost = await this.repo.findById(id, userId);
|
||||||
|
if (!cost) return null;
|
||||||
return this.toResponse(cost);
|
return this.toResponse(cost);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getByVehicleId(vehicleId: string, userId: string): Promise<OwnershipCostResponse[]> {
|
async getCosts(userId: string, filters?: { vehicleId?: string; costType?: OwnershipCostType; documentId?: string }): Promise<OwnershipCostResponse[]> {
|
||||||
// Verify vehicle ownership
|
const costs = await this.repo.findByUserId(userId, filters);
|
||||||
const vehicle = await this.vehiclesRepository.findById(vehicleId);
|
return costs.map(c => this.toResponse(c));
|
||||||
if (!vehicle) {
|
|
||||||
throw new Error('Vehicle not found');
|
|
||||||
}
|
|
||||||
if (vehicle.userId !== userId) {
|
|
||||||
throw new Error('Unauthorized');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const costs = await this.repository.findByVehicleId(vehicleId, userId);
|
async getCostsByVehicle(userId: string, vehicleId: string): Promise<OwnershipCostResponse[]> {
|
||||||
return costs.map(cost => this.toResponse(cost));
|
const costs = await this.repo.findByVehicleId(vehicleId, userId);
|
||||||
|
return costs.map(c => this.toResponse(c));
|
||||||
}
|
}
|
||||||
|
|
||||||
async getById(id: string, userId: string): Promise<OwnershipCostResponse> {
|
async getVehicleOwnershipCosts(vehicleId: string, userId: string): Promise<OwnershipCostStats> {
|
||||||
const cost = await this.repository.findById(id);
|
const costs = await this.repo.findByVehicleId(vehicleId, userId);
|
||||||
|
|
||||||
if (!cost) {
|
let totalCost = 0;
|
||||||
throw new Error('Ownership cost not found');
|
let insuranceCosts = 0;
|
||||||
|
let registrationCosts = 0;
|
||||||
|
let taxCosts = 0;
|
||||||
|
let otherCosts = 0;
|
||||||
|
|
||||||
|
for (const c of costs) {
|
||||||
|
if (c.amount === null || c.amount === undefined) continue;
|
||||||
|
const amount = Number(c.amount);
|
||||||
|
if (isNaN(amount)) {
|
||||||
|
throw new Error(`Invalid amount value for ownership cost ${c.id}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cost.userId !== userId) {
|
totalCost += amount;
|
||||||
throw new Error('Unauthorized');
|
|
||||||
|
// Breakdown by cost type for backward compatibility
|
||||||
|
switch (c.costType) {
|
||||||
|
case 'insurance':
|
||||||
|
insuranceCosts += amount;
|
||||||
|
break;
|
||||||
|
case 'registration':
|
||||||
|
registrationCosts += amount;
|
||||||
|
break;
|
||||||
|
case 'tax':
|
||||||
|
taxCosts += amount;
|
||||||
|
break;
|
||||||
|
case 'inspection':
|
||||||
|
case 'parking':
|
||||||
|
case 'other':
|
||||||
|
otherCosts += amount;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.toResponse(cost);
|
return {
|
||||||
|
totalCost,
|
||||||
|
recordCount: costs.length,
|
||||||
|
insuranceCosts,
|
||||||
|
registrationCosts,
|
||||||
|
taxCosts,
|
||||||
|
otherCosts
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async update(id: string, data: UpdateOwnershipCostRequest, userId: string): Promise<OwnershipCostResponse> {
|
// Alias for backward compatibility with vehicles service
|
||||||
// Verify ownership
|
async getVehicleCostStats(vehicleId: string, userId: string): Promise<OwnershipCostStats> {
|
||||||
const existing = await this.repository.findById(id);
|
return this.getVehicleOwnershipCosts(vehicleId, userId);
|
||||||
if (!existing) {
|
|
||||||
throw new Error('Ownership cost not found');
|
|
||||||
}
|
|
||||||
if (existing.userId !== userId) {
|
|
||||||
throw new Error('Unauthorized');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const updated = await this.repository.update(id, data);
|
async updateCost(userId: string, id: string, patch: UpdateOwnershipCostRequest): Promise<OwnershipCostResponse | null> {
|
||||||
if (!updated) {
|
const existing = await this.repo.findById(id, userId);
|
||||||
throw new Error('Update failed');
|
if (!existing) return null;
|
||||||
}
|
|
||||||
|
|
||||||
|
// Convert nulls to undefined for repository compatibility
|
||||||
|
const cleanPatch = Object.fromEntries(
|
||||||
|
Object.entries(patch).map(([k, v]) => [k, v === null ? undefined : v])
|
||||||
|
) as Partial<Pick<OwnershipCost, 'documentId' | 'costType' | 'amount' | 'description' | 'periodStart' | 'periodEnd' | 'notes'>>;
|
||||||
|
|
||||||
|
const updated = await this.repo.update(id, userId, cleanPatch);
|
||||||
|
if (!updated) return null;
|
||||||
return this.toResponse(updated);
|
return this.toResponse(updated);
|
||||||
}
|
}
|
||||||
|
|
||||||
async delete(id: string, userId: string): Promise<void> {
|
async deleteCost(userId: string, id: string): Promise<void> {
|
||||||
// Verify ownership
|
await this.repo.delete(id, userId);
|
||||||
const existing = await this.repository.findById(id);
|
|
||||||
if (!existing) {
|
|
||||||
throw new Error('Ownership cost not found');
|
|
||||||
}
|
|
||||||
if (existing.userId !== userId) {
|
|
||||||
throw new Error('Unauthorized');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.repository.delete(id);
|
private async assertVehicleOwnership(userId: string, vehicleId: string) {
|
||||||
logger.info('Ownership cost deleted', { id, userId });
|
const res = await this.db.query('SELECT id FROM vehicles WHERE id = $1 AND user_id = $2', [vehicleId, userId]);
|
||||||
|
if (!res.rows[0]) {
|
||||||
|
const err: any = new Error('Vehicle not found or not owned by user');
|
||||||
|
err.statusCode = 403;
|
||||||
|
throw err;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get aggregated cost statistics for a vehicle
|
|
||||||
* Used by TCO calculation in vehicles service
|
|
||||||
*/
|
|
||||||
async getVehicleCostStats(vehicleId: string, userId: string): Promise<OwnershipCostStats> {
|
|
||||||
const costs = await this.repository.findByVehicleId(vehicleId, userId);
|
|
||||||
const now = new Date();
|
|
||||||
|
|
||||||
const stats: OwnershipCostStats = {
|
|
||||||
insuranceCosts: 0,
|
|
||||||
registrationCosts: 0,
|
|
||||||
taxCosts: 0,
|
|
||||||
otherCosts: 0,
|
|
||||||
totalCosts: 0
|
|
||||||
};
|
|
||||||
|
|
||||||
for (const cost of costs) {
|
|
||||||
const startDate = new Date(cost.startDate);
|
|
||||||
const endDate = cost.endDate ? new Date(cost.endDate) : now;
|
|
||||||
|
|
||||||
// Skip future costs
|
|
||||||
if (startDate > now) continue;
|
|
||||||
|
|
||||||
// Calculate effective end date (either specified end, or now for ongoing costs)
|
|
||||||
const effectiveEnd = endDate < now ? endDate : now;
|
|
||||||
const monthsCovered = this.calculateMonthsBetween(startDate, effectiveEnd);
|
|
||||||
const normalizedCost = this.normalizeToTotal(cost.amount, cost.interval, monthsCovered);
|
|
||||||
|
|
||||||
// Type-safe key access
|
|
||||||
const keyMap: Record<OwnershipCostType, keyof Omit<OwnershipCostStats, 'totalCosts'>> = {
|
|
||||||
insurance: 'insuranceCosts',
|
|
||||||
registration: 'registrationCosts',
|
|
||||||
tax: 'taxCosts',
|
|
||||||
other: 'otherCosts'
|
|
||||||
};
|
|
||||||
|
|
||||||
const key = keyMap[cost.costType];
|
|
||||||
if (key) {
|
|
||||||
stats[key] += normalizedCost;
|
|
||||||
} else {
|
|
||||||
logger.warn('Unknown cost type in aggregation', { costType: cost.costType, costId: cost.id });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
stats.totalCosts = stats.insuranceCosts + stats.registrationCosts + stats.taxCosts + stats.otherCosts;
|
|
||||||
return stats;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Calculate months between two dates
|
|
||||||
*/
|
|
||||||
private calculateMonthsBetween(startDate: Date, endDate: Date): number {
|
|
||||||
const yearDiff = endDate.getFullYear() - startDate.getFullYear();
|
|
||||||
const monthDiff = endDate.getMonth() - startDate.getMonth();
|
|
||||||
return Math.max(1, yearDiff * 12 + monthDiff);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Normalize recurring cost to total based on interval and months covered
|
|
||||||
*/
|
|
||||||
private normalizeToTotal(amount: number, interval: string, monthsCovered: number): number {
|
|
||||||
// One-time costs are just the amount
|
|
||||||
if (interval === 'one_time') {
|
|
||||||
return amount;
|
|
||||||
}
|
|
||||||
|
|
||||||
const paymentsPerYear = PAYMENTS_PER_YEAR[interval as keyof typeof PAYMENTS_PER_YEAR];
|
|
||||||
if (!paymentsPerYear) {
|
|
||||||
logger.warn('Invalid cost interval', { interval });
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calculate total payments over the covered period
|
|
||||||
const yearsOwned = monthsCovered / 12;
|
|
||||||
const totalPayments = yearsOwned * paymentsPerYear;
|
|
||||||
return amount * totalPayments;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private toResponse(cost: OwnershipCost): OwnershipCostResponse {
|
private toResponse(cost: OwnershipCost): OwnershipCostResponse {
|
||||||
return {
|
return cost;
|
||||||
id: cost.id,
|
|
||||||
userId: cost.userId,
|
|
||||||
vehicleId: cost.vehicleId,
|
|
||||||
documentId: cost.documentId,
|
|
||||||
costType: cost.costType,
|
|
||||||
description: cost.description,
|
|
||||||
amount: cost.amount,
|
|
||||||
interval: cost.interval,
|
|
||||||
startDate: cost.startDate instanceof Date ? cost.startDate.toISOString().split('T')[0] : cost.startDate as unknown as string,
|
|
||||||
endDate: cost.endDate ? (cost.endDate instanceof Date ? cost.endDate.toISOString().split('T')[0] : cost.endDate as unknown as string) : undefined,
|
|
||||||
createdAt: cost.createdAt instanceof Date ? cost.createdAt.toISOString() : cost.createdAt as unknown as string,
|
|
||||||
updatedAt: cost.updatedAt instanceof Date ? cost.updatedAt.toISOString() : cost.updatedAt as unknown as string,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,108 +1,69 @@
|
|||||||
/**
|
/**
|
||||||
* @ai-summary Type definitions for ownership-costs feature
|
* @ai-summary Type definitions for ownership-costs feature
|
||||||
* @ai-context Tracks vehicle ownership costs (insurance, registration, tax, other)
|
* @ai-context Tracks vehicle ownership costs (insurance, registration, tax, inspection, parking, other)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// Cost types supported by ownership-costs feature
|
import { z } from 'zod';
|
||||||
export type OwnershipCostType = 'insurance' | 'registration' | 'tax' | 'other';
|
|
||||||
|
|
||||||
// Cost interval types (one_time added for things like purchase price)
|
// Ownership cost types
|
||||||
export type CostInterval = 'monthly' | 'semi_annual' | 'annual' | 'one_time';
|
export type OwnershipCostType = 'insurance' | 'registration' | 'tax' | 'inspection' | 'parking' | 'other';
|
||||||
|
|
||||||
// Payments per year for each interval type
|
|
||||||
export const PAYMENTS_PER_YEAR: Record<CostInterval, number> = {
|
|
||||||
monthly: 12,
|
|
||||||
semi_annual: 2,
|
|
||||||
annual: 1,
|
|
||||||
one_time: 0, // Special case: calculated differently
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
|
// Database record type (camelCase for TypeScript)
|
||||||
export interface OwnershipCost {
|
export interface OwnershipCost {
|
||||||
id: string;
|
id: string;
|
||||||
userId: string;
|
userId: string;
|
||||||
vehicleId: string;
|
vehicleId: string;
|
||||||
documentId?: string;
|
documentId?: string;
|
||||||
costType: OwnershipCostType;
|
costType: OwnershipCostType;
|
||||||
description?: string;
|
|
||||||
amount: number;
|
amount: number;
|
||||||
interval: CostInterval;
|
|
||||||
startDate: Date;
|
|
||||||
endDate?: Date;
|
|
||||||
createdAt: Date;
|
|
||||||
updatedAt: Date;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface CreateOwnershipCostRequest {
|
|
||||||
vehicleId: string;
|
|
||||||
documentId?: string;
|
|
||||||
costType: OwnershipCostType;
|
|
||||||
description?: string;
|
description?: string;
|
||||||
amount: number;
|
periodStart?: string;
|
||||||
interval: CostInterval;
|
periodEnd?: string;
|
||||||
startDate: string; // ISO date string
|
notes?: string;
|
||||||
endDate?: string; // ISO date string
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface UpdateOwnershipCostRequest {
|
|
||||||
documentId?: string | null;
|
|
||||||
costType?: OwnershipCostType;
|
|
||||||
description?: string | null;
|
|
||||||
amount?: number;
|
|
||||||
interval?: CostInterval;
|
|
||||||
startDate?: string;
|
|
||||||
endDate?: string | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface OwnershipCostResponse {
|
|
||||||
id: string;
|
|
||||||
userId: string;
|
|
||||||
vehicleId: string;
|
|
||||||
documentId?: string;
|
|
||||||
costType: OwnershipCostType;
|
|
||||||
description?: string;
|
|
||||||
amount: number;
|
|
||||||
interval: CostInterval;
|
|
||||||
startDate: string;
|
|
||||||
endDate?: string;
|
|
||||||
createdAt: string;
|
createdAt: string;
|
||||||
updatedAt: string;
|
updatedAt: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Aggregated cost statistics for TCO calculation
|
// Zod schemas for validation (camelCase for API)
|
||||||
|
export const OwnershipCostTypeSchema = z.enum(['insurance', 'registration', 'tax', 'inspection', 'parking', 'other']);
|
||||||
|
|
||||||
|
export const CreateOwnershipCostSchema = z.object({
|
||||||
|
vehicleId: z.string().uuid(),
|
||||||
|
documentId: z.string().uuid().optional(),
|
||||||
|
costType: OwnershipCostTypeSchema,
|
||||||
|
amount: z.number().positive(),
|
||||||
|
description: z.string().max(200).optional(),
|
||||||
|
periodStart: z.string().optional(),
|
||||||
|
periodEnd: z.string().optional(),
|
||||||
|
notes: z.string().max(10000).optional(),
|
||||||
|
});
|
||||||
|
export type CreateOwnershipCostRequest = z.infer<typeof CreateOwnershipCostSchema>;
|
||||||
|
|
||||||
|
export const UpdateOwnershipCostSchema = z.object({
|
||||||
|
documentId: z.string().uuid().nullable().optional(),
|
||||||
|
costType: OwnershipCostTypeSchema.optional(),
|
||||||
|
amount: z.number().positive().optional(),
|
||||||
|
description: z.string().max(200).nullable().optional(),
|
||||||
|
periodStart: z.string().nullable().optional(),
|
||||||
|
periodEnd: z.string().nullable().optional(),
|
||||||
|
notes: z.string().max(10000).nullable().optional(),
|
||||||
|
});
|
||||||
|
export type UpdateOwnershipCostRequest = z.infer<typeof UpdateOwnershipCostSchema>;
|
||||||
|
|
||||||
|
// Response types
|
||||||
|
export interface OwnershipCostResponse extends OwnershipCost {}
|
||||||
|
|
||||||
|
// TCO aggregation stats
|
||||||
|
// NOTE: Extended for backward compatibility with vehicles service
|
||||||
|
// The spec calls for totalCost and recordCount only, but the vehicles service
|
||||||
|
// expects breakdown by cost type. This will be cleaned up when vehicles service
|
||||||
|
// is refactored to use the new ownership-costs table directly.
|
||||||
export interface OwnershipCostStats {
|
export interface OwnershipCostStats {
|
||||||
insuranceCosts: number;
|
totalCost: number;
|
||||||
registrationCosts: number;
|
recordCount: number;
|
||||||
taxCosts: number;
|
// Backward compatibility fields for vehicles service
|
||||||
otherCosts: number;
|
insuranceCosts?: number;
|
||||||
totalCosts: number;
|
registrationCosts?: number;
|
||||||
}
|
taxCosts?: number;
|
||||||
|
otherCosts?: number;
|
||||||
// Fastify-specific types for HTTP handling
|
|
||||||
export interface CreateOwnershipCostBody {
|
|
||||||
vehicleId: string;
|
|
||||||
documentId?: string;
|
|
||||||
costType: OwnershipCostType;
|
|
||||||
description?: string;
|
|
||||||
amount: number;
|
|
||||||
interval: CostInterval;
|
|
||||||
startDate: string;
|
|
||||||
endDate?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface UpdateOwnershipCostBody {
|
|
||||||
documentId?: string | null;
|
|
||||||
costType?: OwnershipCostType;
|
|
||||||
description?: string | null;
|
|
||||||
amount?: number;
|
|
||||||
interval?: CostInterval;
|
|
||||||
startDate?: string;
|
|
||||||
endDate?: string | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface OwnershipCostParams {
|
|
||||||
id: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface VehicleParams {
|
|
||||||
vehicleId: string;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,11 +12,8 @@ export type {
|
|||||||
UpdateOwnershipCostRequest,
|
UpdateOwnershipCostRequest,
|
||||||
OwnershipCostResponse,
|
OwnershipCostResponse,
|
||||||
OwnershipCostStats,
|
OwnershipCostStats,
|
||||||
OwnershipCostType,
|
OwnershipCostType
|
||||||
CostInterval
|
|
||||||
} from './domain/ownership-costs.types';
|
} from './domain/ownership-costs.types';
|
||||||
|
|
||||||
export { PAYMENTS_PER_YEAR } from './domain/ownership-costs.types';
|
|
||||||
|
|
||||||
// Internal: Register routes with Fastify app
|
// Internal: Register routes with Fastify app
|
||||||
export { ownershipCostsRoutes, registerOwnershipCostsRoutes } from './api/ownership-costs.routes';
|
export { ownershipCostsRoutes } from './api/ownership-costs.routes';
|
||||||
|
|||||||
@@ -1,60 +1,27 @@
|
|||||||
-- Migration: Create ownership_costs table
|
-- Migration: Create ownership_costs table
|
||||||
-- Issue: #15
|
-- Issue: #29
|
||||||
-- Description: Store vehicle ownership costs (insurance, registration, tax, other)
|
-- Description: Create ownership_costs table for tracking insurance, registration, tax, inspection, parking, and other costs
|
||||||
-- with explicit date ranges and optional document association
|
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS ownership_costs (
|
CREATE TABLE ownership_costs (
|
||||||
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
id UUID PRIMARY KEY DEFAULT uuid_generate_v4(),
|
||||||
user_id VARCHAR(255) NOT NULL,
|
user_id VARCHAR(255) NOT NULL,
|
||||||
vehicle_id UUID NOT NULL,
|
vehicle_id UUID NOT NULL REFERENCES vehicles(id) ON DELETE CASCADE,
|
||||||
document_id UUID,
|
document_id UUID NULL REFERENCES documents(id) ON DELETE CASCADE,
|
||||||
cost_type VARCHAR(50) NOT NULL,
|
cost_type VARCHAR(32) NOT NULL CHECK (cost_type IN ('insurance', 'registration', 'tax', 'inspection', 'parking', 'other')),
|
||||||
description TEXT,
|
amount DECIMAL(10, 2) NOT NULL CHECK (amount > 0),
|
||||||
amount DECIMAL(12, 2) NOT NULL,
|
description VARCHAR(200),
|
||||||
interval VARCHAR(20) NOT NULL,
|
period_start DATE,
|
||||||
start_date DATE NOT NULL,
|
period_end DATE,
|
||||||
end_date DATE,
|
notes TEXT,
|
||||||
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
||||||
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP,
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT CURRENT_TIMESTAMP
|
||||||
|
|
||||||
-- Foreign key to vehicles (cascade delete)
|
|
||||||
CONSTRAINT fk_ownership_costs_vehicle
|
|
||||||
FOREIGN KEY (vehicle_id)
|
|
||||||
REFERENCES vehicles(id)
|
|
||||||
ON DELETE CASCADE,
|
|
||||||
|
|
||||||
-- Foreign key to documents (set null on delete)
|
|
||||||
CONSTRAINT fk_ownership_costs_document
|
|
||||||
FOREIGN KEY (document_id)
|
|
||||||
REFERENCES documents(id)
|
|
||||||
ON DELETE SET NULL,
|
|
||||||
|
|
||||||
-- Enforce valid cost types
|
|
||||||
CONSTRAINT chk_ownership_costs_type
|
|
||||||
CHECK (cost_type IN ('insurance', 'registration', 'tax', 'other')),
|
|
||||||
|
|
||||||
-- Enforce valid intervals
|
|
||||||
CONSTRAINT chk_ownership_costs_interval
|
|
||||||
CHECK (interval IN ('monthly', 'semi_annual', 'annual', 'one_time')),
|
|
||||||
|
|
||||||
-- Enforce non-negative amounts
|
|
||||||
CONSTRAINT chk_ownership_costs_amount_non_negative
|
|
||||||
CHECK (amount >= 0),
|
|
||||||
|
|
||||||
-- Enforce end_date >= start_date when end_date is provided
|
|
||||||
CONSTRAINT chk_ownership_costs_date_range
|
|
||||||
CHECK (end_date IS NULL OR end_date >= start_date)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
-- Create indexes for common queries
|
CREATE INDEX idx_ownership_costs_user_id ON ownership_costs(user_id);
|
||||||
CREATE INDEX IF NOT EXISTS idx_ownership_costs_user_id ON ownership_costs(user_id);
|
CREATE INDEX idx_ownership_costs_vehicle_id ON ownership_costs(vehicle_id);
|
||||||
CREATE INDEX IF NOT EXISTS idx_ownership_costs_vehicle_id ON ownership_costs(vehicle_id);
|
CREATE INDEX idx_ownership_costs_document_id ON ownership_costs(document_id);
|
||||||
CREATE INDEX IF NOT EXISTS idx_ownership_costs_cost_type ON ownership_costs(cost_type);
|
CREATE INDEX idx_ownership_costs_cost_type ON ownership_costs(cost_type);
|
||||||
CREATE INDEX IF NOT EXISTS idx_ownership_costs_start_date ON ownership_costs(start_date DESC);
|
|
||||||
CREATE INDEX IF NOT EXISTS idx_ownership_costs_document_id ON ownership_costs(document_id) WHERE document_id IS NOT NULL;
|
|
||||||
|
|
||||||
-- Add updated_at trigger
|
|
||||||
DROP TRIGGER IF EXISTS update_ownership_costs_updated_at ON ownership_costs;
|
|
||||||
CREATE TRIGGER update_ownership_costs_updated_at
|
CREATE TRIGGER update_ownership_costs_updated_at
|
||||||
BEFORE UPDATE ON ownership_costs
|
BEFORE UPDATE ON ownership_costs
|
||||||
FOR EACH ROW
|
FOR EACH ROW
|
||||||
|
|||||||
@@ -1,73 +0,0 @@
|
|||||||
-- Migration: Migrate existing vehicle TCO data to ownership_costs table
|
|
||||||
-- Issue: #15
|
|
||||||
-- Description: Copy insurance and registration costs from vehicles table to ownership_costs
|
|
||||||
|
|
||||||
-- Insert insurance costs from vehicles that have insurance data
|
|
||||||
INSERT INTO ownership_costs (
|
|
||||||
user_id,
|
|
||||||
vehicle_id,
|
|
||||||
document_id,
|
|
||||||
cost_type,
|
|
||||||
description,
|
|
||||||
amount,
|
|
||||||
interval,
|
|
||||||
start_date,
|
|
||||||
end_date
|
|
||||||
)
|
|
||||||
SELECT
|
|
||||||
v.user_id,
|
|
||||||
v.id AS vehicle_id,
|
|
||||||
NULL AS document_id,
|
|
||||||
'insurance' AS cost_type,
|
|
||||||
'Migrated from vehicle record' AS description,
|
|
||||||
v.insurance_cost AS amount,
|
|
||||||
v.insurance_interval AS interval,
|
|
||||||
COALESCE(v.purchase_date, v.created_at::date) AS start_date,
|
|
||||||
NULL AS end_date
|
|
||||||
FROM vehicles v
|
|
||||||
WHERE v.insurance_cost IS NOT NULL
|
|
||||||
AND v.insurance_cost > 0
|
|
||||||
AND v.insurance_interval IS NOT NULL
|
|
||||||
AND v.is_active = true
|
|
||||||
AND NOT EXISTS (
|
|
||||||
-- Only migrate if no insurance cost already exists for this vehicle
|
|
||||||
SELECT 1 FROM ownership_costs oc
|
|
||||||
WHERE oc.vehicle_id = v.id
|
|
||||||
AND oc.cost_type = 'insurance'
|
|
||||||
AND oc.description = 'Migrated from vehicle record'
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Insert registration costs from vehicles that have registration data
|
|
||||||
INSERT INTO ownership_costs (
|
|
||||||
user_id,
|
|
||||||
vehicle_id,
|
|
||||||
document_id,
|
|
||||||
cost_type,
|
|
||||||
description,
|
|
||||||
amount,
|
|
||||||
interval,
|
|
||||||
start_date,
|
|
||||||
end_date
|
|
||||||
)
|
|
||||||
SELECT
|
|
||||||
v.user_id,
|
|
||||||
v.id AS vehicle_id,
|
|
||||||
NULL AS document_id,
|
|
||||||
'registration' AS cost_type,
|
|
||||||
'Migrated from vehicle record' AS description,
|
|
||||||
v.registration_cost AS amount,
|
|
||||||
v.registration_interval AS interval,
|
|
||||||
COALESCE(v.purchase_date, v.created_at::date) AS start_date,
|
|
||||||
NULL AS end_date
|
|
||||||
FROM vehicles v
|
|
||||||
WHERE v.registration_cost IS NOT NULL
|
|
||||||
AND v.registration_cost > 0
|
|
||||||
AND v.registration_interval IS NOT NULL
|
|
||||||
AND v.is_active = true
|
|
||||||
AND NOT EXISTS (
|
|
||||||
-- Only migrate if no registration cost already exists for this vehicle
|
|
||||||
SELECT 1 FROM ownership_costs oc
|
|
||||||
WHERE oc.vehicle_id = v.id
|
|
||||||
AND oc.cost_type = 'registration'
|
|
||||||
AND oc.description = 'Migrated from vehicle record'
|
|
||||||
);
|
|
||||||
Reference in New Issue
Block a user