Add property assessor assess command
This commit is contained in:
@@ -16,8 +16,9 @@ The skill is intended to:
|
|||||||
|
|
||||||
## Standalone helper usage
|
## Standalone helper usage
|
||||||
|
|
||||||
This skill now ships with a small TypeScript helper package for two tasks:
|
This skill now ships with a small TypeScript helper package for three tasks:
|
||||||
|
|
||||||
|
- assembling an address-first preliminary assessment payload
|
||||||
- locating official public-record jurisdiction from an address
|
- locating official public-record jurisdiction from an address
|
||||||
- rendering a fixed-template PDF report
|
- rendering a fixed-template PDF report
|
||||||
|
|
||||||
@@ -33,6 +34,10 @@ The wrapper script uses the skill-local Node dependencies under `node_modules/`.
|
|||||||
## Commands
|
## Commands
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
scripts/property-assessor assess --address "4141 Whiteley Dr, Corpus Christi, TX 78418"
|
||||||
|
scripts/property-assessor assess --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --recipient-email "buyer@example.com"
|
||||||
|
scripts/property-assessor assess --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --recipient-email "buyer@example.com" --output /tmp/property-assessment.pdf
|
||||||
|
|
||||||
scripts/property-assessor locate-public-records --address "4141 Whiteley Dr, Corpus Christi, TX 78418"
|
scripts/property-assessor locate-public-records --address "4141 Whiteley Dr, Corpus Christi, TX 78418"
|
||||||
scripts/property-assessor locate-public-records --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --parcel-id "14069438"
|
scripts/property-assessor locate-public-records --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --parcel-id "14069438"
|
||||||
scripts/property-assessor locate-public-records --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --listing-geo-id "233290"
|
scripts/property-assessor locate-public-records --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --listing-geo-id "233290"
|
||||||
@@ -45,12 +50,34 @@ scripts/property-assessor render-report --input examples/report-payload.example.
|
|||||||
Default operating sequence:
|
Default operating sequence:
|
||||||
|
|
||||||
1. Normalize the address and property type.
|
1. Normalize the address and property type.
|
||||||
2. Discover accessible listing and public-record sources for the same property.
|
2. Resolve public-record jurisdiction from the address.
|
||||||
3. Build a baseline fact set.
|
3. Discover accessible listing sources for the same property.
|
||||||
4. Review listing photos before making condition claims.
|
4. Build a baseline fact set.
|
||||||
5. Pull same-building or nearby comps.
|
5. Review listing photos before making condition claims.
|
||||||
6. Underwrite carry costs and risk factors.
|
6. Pull same-building or nearby comps.
|
||||||
7. Render the final report as a fixed-template PDF.
|
7. Underwrite carry costs and risk factors.
|
||||||
|
8. Render the final report as a fixed-template PDF.
|
||||||
|
|
||||||
|
### `assess`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
scripts/property-assessor assess --address "<street-address>"
|
||||||
|
scripts/property-assessor assess --address "<street-address>" --recipient-email "<target@example.com>"
|
||||||
|
```
|
||||||
|
|
||||||
|
Current behavior:
|
||||||
|
|
||||||
|
- starts from the address
|
||||||
|
- automatically runs public-record / appraisal-district lookup
|
||||||
|
- returns a structured preliminary report payload
|
||||||
|
- asks for recipient email(s) before PDF generation
|
||||||
|
- renders the fixed-template PDF when recipient email(s) are present
|
||||||
|
|
||||||
|
Important limitation:
|
||||||
|
|
||||||
|
- this first implementation wires the address-first assessment spine
|
||||||
|
- it does not yet auto-run listing extraction, photo review, comps, or carry underwriting inside the helper itself
|
||||||
|
- those richer assessment steps are still governed by the skill workflow and the `web-automation` extractors
|
||||||
|
|
||||||
## Source priority
|
## Source priority
|
||||||
|
|
||||||
@@ -178,6 +205,7 @@ For chat-driven runs, prefer file-based commands.
|
|||||||
|
|
||||||
Good:
|
Good:
|
||||||
|
|
||||||
|
- `scripts/property-assessor assess --address "..."`
|
||||||
- `node check-install.js`
|
- `node check-install.js`
|
||||||
- `node zillow-photos.js "<url>"`
|
- `node zillow-photos.js "<url>"`
|
||||||
- `node har-photos.js "<url>"`
|
- `node har-photos.js "<url>"`
|
||||||
@@ -200,6 +228,7 @@ Template reference:
|
|||||||
Current renderer:
|
Current renderer:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
scripts/property-assessor assess --address "<street-address>" --recipient-email "<target@example.com>"
|
||||||
scripts/property-assessor render-report --input "<report-payload-json>" --output "<output-pdf>"
|
scripts/property-assessor render-report --input "<report-payload-json>" --output "<output-pdf>"
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -271,7 +300,21 @@ npm install
|
|||||||
npm test
|
npm test
|
||||||
```
|
```
|
||||||
|
|
||||||
### 2. Run public-record lookup
|
### 2. Run address-first assess without recipient email
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
|
scripts/property-assessor assess --address "4141 Whiteley Dr, Corpus Christi, TX 78418"
|
||||||
|
```
|
||||||
|
|
||||||
|
Expected shape:
|
||||||
|
|
||||||
|
- `needsRecipientEmails: true`
|
||||||
|
- public-record / CAD jurisdiction included in the returned payload
|
||||||
|
- no PDF generated yet
|
||||||
|
- explicit message telling the operator to ask for target recipient email(s)
|
||||||
|
|
||||||
|
### 3. Run public-record lookup directly
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd ~/.openclaw/workspace/skills/property-assessor
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
@@ -285,7 +328,20 @@ Expected shape:
|
|||||||
- for Texas: Comptroller county directory link present
|
- for Texas: Comptroller county directory link present
|
||||||
- for Texas: appraisal district and tax assessor/collector contacts present
|
- for Texas: appraisal district and tax assessor/collector contacts present
|
||||||
|
|
||||||
### 3. Run PDF render with the sample payload
|
### 4. Run assess with recipient email and render the PDF
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
|
scripts/property-assessor assess --address "4141 Whiteley Dr, Corpus Christi, TX 78418" --recipient-email "buyer@example.com" --output /tmp/property-assessment.pdf
|
||||||
|
```
|
||||||
|
|
||||||
|
Expected result:
|
||||||
|
|
||||||
|
- `needsRecipientEmails: false`
|
||||||
|
- JSON success payload with `outputPath`
|
||||||
|
- a non-empty PDF written to `/tmp/property-assessment.pdf`
|
||||||
|
|
||||||
|
### 5. Run PDF render with the sample payload
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd ~/.openclaw/workspace/skills/property-assessor
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
@@ -297,7 +353,7 @@ Expected result:
|
|||||||
- JSON success payload with `outputPath`
|
- JSON success payload with `outputPath`
|
||||||
- a non-empty PDF written to `/tmp/property-assessment.pdf`
|
- a non-empty PDF written to `/tmp/property-assessment.pdf`
|
||||||
|
|
||||||
### 4. Verify the email gate
|
### 6. Verify the email gate
|
||||||
|
|
||||||
Run the renderer with a payload that omits `recipientEmails`.
|
Run the renderer with a payload that omits `recipientEmails`.
|
||||||
|
|
||||||
@@ -306,7 +362,7 @@ Expected result:
|
|||||||
- non-zero exit
|
- non-zero exit
|
||||||
- explicit message telling the operator to stop and ask for target recipient email(s)
|
- explicit message telling the operator to stop and ask for target recipient email(s)
|
||||||
|
|
||||||
### 5. Verify the end-to-end skill behavior
|
### 7. Verify the end-to-end skill behavior
|
||||||
|
|
||||||
When testing `property-assessor` itself, confirm the assessment:
|
When testing `property-assessor` itself, confirm the assessment:
|
||||||
|
|
||||||
|
|||||||
@@ -56,6 +56,7 @@ Use `web_search` sparingly to discover alternate URLs, then return to `web-autom
|
|||||||
## Helper runtime
|
## Helper runtime
|
||||||
|
|
||||||
`property-assessor` now includes TypeScript helper commands for:
|
`property-assessor` now includes TypeScript helper commands for:
|
||||||
|
- address-first preliminary assessment assembly
|
||||||
- public-record jurisdiction lookup
|
- public-record jurisdiction lookup
|
||||||
- fixed-template PDF rendering
|
- fixed-template PDF rendering
|
||||||
|
|
||||||
@@ -66,6 +67,23 @@ cd ~/.openclaw/workspace/skills/property-assessor
|
|||||||
npm install
|
npm install
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Quick command summary:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
|
npm install
|
||||||
|
scripts/property-assessor assess --address "<street-address>"
|
||||||
|
scripts/property-assessor assess --address "<street-address>" --recipient-email "<target@example.com>"
|
||||||
|
scripts/property-assessor locate-public-records --address "<street-address>"
|
||||||
|
scripts/property-assessor render-report --input "<report-payload-json>" --output "<output-pdf>"
|
||||||
|
```
|
||||||
|
|
||||||
|
`assess` is the address-first entrypoint. It should:
|
||||||
|
- resolve official public-record jurisdiction automatically from the address
|
||||||
|
- build the report payload skeleton
|
||||||
|
- stop and ask for recipient email(s) before rendering the PDF
|
||||||
|
- render the PDF only after recipient email(s) are known
|
||||||
|
|
||||||
## Public-record enrichment
|
## Public-record enrichment
|
||||||
|
|
||||||
Public-record / assessor data should be used when available and linked in the final result.
|
Public-record / assessor data should be used when available and linked in the final result.
|
||||||
@@ -85,6 +103,16 @@ npm install
|
|||||||
scripts/property-assessor locate-public-records --address "<street-address>"
|
scripts/property-assessor locate-public-records --address "<street-address>"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
When you want the helper to assemble the preliminary assessment payload in one step, use:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
|
npm install
|
||||||
|
scripts/property-assessor assess --address "<street-address>"
|
||||||
|
```
|
||||||
|
|
||||||
|
This command should automatically include the public-record jurisdiction result in the returned assessment payload.
|
||||||
|
|
||||||
This command currently:
|
This command currently:
|
||||||
- resolves the address through the official Census geocoder
|
- resolves the address through the official Census geocoder
|
||||||
- returns county/state/FIPS/GEOID context
|
- returns county/state/FIPS/GEOID context
|
||||||
@@ -292,6 +320,7 @@ Use the property-assessor helper CLI:
|
|||||||
```bash
|
```bash
|
||||||
cd ~/.openclaw/workspace/skills/property-assessor
|
cd ~/.openclaw/workspace/skills/property-assessor
|
||||||
npm install
|
npm install
|
||||||
|
scripts/property-assessor assess --address "<street-address>" --recipient-email "<target@example.com>"
|
||||||
scripts/property-assessor render-report --input "<report-payload-json>" --output "<output-pdf>"
|
scripts/property-assessor render-report --input "<report-payload-json>" --output "<output-pdf>"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
"description": "Property assessor helpers for public-record lookup and fixed-template PDF rendering",
|
"description": "Property assessor helpers for public-record lookup and fixed-template PDF rendering",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
"assess": "tsx src/cli.ts assess",
|
||||||
"locate-public-records": "tsx src/cli.ts locate-public-records",
|
"locate-public-records": "tsx src/cli.ts locate-public-records",
|
||||||
"render-report": "tsx src/cli.ts render-report",
|
"render-report": "tsx src/cli.ts render-report",
|
||||||
"test": "node --import tsx --test tests/*.test.ts"
|
"test": "node --import tsx --test tests/*.test.ts"
|
||||||
|
|||||||
227
skills/property-assessor/src/assessment.ts
Normal file
227
skills/property-assessor/src/assessment.ts
Normal file
@@ -0,0 +1,227 @@
|
|||||||
|
import os from "node:os";
|
||||||
|
import path from "node:path";
|
||||||
|
|
||||||
|
import { resolvePublicRecords, type PublicRecordsResolution } from "./public-records.js";
|
||||||
|
import { renderReportPdf, type ReportPayload } from "./report-pdf.js";
|
||||||
|
|
||||||
|
export interface AssessPropertyOptions {
|
||||||
|
address: string;
|
||||||
|
recipientEmails?: string[] | string;
|
||||||
|
output?: string;
|
||||||
|
parcelId?: string;
|
||||||
|
listingGeoId?: string;
|
||||||
|
listingSourceUrl?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AssessPropertyResult {
|
||||||
|
ok: true;
|
||||||
|
needsRecipientEmails: boolean;
|
||||||
|
message: string;
|
||||||
|
outputPath: string | null;
|
||||||
|
reportPayload: ReportPayload;
|
||||||
|
publicRecords: PublicRecordsResolution;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AssessPropertyDeps {
|
||||||
|
resolvePublicRecordsFn?: typeof resolvePublicRecords;
|
||||||
|
renderReportPdfFn?: typeof renderReportPdf;
|
||||||
|
}
|
||||||
|
|
||||||
|
function asStringArray(value: unknown): string[] {
|
||||||
|
if (value == null) return [];
|
||||||
|
if (typeof value === "string") {
|
||||||
|
return value
|
||||||
|
.split(",")
|
||||||
|
.map((item) => item.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
return value.flatMap((item) => asStringArray(item));
|
||||||
|
}
|
||||||
|
return [String(value).trim()].filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function slugify(value: string): string {
|
||||||
|
return value
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/[^a-z0-9]+/g, "-")
|
||||||
|
.replace(/^-+|-+$/g, "")
|
||||||
|
.slice(0, 80) || "property";
|
||||||
|
}
|
||||||
|
|
||||||
|
function pushLink(
|
||||||
|
target: Array<{ label: string; url: string }>,
|
||||||
|
label: string,
|
||||||
|
url: unknown
|
||||||
|
): void {
|
||||||
|
if (typeof url !== "string" || !url.trim()) return;
|
||||||
|
const normalized = url.trim();
|
||||||
|
if (!target.some((item) => item.url === normalized)) {
|
||||||
|
target.push({ label, url: normalized });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildPublicRecordLinks(
|
||||||
|
publicRecords: PublicRecordsResolution
|
||||||
|
): Array<{ label: string; url: string }> {
|
||||||
|
const links: Array<{ label: string; url: string }> = [];
|
||||||
|
pushLink(links, "Census Geocoder", publicRecords.officialLinks.censusGeocoder);
|
||||||
|
pushLink(
|
||||||
|
links,
|
||||||
|
"Texas Comptroller County Directory",
|
||||||
|
publicRecords.officialLinks.texasCountyDirectory
|
||||||
|
);
|
||||||
|
pushLink(
|
||||||
|
links,
|
||||||
|
"Texas Property Tax Portal",
|
||||||
|
publicRecords.officialLinks.texasPropertyTaxPortal
|
||||||
|
);
|
||||||
|
pushLink(links, "Appraisal District Website", publicRecords.appraisalDistrict?.Website);
|
||||||
|
pushLink(
|
||||||
|
links,
|
||||||
|
"Appraisal District Directory Page",
|
||||||
|
publicRecords.appraisalDistrict?.directoryPage
|
||||||
|
);
|
||||||
|
pushLink(
|
||||||
|
links,
|
||||||
|
"Tax Assessor / Collector Website",
|
||||||
|
publicRecords.taxAssessorCollector?.Website
|
||||||
|
);
|
||||||
|
pushLink(
|
||||||
|
links,
|
||||||
|
"Tax Assessor / Collector Directory Page",
|
||||||
|
publicRecords.taxAssessorCollector?.directoryPage
|
||||||
|
);
|
||||||
|
return links;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildAssessmentReportPayload(
|
||||||
|
options: AssessPropertyOptions,
|
||||||
|
publicRecords: PublicRecordsResolution
|
||||||
|
): ReportPayload {
|
||||||
|
const recipientEmails = asStringArray(options.recipientEmails);
|
||||||
|
const matchedAddress = publicRecords.matchedAddress || options.address;
|
||||||
|
const publicRecordLinks = buildPublicRecordLinks(publicRecords);
|
||||||
|
const sourceLinks = [...publicRecordLinks];
|
||||||
|
|
||||||
|
pushLink(sourceLinks, "Listing Source", options.listingSourceUrl);
|
||||||
|
|
||||||
|
const jurisdiction =
|
||||||
|
publicRecords.county.name && publicRecords.appraisalDistrict
|
||||||
|
? `${publicRecords.county.name} Appraisal District`
|
||||||
|
: publicRecords.county.name || publicRecords.state.name || "Official public-record jurisdiction";
|
||||||
|
|
||||||
|
const photoAttempts = options.listingSourceUrl
|
||||||
|
? [
|
||||||
|
`Listing source captured: ${options.listingSourceUrl}`,
|
||||||
|
"Photo review has not been run yet. Use the approval-safe Zillow/HAR extractor flow before making condition claims."
|
||||||
|
]
|
||||||
|
: [
|
||||||
|
"No listing source URL was provided to the assess helper.",
|
||||||
|
"Photo review has not been run yet. Use the approval-safe Zillow/HAR extractor flow before making condition claims."
|
||||||
|
];
|
||||||
|
|
||||||
|
return {
|
||||||
|
recipientEmails,
|
||||||
|
reportTitle: "Property Assessment Report",
|
||||||
|
subtitle: "Preliminary address-first intake with public-record enrichment.",
|
||||||
|
subjectProperty: {
|
||||||
|
address: matchedAddress,
|
||||||
|
county: publicRecords.county.name || "N/A",
|
||||||
|
state: publicRecords.state.code || publicRecords.state.name || "N/A",
|
||||||
|
geoid: publicRecords.geoid || "N/A"
|
||||||
|
},
|
||||||
|
verdict: {
|
||||||
|
decision: "pending",
|
||||||
|
fairValueRange: "Not established",
|
||||||
|
offerGuidance:
|
||||||
|
"Preliminary intake only. Official public-record jurisdiction is identified, but listing, photo, comp, and carry analysis still required before a buy/pass/offer conclusion."
|
||||||
|
},
|
||||||
|
snapshot: [
|
||||||
|
`Matched address: ${matchedAddress}`,
|
||||||
|
`Jurisdiction anchor: ${publicRecords.county.name || "Unknown county"}, ${publicRecords.state.code || publicRecords.state.name || "Unknown state"}.`,
|
||||||
|
publicRecords.geoid ? `Census block GEOID: ${publicRecords.geoid}.` : "Census block GEOID not returned."
|
||||||
|
],
|
||||||
|
whatILike: [
|
||||||
|
"Address-first flow avoids treating listing-site geo IDs as assessor record keys.",
|
||||||
|
publicRecords.appraisalDistrict
|
||||||
|
? "Official appraisal-district contact and website were identified from public records."
|
||||||
|
: "Official public-record geography was identified."
|
||||||
|
],
|
||||||
|
whatIDontLike: [
|
||||||
|
"This first-pass assess helper does not yet include listing facts, comp analysis, or a completed photo review.",
|
||||||
|
"Do not make valuation or condition claims from this preliminary output alone."
|
||||||
|
],
|
||||||
|
compView: [
|
||||||
|
"Comp analysis not yet run. Pull same-building or nearby comps before setting fair value."
|
||||||
|
],
|
||||||
|
carryView: [
|
||||||
|
"Carry-cost underwriting not yet run. Add taxes, HOA, insurance, maintenance, and vacancy assumptions before decisioning."
|
||||||
|
],
|
||||||
|
risksAndDiligence: publicRecords.lookupRecommendations,
|
||||||
|
photoReview: {
|
||||||
|
status: "not completed",
|
||||||
|
source: options.listingSourceUrl ? "listing source pending review" : "no listing source provided",
|
||||||
|
attempts: photoAttempts,
|
||||||
|
summary:
|
||||||
|
"Condition review is incomplete until listing photos are actually extracted and inspected."
|
||||||
|
},
|
||||||
|
publicRecords: {
|
||||||
|
jurisdiction,
|
||||||
|
accountNumber: options.parcelId || publicRecords.sourceIdentifierHints.parcelId,
|
||||||
|
ownerName: undefined,
|
||||||
|
landValue: undefined,
|
||||||
|
improvementValue: undefined,
|
||||||
|
assessedTotalValue: undefined,
|
||||||
|
exemptions: undefined,
|
||||||
|
links: publicRecordLinks
|
||||||
|
},
|
||||||
|
sourceLinks
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function assessProperty(
|
||||||
|
options: AssessPropertyOptions,
|
||||||
|
deps: AssessPropertyDeps = {}
|
||||||
|
): Promise<AssessPropertyResult> {
|
||||||
|
const resolvePublicRecordsFn = deps.resolvePublicRecordsFn || resolvePublicRecords;
|
||||||
|
const renderReportPdfFn = deps.renderReportPdfFn || renderReportPdf;
|
||||||
|
|
||||||
|
const publicRecords = await resolvePublicRecordsFn(options.address, {
|
||||||
|
parcelId: options.parcelId,
|
||||||
|
listingGeoId: options.listingGeoId,
|
||||||
|
listingSourceUrl: options.listingSourceUrl
|
||||||
|
});
|
||||||
|
|
||||||
|
const reportPayload = buildAssessmentReportPayload(options, publicRecords);
|
||||||
|
const recipientEmails = asStringArray(options.recipientEmails);
|
||||||
|
|
||||||
|
if (!recipientEmails.length) {
|
||||||
|
return {
|
||||||
|
ok: true,
|
||||||
|
needsRecipientEmails: true,
|
||||||
|
message:
|
||||||
|
"Missing target email. Stop and ask the user for target email address(es) before generating or sending the property assessment PDF.",
|
||||||
|
outputPath: null,
|
||||||
|
reportPayload,
|
||||||
|
publicRecords
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const outputPath =
|
||||||
|
options.output ||
|
||||||
|
path.join(
|
||||||
|
os.tmpdir(),
|
||||||
|
`property-assessment-${slugify(publicRecords.matchedAddress || options.address)}-${Date.now()}.pdf`
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderedPath = await renderReportPdfFn(reportPayload, outputPath);
|
||||||
|
return {
|
||||||
|
ok: true,
|
||||||
|
needsRecipientEmails: false,
|
||||||
|
message: `Property assessment PDF rendered: ${renderedPath}`,
|
||||||
|
outputPath: renderedPath,
|
||||||
|
reportPayload,
|
||||||
|
publicRecords
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -2,12 +2,14 @@
|
|||||||
|
|
||||||
import minimist from "minimist";
|
import minimist from "minimist";
|
||||||
|
|
||||||
|
import { assessProperty } from "./assessment.js";
|
||||||
import { resolvePublicRecords } from "./public-records.js";
|
import { resolvePublicRecords } from "./public-records.js";
|
||||||
import { ReportValidationError, loadReportPayload, renderReportPdf } from "./report-pdf.js";
|
import { ReportValidationError, loadReportPayload, renderReportPdf } from "./report-pdf.js";
|
||||||
|
|
||||||
function usage(): void {
|
function usage(): void {
|
||||||
process.stdout.write(`property-assessor\n
|
process.stdout.write(`property-assessor\n
|
||||||
Commands:
|
Commands:
|
||||||
|
assess --address "<address>" [--recipient-email "<email>"] [--output "<report.pdf>"] [--parcel-id "<id>"] [--listing-geo-id "<id>"] [--listing-source-url "<url>"]
|
||||||
locate-public-records --address "<address>" [--parcel-id "<id>"] [--listing-geo-id "<id>"] [--listing-source-url "<url>"]
|
locate-public-records --address "<address>" [--parcel-id "<id>"] [--listing-geo-id "<id>"] [--listing-source-url "<url>"]
|
||||||
render-report --input "<payload.json>" --output "<report.pdf>"
|
render-report --input "<payload.json>" --output "<report.pdf>"
|
||||||
`);
|
`);
|
||||||
@@ -27,6 +29,22 @@ async function main(): Promise<void> {
|
|||||||
process.exit(0);
|
process.exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (command === "assess") {
|
||||||
|
if (!argv.address) {
|
||||||
|
throw new Error("Missing required option: --address");
|
||||||
|
}
|
||||||
|
const payload = await assessProperty({
|
||||||
|
address: argv.address,
|
||||||
|
recipientEmails: argv["recipient-email"],
|
||||||
|
output: argv.output,
|
||||||
|
parcelId: argv["parcel-id"],
|
||||||
|
listingGeoId: argv["listing-geo-id"],
|
||||||
|
listingSourceUrl: argv["listing-source-url"]
|
||||||
|
});
|
||||||
|
process.stdout.write(`${JSON.stringify(payload, null, 2)}\n`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (command === "locate-public-records") {
|
if (command === "locate-public-records") {
|
||||||
if (!argv.address) {
|
if (!argv.address) {
|
||||||
throw new Error("Missing required option: --address");
|
throw new Error("Missing required option: --address");
|
||||||
|
|||||||
96
skills/property-assessor/tests/assessment.test.ts
Normal file
96
skills/property-assessor/tests/assessment.test.ts
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import fs from "node:fs";
|
||||||
|
import os from "node:os";
|
||||||
|
import path from "node:path";
|
||||||
|
import test from "node:test";
|
||||||
|
import assert from "node:assert/strict";
|
||||||
|
|
||||||
|
import { assessProperty } from "../src/assessment.js";
|
||||||
|
import type { PublicRecordsResolution } from "../src/public-records.js";
|
||||||
|
|
||||||
|
const samplePublicRecords: PublicRecordsResolution = {
|
||||||
|
requestedAddress: "4141 Whiteley Dr, Corpus Christi, TX 78418",
|
||||||
|
matchedAddress: "4141 WHITELEY DR, CORPUS CHRISTI, TX, 78418",
|
||||||
|
latitude: 27.6138,
|
||||||
|
longitude: -97.3024,
|
||||||
|
geoid: "483550031013005",
|
||||||
|
state: {
|
||||||
|
name: "Texas",
|
||||||
|
code: "TX",
|
||||||
|
fips: "48"
|
||||||
|
},
|
||||||
|
county: {
|
||||||
|
name: "Nueces County",
|
||||||
|
fips: "355",
|
||||||
|
geoid: "48355"
|
||||||
|
},
|
||||||
|
officialLinks: {
|
||||||
|
censusGeocoder: "https://geocoding.geo.census.gov/example",
|
||||||
|
texasCountyDirectory:
|
||||||
|
"https://comptroller.texas.gov/taxes/property-tax/county-directory/nueces.php",
|
||||||
|
texasPropertyTaxPortal: "https://texas.gov/PropertyTaxes"
|
||||||
|
},
|
||||||
|
appraisalDistrict: {
|
||||||
|
"Chief Appraiser": "Debra Morin, Interim",
|
||||||
|
Website: "http://www.ncadistrict.com/",
|
||||||
|
directoryPage:
|
||||||
|
"https://comptroller.texas.gov/taxes/property-tax/county-directory/nueces.php"
|
||||||
|
},
|
||||||
|
taxAssessorCollector: {
|
||||||
|
"Tax Assessor-Collector": "Kevin Kieschnick",
|
||||||
|
Website: "http://www.nuecesco.com"
|
||||||
|
},
|
||||||
|
lookupRecommendations: [
|
||||||
|
"Start from the official public-record jurisdiction instead of a listing-site geo ID."
|
||||||
|
],
|
||||||
|
sourceIdentifierHints: {
|
||||||
|
parcelId: "14069438"
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
test("assessProperty auto-enriches public-record data from address and asks for recipient email", async () => {
|
||||||
|
const result = await assessProperty(
|
||||||
|
{
|
||||||
|
address: "4141 Whiteley Dr, Corpus Christi, TX 78418",
|
||||||
|
listingSourceUrl: "https://www.zillow.com/homedetails/example",
|
||||||
|
listingGeoId: "233290",
|
||||||
|
parcelId: "14069438"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
resolvePublicRecordsFn: async () => samplePublicRecords
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.equal(result.ok, true);
|
||||||
|
assert.equal(result.needsRecipientEmails, true);
|
||||||
|
assert.equal(result.outputPath, null);
|
||||||
|
assert.match(result.message, /target email/i);
|
||||||
|
assert.equal(result.reportPayload.subjectProperty?.address, samplePublicRecords.matchedAddress);
|
||||||
|
assert.equal(result.reportPayload.publicRecords?.jurisdiction, "Nueces County Appraisal District");
|
||||||
|
assert.equal(result.reportPayload.publicRecords?.accountNumber, "14069438");
|
||||||
|
assert.equal(result.reportPayload.photoReview?.status, "not completed");
|
||||||
|
assert.match(
|
||||||
|
String(result.reportPayload.verdict?.offerGuidance),
|
||||||
|
/listing, photo, comp, and carry analysis still required/i
|
||||||
|
);
|
||||||
|
assert.deepEqual(result.reportPayload.recipientEmails, []);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("assessProperty renders a PDF when recipient email is present", async () => {
|
||||||
|
const outputPath = path.join(os.tmpdir(), `property-assess-command-${Date.now()}.pdf`);
|
||||||
|
const result = await assessProperty(
|
||||||
|
{
|
||||||
|
address: "4141 Whiteley Dr, Corpus Christi, TX 78418",
|
||||||
|
recipientEmails: ["buyer@example.com"],
|
||||||
|
output: outputPath
|
||||||
|
},
|
||||||
|
{
|
||||||
|
resolvePublicRecordsFn: async () => samplePublicRecords
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.equal(result.ok, true);
|
||||||
|
assert.equal(result.needsRecipientEmails, false);
|
||||||
|
assert.equal(result.outputPath, outputPath);
|
||||||
|
const stat = await fs.promises.stat(outputPath);
|
||||||
|
assert.ok(stat.size > 1000);
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user