Add honest accounting and automatic refund mechanism for completed jobs

Implement honest accounting post-job completion, calculating actual costs, adding margin, and enabling automatic refunds for overpayments via a new endpoint.

Replit-Commit-Author: Agent
Replit-Commit-Session-Id: 418bf6f8-212b-4bb0-a7a5-8231a061da4e
Replit-Commit-Checkpoint-Type: full_checkpoint
Replit-Commit-Event-Id: c6386de2-d5f4-47cc-a557-73416f09e118
Replit-Commit-Screenshot-Url: https://storage.googleapis.com/screenshot-production-us-central1/9f85e954-647c-46a5-90a7-396e495a805a/418bf6f8-212b-4bb0-a7a5-8231a061da4e/sPDHkg8
Replit-Helium-Checkpoint-Created: true
This commit is contained in:
alexpaynex
2026-03-18 19:32:34 +00:00
parent e5bdae7159
commit dfc9ecdc7b
15 changed files with 587 additions and 38 deletions

View File

@@ -26,6 +26,8 @@ export class LNbitsService {
}
}
// ── Inbound invoices ─────────────────────────────────────────────────────
async createInvoice(amountSats: number, memo: string): Promise<LNbitsInvoice> {
if (this.stubMode) {
const paymentHash = randomBytes(32).toString("hex");
@@ -34,7 +36,7 @@ export class LNbitsService {
return { paymentHash, paymentRequest };
}
const response = await fetch(`${this.url.replace(/\/$/, "")}/api/v1/payments`, {
const response = await fetch(`${this.base()}/api/v1/payments`, {
method: "POST",
headers: this.headers(),
body: JSON.stringify({ out: false, amount: amountSats, memo }),
@@ -58,7 +60,7 @@ export class LNbitsService {
}
const response = await fetch(
`${this.url.replace(/\/$/, "")}/api/v1/payments/${paymentHash}`,
`${this.base()}/api/v1/payments/${paymentHash}`,
{ method: "GET", headers: this.headers() },
);
@@ -71,7 +73,68 @@ export class LNbitsService {
return data.paid;
}
/** Stub-only helper: mark an invoice as paid for testing/dev flows. */
// ── Outbound payments (refunds) ──────────────────────────────────────────
/**
* Decode a BOLT11 invoice and return its amount in satoshis.
* Stub mode: parses the embedded amount from our known stub format (lnbcrtXu1stub_...),
* or returns null for externally-generated invoices.
*/
async decodeInvoice(bolt11: string): Promise<{ amountSats: number } | null> {
if (this.stubMode) {
// Our stub format: lnbcrtNNNu1stub_... where NNN is the amount in sats
const m = bolt11.match(/^lnbcrt(\d+)u1stub_/i);
if (m) return { amountSats: parseInt(m[1], 10) };
// Unknown format in stub mode — accept blindly (simulated environment)
return null;
}
const response = await fetch(`${this.base()}/api/v1/payments/decode`, {
method: "POST",
headers: this.headers(),
body: JSON.stringify({ data: bolt11 }),
});
if (!response.ok) {
const body = await response.text();
throw new Error(`LNbits decodeInvoice failed (${response.status}): ${body}`);
}
const data = (await response.json()) as { amount_msat?: number };
if (!data.amount_msat) return null;
return { amountSats: Math.floor(data.amount_msat / 1000) };
}
/**
* Pay an outgoing BOLT11 invoice (e.g. to return a refund to a user).
* Returns the payment hash.
* Stub mode: simulates a successful payment.
*/
async payInvoice(bolt11: string): Promise<string> {
if (this.stubMode) {
const paymentHash = randomBytes(32).toString("hex");
console.log(`[stub] Paid outgoing invoice — fake hash=${paymentHash}`);
return paymentHash;
}
const response = await fetch(`${this.base()}/api/v1/payments`, {
method: "POST",
headers: this.headers(),
body: JSON.stringify({ out: true, bolt11 }),
});
if (!response.ok) {
const body = await response.text();
throw new Error(`LNbits payInvoice failed (${response.status}): ${body}`);
}
const data = (await response.json()) as { payment_hash: string };
return data.payment_hash;
}
// ── Stub helpers ─────────────────────────────────────────────────────────
/** Stub-only: mark an inbound invoice as paid for testing/dev flows. */
stubMarkPaid(paymentHash: string): void {
if (!this.stubMode) {
throw new Error("stubMarkPaid called on a real LNbitsService instance");
@@ -80,6 +143,12 @@ export class LNbitsService {
console.log(`[stub] Marked invoice paid: hash=${paymentHash}`);
}
// ── Private helpers ──────────────────────────────────────────────────────
private base(): string {
return this.url.replace(/\/$/, "");
}
private headers(): Record<string, string> {
return {
"Content-Type": "application/json",

View File

@@ -158,6 +158,35 @@ export class PricingService {
const amountSats = usdToSats(estimatedCostUsd, btcPriceUsd);
return { amountSats, estimatedCostUsd, marginPct: this.marginPct, btcPriceUsd };
}
// ── Post-work honest accounting ──────────────────────────────────────────
/**
* Full actual charge in USD: raw Anthropic token cost + DO infra amortisation + margin.
* Pass in the already-computed actualCostUsd (raw token cost, no extras).
*/
calculateActualChargeUsd(actualCostUsd: number): number {
const rawCostUsd = actualCostUsd + DO_INFRA_PER_REQUEST_USD;
return rawCostUsd * (1 + this.marginPct / 100);
}
/**
* Convert the actual charge to satoshis using the BTC price that was locked
* at invoice-creation time. This keeps pre- and post-work accounting in the
* same BTC denomination without a second oracle call.
*/
calculateActualChargeSats(actualCostUsd: number, lockedBtcPriceUsd: number): number {
const chargeUsd = this.calculateActualChargeUsd(actualCostUsd);
return usdToSats(chargeUsd, lockedBtcPriceUsd);
}
/**
* Refund amount in sats: what was overpaid by the user.
* Always >= 0 (clamped — never ask the user to top up due to BTC price swings).
*/
calculateRefundSats(workAmountSats: number, actualAmountSats: number): number {
return Math.max(0, workAmountSats - actualAmountSats);
}
}
export const pricingService = new PricingService();