76
									
								
								cli.js
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										76
									
								
								cli.js
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,76 @@ | ||||
| #!/usr/bin/env node | ||||
|  | ||||
| const actual = require('@actual-app/api'); | ||||
| const yargs = require('yargs/yargs'); | ||||
| const { hideBin } = require('yargs/helpers'); | ||||
|  | ||||
| (async () => { | ||||
|     const argv = yargs(hideBin(process.argv)) | ||||
|         .option('account', { | ||||
|             alias: 'a', | ||||
|             type: 'string', | ||||
|             demandOption: true, | ||||
|             describe: 'The UUID of the account', | ||||
|         }) | ||||
|         .option('payee', { | ||||
|             alias: 'p', | ||||
|             type: 'string', | ||||
|             demandOption: true, | ||||
|             describe: 'The name of the payee', | ||||
|         }) | ||||
|         .option('amount', { | ||||
|             alias: 'm', | ||||
|             type: 'number', | ||||
|             demandOption: true, | ||||
|             describe: 'The amount in dollars (positive or negative)', | ||||
|         }) | ||||
|         .option('date', { alias: 'd', type: 'string', demandOption: true, describe: 'The date in ISO format' }) | ||||
|         .option('notes', { alias: 'n', type: 'string', demandOption: false, describe: 'Notes' }) | ||||
|         .option('server', { | ||||
|             alias: 's', | ||||
|             type: 'string', | ||||
|             describe: 'Actual server URL (fallback if ACTUAL_SERVER is not set)', | ||||
|         }) | ||||
|         .option('password', { | ||||
|             alias: 'w', | ||||
|             type: 'string', | ||||
|             describe: 'Password (fallback if ACTUAL_PASSWORD is not set)', | ||||
|         }) | ||||
|         .option('sync-id', { alias: 'b', type: 'string', describe: 'UUID of the budget to connect to' }) | ||||
|         .help() | ||||
|         .argv; | ||||
|  | ||||
|     const serverURL = process.env.ACTUAL_SERVER || argv.server; | ||||
|     const password = process.env.ACTUAL_PASSWORD || argv.password; | ||||
|     const syncId = process.env.ACTUAL_SYNC_ID || argv.sync_id; | ||||
|  | ||||
|     if (!serverURL || !password || !syncId) { | ||||
|         console.error('❌ Error: server URL, budget ID, and password must be provided (via CLI or environment)'); | ||||
|         process.exit(1); | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|         await actual.init({ | ||||
|             dataDir: './cache', | ||||
|             serverURL: serverURL, | ||||
|             password: password, | ||||
|         }); | ||||
|         await actual.downloadBudget(syncId) | ||||
|  | ||||
|         const amount = Math.round(argv.amount * 100); // Convert to milli-units | ||||
|  | ||||
|         await actual.importTransactions(argv.account, [{ | ||||
|             imported_payee: argv.payee, | ||||
|             payee_name: argv.payee, | ||||
|             amount, | ||||
|             date: argv.date, // today's date | ||||
|             notes: argv.notes ?? '' | ||||
|         }]); | ||||
|  | ||||
|         console.log('✅ Transaction added successfully.'); | ||||
|         await actual.shutdown(); | ||||
|     } catch (err) { | ||||
|         console.error('❌ Error:', err.message || err); | ||||
|         process.exit(1); | ||||
|     } | ||||
| })(); | ||||
							
								
								
									
										13
									
								
								model.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								model.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| from dataclasses import dataclass | ||||
| from uuid import UUID | ||||
| from datetime import date | ||||
| from decimal import Decimal | ||||
|  | ||||
| @dataclass | ||||
| class Transaction: | ||||
|     account: UUID | ||||
|     date: date | ||||
|     amount: Decimal | ||||
|     payee: str  # imported_payee in API | ||||
|     notes: str | ||||
|     imported_id: str | ||||
							
								
								
									
										794
									
								
								package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										794
									
								
								package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,794 @@ | ||||
| { | ||||
|   "name": "actual-cli-push", | ||||
|   "version": "1.0.0", | ||||
|   "lockfileVersion": 3, | ||||
|   "requires": true, | ||||
|   "packages": { | ||||
|     "": { | ||||
|       "name": "actual-cli-push", | ||||
|       "version": "1.0.0", | ||||
|       "license": "ISC", | ||||
|       "dependencies": { | ||||
|         "@actual-app/api": "^25.4.0", | ||||
|         "yargs": "^17.7.2" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "actual-cli-push": "cli.js" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@actual-app/api": { | ||||
|       "version": "25.4.0", | ||||
|       "resolved": "https://registry.npmjs.org/@actual-app/api/-/api-25.4.0.tgz", | ||||
|       "integrity": "sha512-Amxn18rKfUDrLyebQ040NITAbymB2k+REd/XP1lnh99VYxakqLY6MHAH8WNLLfgIJETBZAf5mKYYCBOv05t2gg==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "@actual-app/crdt": "^2.1.0", | ||||
|         "better-sqlite3": "^11.9.1", | ||||
|         "compare-versions": "^6.1.1", | ||||
|         "node-fetch": "^3.3.2", | ||||
|         "uuid": "^9.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=18.12.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@actual-app/crdt": { | ||||
|       "version": "2.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/@actual-app/crdt/-/crdt-2.1.0.tgz", | ||||
|       "integrity": "sha512-Qb8hMq10Wi2kYIDj0fG4uy00f9Mloghd+xQrHQiPQfgx022VPJ/No+z/bmfj4MuFH8FrPiLysSzRsj2PNQIedw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "google-protobuf": "^3.12.0-rc.1", | ||||
|         "murmurhash": "^2.0.1", | ||||
|         "uuid": "^9.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ansi-regex": { | ||||
|       "version": "5.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", | ||||
|       "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ansi-styles": { | ||||
|       "version": "4.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", | ||||
|       "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "color-convert": "^2.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/chalk/ansi-styles?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/base64-js": { | ||||
|       "version": "1.5.1", | ||||
|       "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", | ||||
|       "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/better-sqlite3": { | ||||
|       "version": "11.9.1", | ||||
|       "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-11.9.1.tgz", | ||||
|       "integrity": "sha512-Ba0KR+Fzxh2jDRhdg6TSH0SJGzb8C0aBY4hR8w8madIdIzzC6Y1+kx5qR6eS1Z+Gy20h6ZU28aeyg0z1VIrShQ==", | ||||
|       "hasInstallScript": true, | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "bindings": "^1.5.0", | ||||
|         "prebuild-install": "^7.1.1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/bindings": { | ||||
|       "version": "1.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", | ||||
|       "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "file-uri-to-path": "1.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/bl": { | ||||
|       "version": "4.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", | ||||
|       "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "buffer": "^5.5.0", | ||||
|         "inherits": "^2.0.4", | ||||
|         "readable-stream": "^3.4.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/buffer": { | ||||
|       "version": "5.7.1", | ||||
|       "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", | ||||
|       "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "base64-js": "^1.3.1", | ||||
|         "ieee754": "^1.1.13" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/chownr": { | ||||
|       "version": "1.1.4", | ||||
|       "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", | ||||
|       "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", | ||||
|       "license": "ISC" | ||||
|     }, | ||||
|     "node_modules/cliui": { | ||||
|       "version": "8.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", | ||||
|       "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", | ||||
|       "license": "ISC", | ||||
|       "dependencies": { | ||||
|         "string-width": "^4.2.0", | ||||
|         "strip-ansi": "^6.0.1", | ||||
|         "wrap-ansi": "^7.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=12" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/color-convert": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", | ||||
|       "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "color-name": "~1.1.4" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=7.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/color-name": { | ||||
|       "version": "1.1.4", | ||||
|       "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", | ||||
|       "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/compare-versions": { | ||||
|       "version": "6.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/compare-versions/-/compare-versions-6.1.1.tgz", | ||||
|       "integrity": "sha512-4hm4VPpIecmlg59CHXnRDnqGplJFrbLG4aFEl5vl6cK1u76ws3LLvX7ikFnTDl5vo39sjWD6AaDPYodJp/NNHg==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/data-uri-to-buffer": { | ||||
|       "version": "4.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz", | ||||
|       "integrity": "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">= 12" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/decompress-response": { | ||||
|       "version": "6.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", | ||||
|       "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "mimic-response": "^3.1.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/sindresorhus" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/deep-extend": { | ||||
|       "version": "0.6.0", | ||||
|       "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", | ||||
|       "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=4.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/detect-libc": { | ||||
|       "version": "2.0.4", | ||||
|       "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.4.tgz", | ||||
|       "integrity": "sha512-3UDv+G9CsCKO1WKMGw9fwq/SWJYbI0c5Y7LU1AXYoDdbhE2AHQ6N6Nb34sG8Fj7T5APy8qXDCKuuIHd1BR0tVA==", | ||||
|       "license": "Apache-2.0", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/emoji-regex": { | ||||
|       "version": "8.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", | ||||
|       "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/end-of-stream": { | ||||
|       "version": "1.4.4", | ||||
|       "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", | ||||
|       "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "once": "^1.4.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/escalade": { | ||||
|       "version": "3.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", | ||||
|       "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/expand-template": { | ||||
|       "version": "2.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", | ||||
|       "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==", | ||||
|       "license": "(MIT OR WTFPL)", | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/fetch-blob": { | ||||
|       "version": "3.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.2.0.tgz", | ||||
|       "integrity": "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/jimmywarting" | ||||
|         }, | ||||
|         { | ||||
|           "type": "paypal", | ||||
|           "url": "https://paypal.me/jimmywarting" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "node-domexception": "^1.0.0", | ||||
|         "web-streams-polyfill": "^3.0.3" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": "^12.20 || >= 14.13" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/file-uri-to-path": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", | ||||
|       "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/formdata-polyfill": { | ||||
|       "version": "4.0.10", | ||||
|       "resolved": "https://registry.npmjs.org/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz", | ||||
|       "integrity": "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "fetch-blob": "^3.1.2" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=12.20.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/fs-constants": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", | ||||
|       "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/get-caller-file": { | ||||
|       "version": "2.0.5", | ||||
|       "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", | ||||
|       "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", | ||||
|       "license": "ISC", | ||||
|       "engines": { | ||||
|         "node": "6.* || 8.* || >= 10.*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/github-from-package": { | ||||
|       "version": "0.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", | ||||
|       "integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/google-protobuf": { | ||||
|       "version": "3.21.4", | ||||
|       "resolved": "https://registry.npmjs.org/google-protobuf/-/google-protobuf-3.21.4.tgz", | ||||
|       "integrity": "sha512-MnG7N936zcKTco4Jd2PX2U96Kf9PxygAPKBug+74LHzmHXmceN16MmRcdgZv+DGef/S9YvQAfRsNCn4cjf9yyQ==", | ||||
|       "license": "(BSD-3-Clause AND Apache-2.0)" | ||||
|     }, | ||||
|     "node_modules/ieee754": { | ||||
|       "version": "1.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", | ||||
|       "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ], | ||||
|       "license": "BSD-3-Clause" | ||||
|     }, | ||||
|     "node_modules/inherits": { | ||||
|       "version": "2.0.4", | ||||
|       "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", | ||||
|       "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", | ||||
|       "license": "ISC" | ||||
|     }, | ||||
|     "node_modules/ini": { | ||||
|       "version": "1.3.8", | ||||
|       "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", | ||||
|       "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", | ||||
|       "license": "ISC" | ||||
|     }, | ||||
|     "node_modules/is-fullwidth-code-point": { | ||||
|       "version": "3.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", | ||||
|       "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/mimic-response": { | ||||
|       "version": "3.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", | ||||
|       "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/sindresorhus" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/minimist": { | ||||
|       "version": "1.2.8", | ||||
|       "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", | ||||
|       "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", | ||||
|       "license": "MIT", | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/ljharb" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/mkdirp-classic": { | ||||
|       "version": "0.5.3", | ||||
|       "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", | ||||
|       "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/murmurhash": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/murmurhash/-/murmurhash-2.0.1.tgz", | ||||
|       "integrity": "sha512-5vQEh3y+DG/lMPM0mCGPDnyV8chYg/g7rl6v3Gd8WMF9S429ox3Xk8qrk174kWhG767KQMqqxLD1WnGd77hiew==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/napi-build-utils": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-2.0.0.tgz", | ||||
|       "integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/node-abi": { | ||||
|       "version": "3.74.0", | ||||
|       "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.74.0.tgz", | ||||
|       "integrity": "sha512-c5XK0MjkGBrQPGYG24GBADZud0NCbznxNx0ZkS+ebUTrmV1qTDxPxSL8zEAPURXSbLRWVexxmP4986BziahL5w==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "semver": "^7.3.5" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/node-domexception": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", | ||||
|       "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==", | ||||
|       "deprecated": "Use your platform's native DOMException instead", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/jimmywarting" | ||||
|         }, | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://paypal.me/jimmywarting" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=10.5.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/node-fetch": { | ||||
|       "version": "3.3.2", | ||||
|       "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.3.2.tgz", | ||||
|       "integrity": "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "data-uri-to-buffer": "^4.0.0", | ||||
|         "fetch-blob": "^3.1.4", | ||||
|         "formdata-polyfill": "^4.0.10" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": "^12.20.0 || ^14.13.1 || >=16.0.0" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "type": "opencollective", | ||||
|         "url": "https://opencollective.com/node-fetch" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/once": { | ||||
|       "version": "1.4.0", | ||||
|       "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", | ||||
|       "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", | ||||
|       "license": "ISC", | ||||
|       "dependencies": { | ||||
|         "wrappy": "1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/prebuild-install": { | ||||
|       "version": "7.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.3.tgz", | ||||
|       "integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "detect-libc": "^2.0.0", | ||||
|         "expand-template": "^2.0.3", | ||||
|         "github-from-package": "0.0.0", | ||||
|         "minimist": "^1.2.3", | ||||
|         "mkdirp-classic": "^0.5.3", | ||||
|         "napi-build-utils": "^2.0.0", | ||||
|         "node-abi": "^3.3.0", | ||||
|         "pump": "^3.0.0", | ||||
|         "rc": "^1.2.7", | ||||
|         "simple-get": "^4.0.0", | ||||
|         "tar-fs": "^2.0.0", | ||||
|         "tunnel-agent": "^0.6.0" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "prebuild-install": "bin.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/pump": { | ||||
|       "version": "3.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.2.tgz", | ||||
|       "integrity": "sha512-tUPXtzlGM8FE3P0ZL6DVs/3P58k9nk8/jZeQCurTJylQA8qFYzHFfhBJkuqyE0FifOsQ0uKWekiZ5g8wtr28cw==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "end-of-stream": "^1.1.0", | ||||
|         "once": "^1.3.1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/rc": { | ||||
|       "version": "1.2.8", | ||||
|       "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", | ||||
|       "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", | ||||
|       "license": "(BSD-2-Clause OR MIT OR Apache-2.0)", | ||||
|       "dependencies": { | ||||
|         "deep-extend": "^0.6.0", | ||||
|         "ini": "~1.3.0", | ||||
|         "minimist": "^1.2.0", | ||||
|         "strip-json-comments": "~2.0.1" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "rc": "cli.js" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/readable-stream": { | ||||
|       "version": "3.6.2", | ||||
|       "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", | ||||
|       "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "inherits": "^2.0.3", | ||||
|         "string_decoder": "^1.1.1", | ||||
|         "util-deprecate": "^1.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/require-directory": { | ||||
|       "version": "2.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", | ||||
|       "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/safe-buffer": { | ||||
|       "version": "5.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", | ||||
|       "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/semver": { | ||||
|       "version": "7.7.1", | ||||
|       "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", | ||||
|       "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", | ||||
|       "license": "ISC", | ||||
|       "bin": { | ||||
|         "semver": "bin/semver.js" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/simple-concat": { | ||||
|       "version": "1.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", | ||||
|       "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/simple-get": { | ||||
|       "version": "4.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz", | ||||
|       "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "decompress-response": "^6.0.0", | ||||
|         "once": "^1.3.1", | ||||
|         "simple-concat": "^1.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/string_decoder": { | ||||
|       "version": "1.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", | ||||
|       "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "safe-buffer": "~5.2.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/string-width": { | ||||
|       "version": "4.2.3", | ||||
|       "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", | ||||
|       "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "emoji-regex": "^8.0.0", | ||||
|         "is-fullwidth-code-point": "^3.0.0", | ||||
|         "strip-ansi": "^6.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/strip-ansi": { | ||||
|       "version": "6.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", | ||||
|       "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "ansi-regex": "^5.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/strip-json-comments": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", | ||||
|       "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/tar-fs": { | ||||
|       "version": "2.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.2.tgz", | ||||
|       "integrity": "sha512-EsaAXwxmx8UB7FRKqeozqEPop69DXcmYwTQwXvyAPF352HJsPdkVhvTaDPYqfNgruveJIJy3TA2l+2zj8LJIJA==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "chownr": "^1.1.1", | ||||
|         "mkdirp-classic": "^0.5.2", | ||||
|         "pump": "^3.0.0", | ||||
|         "tar-stream": "^2.1.4" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/tar-stream": { | ||||
|       "version": "2.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", | ||||
|       "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "bl": "^4.0.3", | ||||
|         "end-of-stream": "^1.4.1", | ||||
|         "fs-constants": "^1.0.0", | ||||
|         "inherits": "^2.0.3", | ||||
|         "readable-stream": "^3.1.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/tunnel-agent": { | ||||
|       "version": "0.6.0", | ||||
|       "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", | ||||
|       "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", | ||||
|       "license": "Apache-2.0", | ||||
|       "dependencies": { | ||||
|         "safe-buffer": "^5.0.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": "*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/util-deprecate": { | ||||
|       "version": "1.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", | ||||
|       "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", | ||||
|       "license": "MIT" | ||||
|     }, | ||||
|     "node_modules/uuid": { | ||||
|       "version": "9.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", | ||||
|       "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==", | ||||
|       "funding": [ | ||||
|         "https://github.com/sponsors/broofa", | ||||
|         "https://github.com/sponsors/ctavan" | ||||
|       ], | ||||
|       "license": "MIT", | ||||
|       "bin": { | ||||
|         "uuid": "dist/bin/uuid" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/web-streams-polyfill": { | ||||
|       "version": "3.3.3", | ||||
|       "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz", | ||||
|       "integrity": "sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==", | ||||
|       "license": "MIT", | ||||
|       "engines": { | ||||
|         "node": ">= 8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/wrap-ansi": { | ||||
|       "version": "7.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", | ||||
|       "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "ansi-styles": "^4.0.0", | ||||
|         "string-width": "^4.1.0", | ||||
|         "strip-ansi": "^6.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/chalk/wrap-ansi?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/wrappy": { | ||||
|       "version": "1.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", | ||||
|       "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", | ||||
|       "license": "ISC" | ||||
|     }, | ||||
|     "node_modules/y18n": { | ||||
|       "version": "5.0.8", | ||||
|       "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", | ||||
|       "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", | ||||
|       "license": "ISC", | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/yargs": { | ||||
|       "version": "17.7.2", | ||||
|       "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", | ||||
|       "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", | ||||
|       "license": "MIT", | ||||
|       "dependencies": { | ||||
|         "cliui": "^8.0.1", | ||||
|         "escalade": "^3.1.1", | ||||
|         "get-caller-file": "^2.0.5", | ||||
|         "require-directory": "^2.1.1", | ||||
|         "string-width": "^4.2.3", | ||||
|         "y18n": "^5.0.5", | ||||
|         "yargs-parser": "^21.1.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=12" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/yargs-parser": { | ||||
|       "version": "21.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", | ||||
|       "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", | ||||
|       "license": "ISC", | ||||
|       "engines": { | ||||
|         "node": ">=12" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										20
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| { | ||||
|   "name": "actual-cli-push", | ||||
|   "version": "1.0.0", | ||||
|   "bin": { | ||||
|     "actual-cli-push": "./cli.js" | ||||
|   }, | ||||
|   "description": "", | ||||
|   "main": "cli.js", | ||||
|   "scripts": { | ||||
|     "test": "echo \"Error: no test specified\" && exit 1" | ||||
|   }, | ||||
|   "keywords": [], | ||||
|   "author": "", | ||||
|   "license": "ISC", | ||||
|   "type": "commonjs", | ||||
|   "dependencies": { | ||||
|     "@actual-app/api": "^25.4.0", | ||||
|     "yargs": "^17.7.2" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										54
									
								
								parsers.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								parsers.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| from decimal import Decimal | ||||
| from email.message import EmailMessage | ||||
| from model import Transaction | ||||
| import re | ||||
| from uuid import UUID | ||||
| from datetime import datetime | ||||
| from logging import debug | ||||
| from abc import ABC, abstractmethod | ||||
|  | ||||
|  | ||||
| class TransactionParser(ABC): | ||||
|     @abstractmethod | ||||
|     def match(self, msg: EmailMessage) -> bool: | ||||
|         pass | ||||
|  | ||||
|     @abstractmethod | ||||
|     def extract(self, msg: EmailMessage) -> Transaction: | ||||
|         pass | ||||
|  | ||||
|  | ||||
| class TransactionParsingFailed(Exception): | ||||
|     pass | ||||
|  | ||||
|  | ||||
| class RogersBankParser(TransactionParser): | ||||
|     EXTRACT_RE = re.compile( | ||||
|         r"Attempt of \$(\d+\.\d{2}) was made on ([A-z]{3} \d{1,2}, \d{4})[^<]*at ([^<]+) in" | ||||
|     ) | ||||
|  | ||||
|     def __init__(self, account_id: UUID): | ||||
|         self.account_id = account_id | ||||
|  | ||||
|     def match(self, msg: EmailMessage) -> bool: | ||||
|         return ( | ||||
|             msg["From"] == "Rogers Bank <onlineservices@RogersBank.com>" | ||||
|             and msg["Subject"] == "Purchase amount alert" | ||||
|         ) | ||||
|  | ||||
|     def extract(self, msg: EmailMessage) -> Transaction: | ||||
|         matches = RogersBankParser.EXTRACT_RE.search(msg.get_body().as_string()) | ||||
|         if matches is None: | ||||
|             return TransactionParsingFailed("No matches for extraction RE") | ||||
|         amount = Decimal(matches[1]) | ||||
|         date_raw = matches[2] | ||||
|         payee = matches[3] | ||||
|         date = datetime.strptime(date_raw, "%b %d, %Y").date() | ||||
|         return Transaction( | ||||
|             account=self.account_id, | ||||
|             date=date, | ||||
|             amount=amount, | ||||
|             payee=payee, | ||||
|             notes="via email", | ||||
|             imported_id=msg["Message-ID"], | ||||
|         ) | ||||
							
								
								
									
										96
									
								
								watcher.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								watcher.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| from email.message import EmailMessage | ||||
| import logging | ||||
| import asyncio, ssl, email | ||||
| from os import getenv | ||||
| from imaplib import IMAP4 | ||||
| from logging import info, debug, error, warning | ||||
| from pprint import pprint | ||||
| from model import Transaction | ||||
| from config import PARSERS | ||||
| from parsers import TransactionParsingFailed | ||||
| import email.policy | ||||
| import os | ||||
|  | ||||
| IMAP_SERVER = getenv("IMAP_SERVER") | ||||
| IMAP_PORT = int(getenv("IMAP_PORT", 143)) | ||||
| IMAP_USER = getenv("IMAP_USER") | ||||
| IMAP_PASS = getenv("IMAP_PASS") | ||||
| IMAP_MAILBOX = getenv("IMAP_MAILBOX", "INBOX") | ||||
| IMAP_INTERVAL = float(getenv("IMAP_INTERVAL", 300)) | ||||
|  | ||||
| ACTUAL_PATH = "./cli.js" | ||||
|  | ||||
| TIMEOUT = 30 | ||||
|  | ||||
|  | ||||
| async def ticker(interval: float): | ||||
|     while True: | ||||
|         yield | ||||
|         await asyncio.sleep(interval) | ||||
|  | ||||
|  | ||||
| async def submit_transaction(t: Transaction): | ||||
|     cmd=ACTUAL_PATH + f' -a "{t.account}"' + f' -p "{t.payee}"' + f' -m "{t.amount}"' + f' -d "{t.date}"' f' -n "{t.notes}"' | ||||
|     debug("Actual command: %s", cmd) | ||||
|     proc = await asyncio.create_subprocess_shell( | ||||
|         cmd=cmd, | ||||
|         env=os.environ, | ||||
|         stderr=asyncio.subprocess.STDOUT, | ||||
|         stdout=asyncio.subprocess.PIPE, | ||||
|     ) | ||||
|     stdout, stderr = await proc.communicate() | ||||
|     if proc.returncode != 0: | ||||
|         error("Submitting to actual failed: %s", stdout) | ||||
|  | ||||
|  | ||||
| async def process_message(msg_b: bytes): | ||||
|     debug("parsing message") | ||||
|     msg = email.message_from_bytes(msg_b, policy=email.policy.default) | ||||
|     pprint(msg) | ||||
|     info( | ||||
|         "Found message from %s to %s subject %s", | ||||
|         msg.get("From", "<unknown>"), | ||||
|         msg.get("To", "<unknown>"), | ||||
|         msg.get("Subject", "<no subject>"), | ||||
|     ) | ||||
|  | ||||
|     for parser in PARSERS: | ||||
|         debug("Running parser %s", type(parser).__name__) | ||||
|         try: | ||||
|             if parser.match(msg): | ||||
|                 info("Parser %s claimed message", type(parser).__name__) | ||||
|                 trans = parser.extract(msg) | ||||
|                 info("Submitting transaction to Actual: %s", trans) | ||||
|                 await submit_transaction(trans) | ||||
|         except TransactionParsingFailed: | ||||
|             warning("Unable to parse message") | ||||
|         except Exception as e: | ||||
|             warning("Unexpected exception %s", e) | ||||
|  | ||||
|  | ||||
| async def poll_imap(): | ||||
|     info("polling mailbox") | ||||
|     with IMAP4(IMAP_SERVER, IMAP_PORT, 30) as M: | ||||
|         context = ssl.create_default_context() | ||||
|         M.starttls(context) | ||||
|         M.login(IMAP_USER, IMAP_PASS) | ||||
|         M.select(IMAP_MAILBOX) | ||||
|         status, m_set = M.search(None, "UNSEEN") | ||||
|  | ||||
|         for msg_id in m_set[0].split(): | ||||
|             debug("Retrieving msg id %s", msg_id) | ||||
|             status, msg = M.fetch(msg_id, "(RFC822)") | ||||
|             await process_message(msg[0][1]) | ||||
|  | ||||
|  | ||||
| async def app(): | ||||
|     async for tick in ticker(IMAP_INTERVAL): | ||||
|         await poll_imap() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     logging.basicConfig(level=getenv("LOG_LEVEL", "INFO")) | ||||
|     debug("Parsers: %s", PARSERS) | ||||
|     asyncio.run(app()) | ||||
		Reference in New Issue
	
	Block a user