Transaction

a858c42e6451d15dcec1126a42c071feaf3a2d7150b6d9c4d0c1291228b31ad9
Timestamp (utc)
2025-02-04 16:38:13
Fee Paid
0.00000003 BSV
(
0.00022932 BSV
-
0.00022929 BSV
)
Fee Rate
1.234 sat/KB
Version
1
Confirmations
38,723
Size Stats
2,430 B

42 Inputs

Total Input:
0.00022932 BSV
  • jM&async function updateWalletBalance(address) { try { const response = await fetch(`https://api.whatsonchain.com/v1/bsv/main/address/${address}/balance`); const data = await response.json(); const confirmedBalance = data.confirmed || 0; const unconfirmedBalance = data.unconfirmed || 0; const totalSatoshis = confirmedBalance + unconfirmedBalance; const bsvBalance = totalSatoshis / 1e8; return { success: true, bsv: bsvBalance.toFixed(8), satoshis: totalSatoshis, confirmed: confirmedBalance, unconfirmed: unconfirmedBalance }; } catch (error) { return { success: false, error: 'Failed to fetch balance' }; } }
    https://whatsonchain.com/tx/undefined
  • jM@{"name":"EncodedFiles","files":[{"name":"publishmodaleC4.html","size":12209,"type":"text/html","content":"const PublishModal = ({ txid, postgresRecord, onSuccess, onClose }) => {
    const [deployStatus, setDeployStatus] = useState('');
    const [isDeploying, setIsDeploying] = useState(false);
    const [cost, setCost] = useState(null);
    const [txids, setTxids] = useState({ payment: null, deploy: null });
    const [timeLeft, setTimeLeft] = useState(600);
    const [paymentAddress, setPaymentAddress] = useState(null);
    const qrRef = useRef(null);

    // Add QR code effect
    useEffect(() => {
        if (deployStatus === 'waiting_payment' && paymentAddress && cost && qrRef.current) {
            // Clear any existing QR code
            qrRef.current.innerHTML = '';
            // Create new QR code
            new QRCode(qrRef.current, {
                text: `bitcoin:${paymentAddress}?amount=${cost.toFixed(8)}`,
                width: 128,
                height: 128
            });
        }
    }, [deployStatus, paymentAddress, cost]);

    // Add timer effect
    useEffect(() => {
        let timer;
        if (deployStatus === 'waiting_payment' && timeLeft > 0) {
            timer = setInterval(() => {
                setTimeLeft(prev => {
                    if (prev <= 1) {
                        setDeployStatus('timeout');
                        return 0;
                    }
                    return prev - 1;
                });
            }, 1000);
        }
        return () => clearInterval(timer);
    }, [deployStatus, timeLeft]);


    // Add the checkPayment function with its endpoints
    const PAYMENT_ENDPOINTS = [
        {
            name: 'WhatsOnChain',
            url: (address) => `https://api.whatsonchain.com/v1/bsv/main/address/${address}/unspent`,
            parser: (data, expectedSatoshis) => {
                if (!Array.isArray(data)) return null;
                console.log('WhatsOnChain UTXOs:', data);
                const utxo = data.find((utxo) => utxo.value >= expectedSatoshis);
                if (utxo) {
                    console.log(`WhatsOnChain found payment of ${utxo.value} satoshis - TXID: ${utxo.tx_hash}`);
                }
                return utxo ? utxo.tx_hash : null;
            },
        },
        {
            name: 'Bitails',
            url: (address) => `https://api.bitails.io/address/${address}/unspent`,
            parser: (data, expectedSatoshis) => {
                if (!data?.unspent || !Array.isArray(data.unspent)) return null;
                console.log('Bitails data:', data);
                const utxo = data.unspent.find((utxo) => utxo.satoshis >= expectedSatoshis);
                if (utxo) {
                    console.log(`Bitails found payment of ${utxo.satoshis} satoshis - TXID: ${utxo.txid}`);
                }
                return utxo ? utxo.txid : null;
            },
        },
    ];

    const checkPayment = async (address, expectedAmount) => {
        const expectedSatoshis = Math.floor(expectedAmount * 100000000);
        const maxAttempts = 120;
        let attempts = 0;
        let endpointIndex = 0;

        console.log(`Starting payment check for address: ${address}`);
        console.log(`Expected payment: ${expectedSatoshis} satoshis`);

        return new Promise((resolve, reject) => {
            const checkInterval = setInterval(async () => {
                attempts++;
                console.log(`Attempt ${attempts}/${maxAttempts}`);

                if (attempts > maxAttempts) {
                    console.error('Payment check timed out.');
                    clearInterval(checkInterval);
                    reject(new Error('Payment timeout'));
                    return;
                }

                const endpoint = PAYMENT_ENDPOINTS[endpointIndex];
                console.log(`Checking with endpoint: ${endpoint.name}`);

                try {
                    const response = await fetch(endpoint.url(address));
                    if (response.ok) {
                        const data = await response.json();
                        console.log(`${endpoint.name} response:`, data);

                        const txid = endpoint.parser(data, expectedSatoshis);
                        if (txid) {
                            console.log(`Payment found! TXID: ${txid}`);
                            clearInterval(checkInterval);
                            resolve(txid);
                            return;
                        } else {
                            console.log(`No valid UTXO found in ${endpoint.name} response.`);
                        }
                    }
                } catch (error) {
                    console.error(`Error during ${endpoint.name} check:`, error);
                }

                endpointIndex = (endpointIndex + 1) % PAYMENT_ENDPOINTS.length;
            }, 5000);
        });
    };

const deployToPostgre = async () => {
    console.log('Starting PostgreSQL deployment process');
    console.log('Using postgresRecord:', postgresRecord);
    console.log('Using txid:', txid);
    
    // Validate required fields first
    if (!postgresRecord || !txid) {
        console.error('Missing base requirements:', { postgresRecord, txid });
        throw new Error('Missing required fields');
    }

    // Validate all required fields exist
    const requiredFields = [
        'txid',
        'wallet',
        'internet_indicator',
        'content_type',
        'start_date',
        'status'
    ];

    const missingFields = requiredFields.filter(field => !postgresRecord[field]);
    if (missingFields.length > 0) {
        console.error('Missing required fields:', missingFields);
        throw new Error(`Missing required fields: ${missingFields.join(', ')}`);
    }

    setIsDeploying(true);
    try {
        // Set a fixed cost of 0.0001 BSV
        const fixedCost = 0.0001;
        setCost(fixedCost);

        // Get payment address
        const addressResponse = await fetch('https://bsvhost.com/address-server/getPaymentAddress');
        if (!addressResponse.ok) throw new Error('Failed to get payment address');
        const { address } = await addressResponse.json();
        setPaymentAddress(address);
        setDeployStatus('waiting_payment');
        setTimeLeft(600);

        // Monitor for payment
        const paymentTxid = await checkPayment(address, fixedCost);
        setTxids(prev => ({ ...prev, payment: paymentTxid }));
        setDeployStatus('payment_received');

        // Create sanitized record
        const sanitizedRecord = {
            txid: String(postgresRecord.txid || '').trim(),
            wallet: String(postgresRecord.wallet || '').trim(),
            internet_indicator: String(postgresRecord.internet_indicator || 'B').trim(),
            content_type: String(postgresRecord.content_type || 'B').trim(),
            product_type: String(postgresRecord.product_type || 'D').trim(),
            cost_type: String(postgresRecord.cost_type || 'F').trim(),
            language: String(postgresRecord.language || 'E').trim(),
            view_url: String(postgresRecord.view_url || '').trim(),
            content_title: String(postgresRecord.content_title || '').trim(),
            search_tags: String(postgresRecord.search_tags || '').trim(),
            start_date: postgresRecord.start_date || new Date().toISOString().split('T')[0],
            end_date: postgresRecord.end_date || null,
            status: String(postgresRecord.status || 'A').trim()
        };

        const publishData = {
            ...sanitizedRecord,
            payment_txid: paymentTxid
        };

        const serverUrl = window.location.hostname === 'localhost' 
            ? 'https://localhost:8000'
            : 'https://bsvgolf.com:8000';

        console.log('Sending to server:', JSON.stringify(publishData, null, 2));

        const response = await fetch(`${serverUrl}/publish`, {
            method: 'POST',
            headers: {
                'Content-Type': 'application/json',
                'Accept': 'application/json'
            },
            body: JSON.stringify(publishData)
        });

        if (!response.ok) {
            const errorData = await response.json().catch(() => null);
            console.error('Server error:', errorData);
            throw new Error(errorData?.message || 'Failed to publish to PostgreSQL');
        }

        const result = await response.json();
        console.log('Publish successful:', result);
        setDeployStatus('success');
        onSuccess({ txid, paymentTxid });

    } catch (error) {
        console.error('Deployment error:', error);
        setDeployStatus('error');
        throw error;
    } finally {
        setIsDeploying(false);
    }
};

    return (
        <div className="fixed inset-0 z-50 flex items-center justify-center bg-black bg-opacity-50">
            <div className="bg-white p-6 rounded-lg shadow-xl max-w-md w-full">
                <h2 className="text-xl font-bold mb-4">Publish to Index</h2>

                {deployStatus === 'waiting_payment' && (
                    <div className="text-center mb-4">
                        {/* QR Code */}
                        <div ref={qrRef} className="mb-4 flex justify-center"/>
                        
                        {/* Payment Details */}
                        <div className="p-4 border rounded">
                            <div className="mb-4">
                                <label className="block font-bold mb-2">Payment Address:</label>
                                <input 
                                    type="text"
                                    readOnly
                                    value={paymentAddress || ''}
                                    className="w-full p-2 border rounded bg-gray-50"
                                    onClick={e => e.target.select()}
                                />
                            </div>
                            
                            <div>
                                <label className="block font-bold mb-2">Amount (BSV):</label>
                                <input 
                                    type="text"
                                    readOnly
                                    value={cost?.toFixed(8) || ''}
                                    className="w-full p-2 border rounded bg-gray-50"
                                    onClick={e => e.target.select()}
                                />
                            </div>
                        </div>

                        {/* Timer */}
                        <p className="mt-4">
                            Time remaining: {Math.floor(timeLeft / 60)}:{(timeLeft % 60).toString().padStart(2, '0')}
                        </p>
                    </div>
                )}

                {deployStatus === 'success' && (
                    <div className="text-center mb-4">
                        <p className="text-green-600 text-xl">✓ Published successfully!</p>
                        <div className="mt-4">
                            <p className="font-bold">Payment TXID:</p>
                            <code className="break-all">{txids.payment}</code>
                        </div>
                    </div>
                )}

                {deployStatus === 'error' && (
                    <div className="text-red-600 text-center mb-4">
                        Publication failed. Please try again.
                    </div>
                )}

                <div className="flex justify-end gap-4">
                    {!isDeploying && !deployStatus && (
    <button 
        onClick={() => deployToPostgre(postgresRecord, txid)} // Add arrow function here
        className="px-4 py-2 bg-blue-600 text-white rounded hover:bg-blue-700"
    >
        Start Publication
    </button>
)}
                    <button 
                        onClick={onClose}
                        className="px-4 py-2 bg-gray-600 text-white rounded hover:bg-gray-700"
                    >
                        {deployStatus === 'success' ? 'Close' : 'Cancel'}
                    </button>
                </div>
            </div>
        </div>
    );
};"}]}
    https://whatsonchain.com/tx/undefined
Total Output:
0.00022932 BSV

2 Outputs

Total Output:
0.00022929 BSV