Some checks failed
Deploy / Deploy to Production (push) Has been skipped
Deploy / Smoke Test Production (push) Has been skipped
Security Scanning / Security Gate (push) Failing after 0s
Deploy / Rollback Production (push) Has been skipped
CI / Lint → Typecheck → Test → Build (22) (push) Failing after 9s
CI / E2E Tests (push) Has been skipped
CodeQL Analysis / CodeQL (javascript-typescript) (push) Failing after 53s
Deploy / Build API Image (push) Failing after 13s
Deploy / Build Web Image (push) Failing after 9s
Deploy / Build AI Services Image (push) Failing after 11s
E2E Tests / Playwright E2E (push) Failing after 10s
Security Scanning / Dependency Audit (pnpm) (push) Failing after 4s
Security Scanning / Trivy Scan — API Image (push) Failing after 50s
Security Scanning / Trivy Scan — Web Image (push) Failing after 41s
Security Scanning / Trivy Scan — AI Services Image (push) Failing after 31s
Security Scanning / Trivy Filesystem Scan (push) Failing after 23s
Deploy / Deploy to Staging (push) Has been skipped
Deploy / Smoke Test Staging (push) Has been skipped
Deploy / Rollback Staging (push) Has been skipped
Phase 1 — live POI + neighborhood score on project detail
- du-an-detail-client fetches `/analytics/pois/nearby` + `/analytics/neighborhoods/:district/score`
- Falls back to admin-entered `project.pois` / `neighborhoodScores` when endpoint returns nothing
- Adds total-score badge next to the radar chart (matches listings)
Phase 2 — project personas derivation (`lib/project-personas.ts`)
- Derives 8 personas from project-specific signals: property-type mix, amenity keywords,
developer reputation, completion timing, status, live score + POIs
- Merges admin-authored `suitableFor` chips (badged "Chủ đầu tư chọn") with derived chips
- `composeWhyThisProject()` narrative used as fallback when admin hasn't authored one;
badged "Tự động tổng hợp" so users know it's derived
Phase 3 — AI advisor for projects
- Extract shared Anthropic transport + JSON parsers to
`analytics/application/queries/_shared/ai-json-client.ts` (dual auth: x-api-key +
Bearer for proxy gateways)
- Refactor `GetListingAiAdviceHandler` to use the shared client
- New `GetProjectAiAdviceHandler` (CQRS) pulls project detail + optional POIs + score,
builds project-flavored prompt, returns `{ advice: { summary, pros, cons, suitableFor } }`.
No valuation block — project price is a range, not a single unit.
- `POST /analytics/projects/:id/ai-advice` endpoint (JWT-guarded)
- `ErrorCode.PROJECT_NOT_FOUND` added
- Frontend: `ProjectAiAdviceCard` mirrors listings card minus valuation, with loading /
not-configured (503) / error states; dedupes AI-suggested personas against existing chips
Phase 4 — Mapbox LocationPicker in project create form
- New project page now renders `<LocationPicker>` with Vietnam-scoped geocoder; click /
drag / search autofills lat+lng and (when empty) address/ward/district/city
- Edit page notes location immutability — backend `UpdateProjectCommand` does not yet
accept lat/lng/address mutations (follow-up needed to enable editing coords)
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
170 lines
4.0 KiB
TypeScript
170 lines
4.0 KiB
TypeScript
import { apiClient } from './api-client';
|
|
|
|
export interface MarketReportDistrict {
|
|
district: string;
|
|
city: string;
|
|
propertyType: string;
|
|
period: string;
|
|
medianPrice: string;
|
|
avgPriceM2: number;
|
|
totalListings: number;
|
|
daysOnMarket: number;
|
|
inventoryLevel: number;
|
|
absorptionRate: number | null;
|
|
yoyChange: number | null;
|
|
}
|
|
|
|
export interface MarketReportResponse {
|
|
city: string;
|
|
period: string;
|
|
districts: MarketReportDistrict[];
|
|
}
|
|
|
|
export interface HeatmapDataPoint {
|
|
district: string;
|
|
city: string;
|
|
avgPriceM2: number;
|
|
totalListings: number;
|
|
medianPrice: string;
|
|
}
|
|
|
|
export interface HeatmapResponse {
|
|
city: string;
|
|
period: string;
|
|
dataPoints: HeatmapDataPoint[];
|
|
}
|
|
|
|
export interface PriceTrendPoint {
|
|
period: string;
|
|
medianPrice: string;
|
|
avgPriceM2: number;
|
|
totalListings: number;
|
|
}
|
|
|
|
export interface PriceTrendResponse {
|
|
district: string;
|
|
city: string;
|
|
propertyType: string;
|
|
trend: PriceTrendPoint[];
|
|
}
|
|
|
|
export interface DistrictStats {
|
|
district: string;
|
|
city: string;
|
|
propertyType: string;
|
|
medianPrice: string;
|
|
avgPriceM2: number;
|
|
totalListings: number;
|
|
daysOnMarket: number;
|
|
inventoryLevel: number;
|
|
absorptionRate: number | null;
|
|
yoyChange: number | null;
|
|
}
|
|
|
|
export interface DistrictStatsResponse {
|
|
city: string;
|
|
period: string;
|
|
districts: DistrictStats[];
|
|
}
|
|
|
|
export type NearbyPOICategory =
|
|
| 'school'
|
|
| 'hospital'
|
|
| 'transit'
|
|
| 'shopping'
|
|
| 'restaurant'
|
|
| 'park';
|
|
|
|
export interface NearbyPOI {
|
|
id: string;
|
|
name: string;
|
|
type: string;
|
|
category: NearbyPOICategory;
|
|
lat: number;
|
|
lng: number;
|
|
distance: number;
|
|
address: string | null;
|
|
}
|
|
|
|
export interface NearbyPOIsResponse {
|
|
pois: NearbyPOI[];
|
|
center: { lat: number; lng: number };
|
|
}
|
|
|
|
export type AiConfidence = 'low' | 'medium' | 'high';
|
|
|
|
export interface ListingAiValuation {
|
|
estimateVND: number;
|
|
lowVND: number;
|
|
highVND: number;
|
|
confidence: AiConfidence;
|
|
rationale: string;
|
|
}
|
|
|
|
export interface ListingAiAdviceBody {
|
|
summary: string;
|
|
pros: string[];
|
|
cons: string[];
|
|
suitableFor: string[];
|
|
}
|
|
|
|
export interface ListingAiAdvice {
|
|
valuation: ListingAiValuation;
|
|
advice: ListingAiAdviceBody;
|
|
model: string;
|
|
cacheHit: boolean;
|
|
cacheUsage?: {
|
|
input: number;
|
|
cacheCreation: number;
|
|
cacheRead: number;
|
|
output: number;
|
|
};
|
|
}
|
|
|
|
/** Project AI advice — same advice block as listings but no valuation. */
|
|
export interface ProjectAiAdvice {
|
|
advice: ListingAiAdviceBody;
|
|
model: string;
|
|
cacheHit: boolean;
|
|
cacheUsage?: {
|
|
input: number;
|
|
cacheCreation: number;
|
|
cacheRead: number;
|
|
output: number;
|
|
};
|
|
}
|
|
|
|
export const analyticsApi = {
|
|
getMarketReport: (city: string, period: string, propertyType?: string) => {
|
|
const params = new URLSearchParams({ city, period });
|
|
if (propertyType) params.set('propertyType', propertyType);
|
|
return apiClient.get<MarketReportResponse>(`/analytics/market-report?${params}`);
|
|
},
|
|
|
|
getHeatmap: (city: string, period: string) => {
|
|
const params = new URLSearchParams({ city, period });
|
|
return apiClient.get<HeatmapResponse>(`/analytics/heatmap?${params}`);
|
|
},
|
|
|
|
getPriceTrend: (district: string, city: string, propertyType: string, periods: string[]) => {
|
|
const params = new URLSearchParams({ district, city, propertyType, periods: periods.join(',') });
|
|
return apiClient.get<PriceTrendResponse>(`/analytics/price-trend?${params}`);
|
|
},
|
|
|
|
getDistrictStats: (city: string, period: string) => {
|
|
const params = new URLSearchParams({ city, period });
|
|
return apiClient.get<DistrictStatsResponse>(`/analytics/district-stats?${params}`);
|
|
},
|
|
|
|
getNearbyPOIs: (lat: number, lng: number, radius = 2000, limit = 30) =>
|
|
apiClient.get<NearbyPOIsResponse>(
|
|
`/analytics/pois/nearby?lat=${lat}&lng=${lng}&radius=${radius}&limit=${limit}`,
|
|
),
|
|
|
|
getListingAiAdvice: (listingId: string) =>
|
|
apiClient.post<ListingAiAdvice>(`/analytics/listings/${listingId}/ai-advice`),
|
|
|
|
getProjectAiAdvice: (projectId: string) =>
|
|
apiClient.post<ProjectAiAdvice>(`/analytics/projects/${projectId}/ai-advice`),
|
|
};
|