Files
actual-helpers/zestimate.js
T
rlkollman 7899b45e9f
Create and publish Docker image on GHCR / publish-docker-image (push) Has been cancelled
Update zestimate to use puppeteer instead of selenium
2026-05-24 09:00:21 -05:00

106 lines
3.2 KiB
JavaScript

const puppeteer = require('puppeteer-extra');
const StealthPlugin = require('puppeteer-extra-plugin-stealth');
const api = require('@actual-app/api');
const { closeBudget, ensurePayee, getAccountBalance, getAccountNote, getTagValue, openBudget, showPercent, sleep } = require('./utils');
require("dotenv").config();
puppeteer.use(StealthPlugin());
async function getZestimate(URL) {
const browser = await puppeteer.launch({
headless: 'new',
args: ['--no-sandbox', '--disable-setuid-sandbox']
});
const page = await browser.newPage();
try {
await page.goto(URL, { waitUntil: 'domcontentloaded', timeout: 20000 });
const html = await page.content();
try {
console.log('Got html data!');
let match = html.match(/"price":"(\d+)"/);
if (match) {
console.log('matched 1st');
return parseInt(match[1]) * 100;
}
match = html.match(/\\"price\\":(\d+)/);
if (match) {
console.log('matched 2nd');
return parseInt(match[1]) * 100;
}
match = html.match(/\\"price\\":\\"(\d+)\\"/);
if (match) {
console.log('matched 3rd');
return parseInt(match[1]) * 100;
}
console.log('didn\'t match any :(');
console.log('~~!!');
// console.log(html);
console.log('!!~~');
} catch (error) {
console.log('Error parsing Zillow page:');
console.log(error);
console.log(html);
}
} catch (er) {
console.log('Error while fetching zestimate:');
console.log(er);
console.lo(html);
} finally {
await browser.close();
}
return undefined;
}
(async function() {
await openBudget();
const payeeId = await ensurePayee(process.env.ZESTIMATE_PAYEE_NAME || 'Zestimate');
const accounts = await api.getAccounts();
for (const account of accounts) {
const note = await getAccountNote(account);
if (note && note.indexOf('zestimate:') > -1) {
const URL = getTagValue(note, 'zestimate');
let ownership = 1;
if (note.indexOf('ownership:') > -1) {
ownership = parseFloat(getTagValue(note, 'ownership'));
}
console.log('Fetching zestimate for account:', account.name);
console.log('Zillow URL:', URL);
const zestimate = await getZestimate(URL);
if (!zestimate) {
console.log('Was unable to get Zestimate, skipping');
continue;
}
const balance = await getAccountBalance(account);
const diff = (zestimate * ownership) - balance;
console.log('Zestimate:', zestimate);
console.log('Ownership:', zestimate * ownership);
console.log('Balance:', balance);
console.log('Difference:', diff);
if (diff != 0) {
await api.importTransactions(account.id, [{
date: new Date(),
payee: payeeId,
amount: diff,
cleared: true,
reconciled: true,
notes: `Update Zestimate to ${zestimate * ownership / 100} (${zestimate / 100}*${showPercent(ownership)})`,
}]);
}
await sleep(1324);
}
}
await closeBudget();
})();