diff --git a/indiekit.config.js b/indiekit.config.js index 6e740a37a..a90fc0f05 100644 --- a/indiekit.config.js +++ b/indiekit.config.js @@ -26,6 +26,7 @@ const config = { "@indiekit/post-type-video", "@indiekit/preset-eleventy", "@indiekit/store-github", + "@indiekit/syndicator-atproto", "@indiekit/syndicator-internet-archive", "@indiekit/syndicator-mastodon", ], @@ -100,6 +101,13 @@ const config = { endpoint: process.env.S3_ENDPOINT, bucket: process.env.S3_BUCKET, }, + "@indiekit/syndicator-atproto": { + checked: true, + profileUrl: "https://bsky.app/profile", + serviceUrl: "https://bsky.social", + user: process.env.ATPROTO_USER, + password: process.env.ATPROTO_PASSWORD, + }, "@indiekit/syndicator-internet-archive": { checked: false, accessKey: process.env.INTERNET_ARCHIVE_ACCESS_KEY, diff --git a/package-lock.json b/package-lock.json index 191e1dfd1..7b1752365 100644 --- a/package-lock.json +++ b/package-lock.json @@ -383,6 +383,63 @@ "node": ">= 14.0.0" } }, + "node_modules/@atproto/api": { + "version": "0.13.23", + "resolved": "https://registry.npmjs.org/@atproto/api/-/api-0.13.23.tgz", + "integrity": "sha512-V1Z5kgfSsqlFaC14sjnZL1Psv/9Lq/YKW1w7TIBq948Rtq8l+c6BpGrOH2Ssdcphpqi4OSeSYRsmJJlD6GGJ5w==", + "license": "MIT", + "dependencies": { + "@atproto/common-web": "^0.3.1", + "@atproto/lexicon": "^0.4.4", + "@atproto/syntax": "^0.3.1", + "@atproto/xrpc": "^0.6.5", + "await-lock": "^2.2.2", + "multiformats": "^9.9.0", + "tlds": "^1.234.0", + "zod": "^3.23.8" + } + }, + "node_modules/@atproto/common-web": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@atproto/common-web/-/common-web-0.3.1.tgz", + "integrity": "sha512-N7wiTnus5vAr+lT//0y8m/FaHHLJ9LpGuEwkwDAeV3LCiPif4m/FS8x/QOYrx1PdZQwKso95RAPzCGWQBH5j6Q==", + "license": "MIT", + "dependencies": { + "graphemer": "^1.4.0", + "multiformats": "^9.9.0", + "uint8arrays": "3.0.0", + "zod": "^3.23.8" + } + }, + "node_modules/@atproto/lexicon": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/@atproto/lexicon/-/lexicon-0.4.4.tgz", + "integrity": "sha512-QFEmr3rpj/RoAmfX9ALU/asBG/rsVtQZnw+9nOB1/AuIwoxXd+ZyndR6lVUc2+DL4GEjl6W2yvBru5xbQIZWyA==", + "license": "MIT", + "dependencies": { + "@atproto/common-web": "^0.3.1", + "@atproto/syntax": "^0.3.1", + "iso-datestring-validator": "^2.2.2", + "multiformats": "^9.9.0", + "zod": "^3.23.8" + } + }, + "node_modules/@atproto/syntax": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@atproto/syntax/-/syntax-0.3.1.tgz", + "integrity": "sha512-fzW0Mg1QUOVCWUD3RgEsDt6d1OZ6DdFmbKcDdbzUfh0t4rhtRAC05KbZYmxuMPWDAiJ4BbbQ5dkAc/mNypMXkw==", + "license": "MIT" + }, + "node_modules/@atproto/xrpc": { + "version": "0.6.5", + "resolved": "https://registry.npmjs.org/@atproto/xrpc/-/xrpc-0.6.5.tgz", + "integrity": "sha512-t6u8iPEVbWge5RhzKZDahSzNDYIAxUtop6Q/X/apAZY1rgreVU0/1sSvvRoRFH19d3UIKjYdLuwFqMi9w8nY3Q==", + "license": "MIT", + "dependencies": { + "@atproto/lexicon": "^0.4.4", + "zod": "^3.23.8" + } + }, "node_modules/@aws-crypto/crc32": { "version": "5.2.0", "resolved": "https://registry.npmjs.org/@aws-crypto/crc32/-/crc32-5.2.0.tgz", @@ -2575,28 +2632,6 @@ "@img/sharp-libvips-darwin-arm64": "1.0.4" } }, - "node_modules/@img/sharp-darwin-x64": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-darwin-x64/-/sharp-darwin-x64-0.33.5.tgz", - "integrity": "sha512-fyHac4jIc1ANYGRDxtiqelIbdWkIuQaI84Mv45KvGRRxSAa7o7d1ZKAOBaYbnepLC1WqxfpimdeWfvqqSGwR2Q==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "darwin" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-darwin-x64": "1.0.4" - } - }, "node_modules/@img/sharp-libvips-darwin-arm64": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/@img/sharp-libvips-darwin-arm64/-/sharp-libvips-darwin-arm64-1.0.4.tgz", @@ -2613,307 +2648,6 @@ "url": "https://opencollective.com/libvips" } }, - "node_modules/@img/sharp-libvips-darwin-x64": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-darwin-x64/-/sharp-libvips-darwin-x64-1.0.4.tgz", - "integrity": "sha512-xnGR8YuZYfJGmWPvmlunFaWJsb9T/AO2ykoP3Fz/0X5XV2aoYBPkX6xqCQvUTKKiLddarLaxpzNe+b1hjeWHAQ==", - "cpu": [ - "x64" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "darwin" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-libvips-linux-arm": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-arm/-/sharp-libvips-linux-arm-1.0.5.tgz", - "integrity": "sha512-gvcC4ACAOPRNATg/ov8/MnbxFDJqf/pDePbBnuBDcjsI8PssmjoKMAz4LtLaVi+OnSb5FK/yIOamqDwGmXW32g==", - "cpu": [ - "arm" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "linux" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-libvips-linux-arm64": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-arm64/-/sharp-libvips-linux-arm64-1.0.4.tgz", - "integrity": "sha512-9B+taZ8DlyyqzZQnoeIvDVR/2F4EbMepXMc/NdVbkzsJbzkUjhXv/70GQJ7tdLA4YJgNP25zukcxpX2/SueNrA==", - "cpu": [ - "arm64" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "linux" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-libvips-linux-s390x": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-s390x/-/sharp-libvips-linux-s390x-1.0.4.tgz", - "integrity": "sha512-u7Wz6ntiSSgGSGcjZ55im6uvTrOxSIS8/dgoVMoiGE9I6JAfU50yH5BoDlYA1tcuGS7g/QNtetJnxA6QEsCVTA==", - "cpu": [ - "s390x" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "linux" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-libvips-linux-x64": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linux-x64/-/sharp-libvips-linux-x64-1.0.4.tgz", - "integrity": "sha512-MmWmQ3iPFZr0Iev+BAgVMb3ZyC4KeFc3jFxnNbEPas60e1cIfevbtuyf9nDGIzOaW9PdnDciJm+wFFaTlj5xYw==", - "cpu": [ - "x64" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "linux" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-libvips-linuxmusl-arm64": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linuxmusl-arm64/-/sharp-libvips-linuxmusl-arm64-1.0.4.tgz", - "integrity": "sha512-9Ti+BbTYDcsbp4wfYib8Ctm1ilkugkA/uscUn6UXK1ldpC1JjiXbLfFZtRlBhjPZ5o1NCLiDbg8fhUPKStHoTA==", - "cpu": [ - "arm64" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "linux" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-libvips-linuxmusl-x64": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@img/sharp-libvips-linuxmusl-x64/-/sharp-libvips-linuxmusl-x64-1.0.4.tgz", - "integrity": "sha512-viYN1KX9m+/hGkJtvYYp+CCLgnJXwiQB39damAO7WMdKWlIhmYTfHjwSbQeUK/20vY154mwezd9HflVFM1wVSw==", - "cpu": [ - "x64" - ], - "license": "LGPL-3.0-or-later", - "optional": true, - "os": [ - "linux" - ], - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-linux-arm": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-linux-arm/-/sharp-linux-arm-0.33.5.tgz", - "integrity": "sha512-JTS1eldqZbJxjvKaAkxhZmBqPRGmxgu+qFKSInv8moZ2AmT5Yib3EQ1c6gp493HvrvV8QgdOXdyaIBrhvFhBMQ==", - "cpu": [ - "arm" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-linux-arm": "1.0.5" - } - }, - "node_modules/@img/sharp-linux-arm64": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-linux-arm64/-/sharp-linux-arm64-0.33.5.tgz", - "integrity": "sha512-JMVv+AMRyGOHtO1RFBiJy/MBsgz0x4AWrT6QoEVVTyh1E39TrCUpTRI7mx9VksGX4awWASxqCYLCV4wBZHAYxA==", - "cpu": [ - "arm64" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-linux-arm64": "1.0.4" - } - }, - "node_modules/@img/sharp-linux-s390x": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-linux-s390x/-/sharp-linux-s390x-0.33.5.tgz", - "integrity": "sha512-y/5PCd+mP4CA/sPDKl2961b+C9d+vPAveS33s6Z3zfASk2j5upL6fXVPZi7ztePZ5CuH+1kW8JtvxgbuXHRa4Q==", - "cpu": [ - "s390x" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-linux-s390x": "1.0.4" - } - }, - "node_modules/@img/sharp-linux-x64": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-linux-x64/-/sharp-linux-x64-0.33.5.tgz", - "integrity": "sha512-opC+Ok5pRNAzuvq1AG0ar+1owsu842/Ab+4qvU879ippJBHvyY5n2mxF1izXqkPYlGuP/M556uh53jRLJmzTWA==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-linux-x64": "1.0.4" - } - }, - "node_modules/@img/sharp-linuxmusl-arm64": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-linuxmusl-arm64/-/sharp-linuxmusl-arm64-0.33.5.tgz", - "integrity": "sha512-XrHMZwGQGvJg2V/oRSUfSAfjfPxO+4DkiRh6p2AFjLQztWUuY/o8Mq0eMQVIY7HJ1CDQUJlxGGZRw1a5bqmd1g==", - "cpu": [ - "arm64" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-linuxmusl-arm64": "1.0.4" - } - }, - "node_modules/@img/sharp-linuxmusl-x64": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-linuxmusl-x64/-/sharp-linuxmusl-x64-0.33.5.tgz", - "integrity": "sha512-WT+d/cgqKkkKySYmqoZ8y3pxx7lx9vVejxW/W4DOFMYVSkErR+w7mf2u8m/y4+xHe7yY9DAXQMWQhpnMuFfScw==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0", - "optional": true, - "os": [ - "linux" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - }, - "optionalDependencies": { - "@img/sharp-libvips-linuxmusl-x64": "1.0.4" - } - }, - "node_modules/@img/sharp-wasm32": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-wasm32/-/sharp-wasm32-0.33.5.tgz", - "integrity": "sha512-ykUW4LVGaMcU9lu9thv85CbRMAwfeadCJHRsg2GmeRa/cJxsVY9Rbd57JcMxBkKHag5U/x7TSBpScF4U8ElVzg==", - "cpu": [ - "wasm32" - ], - "license": "Apache-2.0 AND LGPL-3.0-or-later AND MIT", - "optional": true, - "dependencies": { - "@emnapi/runtime": "^1.2.0" - }, - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-win32-ia32": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-win32-ia32/-/sharp-win32-ia32-0.33.5.tgz", - "integrity": "sha512-T36PblLaTwuVJ/zw/LaH0PdZkRz5rd3SmMHX8GSmR7vtNSP5Z6bQkExdSK7xGWyxLw4sUknBuugTelgw2faBbQ==", - "cpu": [ - "ia32" - ], - "license": "Apache-2.0 AND LGPL-3.0-or-later", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - } - }, - "node_modules/@img/sharp-win32-x64": { - "version": "0.33.5", - "resolved": "https://registry.npmjs.org/@img/sharp-win32-x64/-/sharp-win32-x64-0.33.5.tgz", - "integrity": "sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg==", - "cpu": [ - "x64" - ], - "license": "Apache-2.0 AND LGPL-3.0-or-later", - "optional": true, - "os": [ - "win32" - ], - "engines": { - "node": "^18.17.0 || ^20.3.0 || >=21.0.0" - }, - "funding": { - "url": "https://opencollective.com/libvips" - } - }, "node_modules/@indiekit-test/config": { "resolved": "helpers/config", "link": true @@ -3102,6 +2836,10 @@ "resolved": "packages/store-s3", "link": true }, + "node_modules/@indiekit/syndicator-atproto": { + "resolved": "packages/syndicator-atproto", + "link": true + }, "node_modules/@indiekit/syndicator-internet-archive": { "resolved": "packages/syndicator-internet-archive", "link": true @@ -7279,6 +7017,12 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/await-lock": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/await-lock/-/await-lock-2.2.2.tgz", + "integrity": "sha512-aDczADvlvTGajTDjcjpJMqRkOF6Qdz3YbPZm/PyW6tKPkx2hlYBzxMhEywM/tU72HrVZjgl5VCdRuMlA7pZ8Gw==", + "license": "MIT" + }, "node_modules/aws-sdk-client-mock": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/aws-sdk-client-mock/-/aws-sdk-client-mock-4.1.0.tgz", @@ -12696,6 +12440,12 @@ "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", "license": "ISC" }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "license": "MIT" + }, "node_modules/h3": { "version": "1.13.0", "resolved": "https://registry.npmjs.org/h3/-/h3-1.13.0.tgz", @@ -14142,6 +13892,12 @@ "node": ">=6.0" } }, + "node_modules/iso-datestring-validator": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/iso-datestring-validator/-/iso-datestring-validator-2.2.2.tgz", + "integrity": "sha512-yLEMkBbLZTlVQqOnQ4FiMujR6T4DEcCb1xizmvXS+OxuhwcbtynoosRzdMA69zZCShCNAbi+gJ71FxZBBXx1SA==", + "license": "MIT" + }, "node_modules/isobject": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", @@ -16768,6 +16524,12 @@ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", "license": "MIT" }, + "node_modules/multiformats": { + "version": "9.9.0", + "resolved": "https://registry.npmjs.org/multiformats/-/multiformats-9.9.0.tgz", + "integrity": "sha512-HoMUjhH9T8DDBNT+6xzkrd9ga/XiBI4xLr58LJACwK6G3HTOPeMz4nB4KJs33L2BelrIJa7P0VuNaVF3hMYfjg==", + "license": "(Apache-2.0 AND MIT)" + }, "node_modules/multimatch": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/multimatch/-/multimatch-5.0.0.tgz", @@ -19787,8 +19549,6 @@ }, "node_modules/sharp": { "version": "0.33.5", - "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.33.5.tgz", - "integrity": "sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw==", "hasInstallScript": true, "license": "Apache-2.0", "dependencies": { @@ -21232,6 +20992,15 @@ "dev": true, "license": "MIT" }, + "node_modules/tlds": { + "version": "1.255.0", + "resolved": "https://registry.npmjs.org/tlds/-/tlds-1.255.0.tgz", + "integrity": "sha512-tcwMRIioTcF/FcxLev8MJWxCp+GUALRhFEqbDoZrnowmKSGqPrl5pqS+Sut2m8BgJ6S4FExCSSpGffZ0Tks6Aw==", + "license": "MIT", + "bin": { + "tlds": "bin.js" + } + }, "node_modules/tldts": { "version": "6.1.70", "resolved": "https://registry.npmjs.org/tldts/-/tldts-6.1.70.tgz", @@ -21636,6 +21405,15 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/uint8arrays": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/uint8arrays/-/uint8arrays-3.0.0.tgz", + "integrity": "sha512-HRCx0q6O9Bfbp+HHSfQQKD7wU70+lydKVt4EghkdOvlK/NlrF90z+eXV34mUd48rNvVJXwkrMSPpCATkct8fJA==", + "license": "MIT", + "dependencies": { + "multiformats": "^9.4.2" + } + }, "node_modules/unbox-primitive": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", @@ -22801,6 +22579,15 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/zod": { + "version": "3.24.1", + "resolved": "https://registry.npmjs.org/zod/-/zod-3.24.1.tgz", + "integrity": "sha512-muH7gBL9sI1nciMZV67X5fTKKBLtwpZ5VBp1vsOQzj1MhrBZ4wlVCm3gedKZWLp0Oyel8sIGfeiz54Su+OVT+A==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } + }, "node_modules/zwitch": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/zwitch/-/zwitch-2.0.4.tgz", @@ -24169,6 +23956,21 @@ "node": ">=20" } }, + "packages/syndicator-atproto": { + "name": "@indiekit/syndicator-atproto", + "version": "1.0.0-beta.16", + "license": "MIT", + "dependencies": { + "@atproto/api": "^0.13.23", + "@indiekit/error": "^1.0.0-beta.15", + "@indiekit/util": "^1.0.0-beta.16", + "brevity": "^0.2.9", + "html-to-text": "^9.0.0" + }, + "engines": { + "node": ">=20" + } + }, "packages/syndicator-internet-archive": { "name": "@indiekit/syndicator-internet-archive", "version": "1.0.0-beta.15", diff --git a/packages/syndicator-atproto/README.md b/packages/syndicator-atproto/README.md new file mode 100644 index 000000000..487601fff --- /dev/null +++ b/packages/syndicator-atproto/README.md @@ -0,0 +1,35 @@ +# @indiekit/syndicator-atproto + +[AT Protocol](https://atproto.com) syndicator for Indiekit. + +## Installation + +`npm i @indiekit/syndicator-atproto` + +## Usage + +Add `@indiekit/syndicator-atproto` to your list of plug-ins, specifying options as required: + +```json +{ + "plugins": ["@indiekit/syndicator-atproto"], + "@indiekit/syndicator-atproto": { + "profileUrl": "https://bsky.app/profile", + "serviceUrl": "https://bsky.social", + "user": "username.bsky.social", + "password": "password", + "checked": true + } +} +``` + +## Options + +| Option | Type | Description | +| :----------------- | :-------- | :-------------------------------------------------------------------------------------------------------------- | +| `password` | `string` | Your AT protocol password. _Required_, defaults to `process.env.ATPROTO_PASSWORD`. | +| `profileUrl` | `string` | Your AT protocol profile, i.e. `https://bsky.app/profile`. Used in the URL returned in syndicated URLs. | +| `serviceUrl` | `string` | Your AT protocol service, i.e. `https://bsky.social`. Used to connect to your AT Proto service API. _Required_. | +| `user` | `string` | Your AT protocol identifier (without the `@`). _Required_. | +| `checked` | `boolean` | Tell a Micropub client whether this syndicator should be enabled by default. _Optional_, defaults to `false`. | +| `includePermalink` | `boolean` | Always include a link to the original post. _Optional_, defaults to `false`. | diff --git a/packages/syndicator-atproto/assets/icon.svg b/packages/syndicator-atproto/assets/icon.svg new file mode 100644 index 000000000..8ed5e1d70 --- /dev/null +++ b/packages/syndicator-atproto/assets/icon.svg @@ -0,0 +1,4 @@ + + + + diff --git a/packages/syndicator-atproto/index.js b/packages/syndicator-atproto/index.js new file mode 100644 index 000000000..dcfddeae3 --- /dev/null +++ b/packages/syndicator-atproto/index.js @@ -0,0 +1,120 @@ +import process from "node:process"; + +import { IndiekitError } from "@indiekit/error"; + +import { atproto } from "./lib/atproto.js"; + +const defaults = { + checked: false, + includePermalink: false, + password: process.env.ATPROTO_PASSWORD, + user: "", +}; + +export default class AtProtoSyndicator { + /** + * @param {object} [options] - Plug-in options + * @param {string} [options.profileUrl] - Profile URL + * @param {string} [options.serviceUrl] - Service URL + * @param {string} [options.user] - Username + * @param {string} [options.password] - Password + * @param {boolean} [options.includePermalink] - Include permalink in status + * @param {boolean} [options.checked] - Check syndicator in UI + */ + constructor(options = {}) { + this.name = "AT Protocol syndicator"; + this.options = { ...defaults, ...options }; + } + + get #profileUrl() { + const userName = this.options.user.replace("@", ""); + return this.options?.profileUrl + ? new URL(this.options.profileUrl).href + "/" + userName + : false; + } + + get #serviceUrl() { + return this.options?.serviceUrl + ? new URL(this.options.serviceUrl).href + : false; + } + + get #user() { + return this.options?.user + ? `@${this.options.user.replace("@", "")}` + : false; + } + + get environment() { + return ["ATPROTO_PASSWORD"]; + } + + get info() { + const service = { + name: "AT Protocol", + photo: "/assets/@indiekit-atproto/icon.svg", + }; + const profileUrl = this.#profileUrl; + const serviceUrl = this.#serviceUrl; + const user = this.#user; + + if (!profileUrl) { + return { + error: "Profile URL required", + service, + }; + } + + if (!serviceUrl) { + return { + error: "Service URL required", + service, + }; + } + + if (!user) { + return { + error: "User identifier required", + service, + }; + } + + const uid = profileUrl; + service.url = serviceUrl; + + const thing = { + checked: this.options.checked, + name: user, + uid, + service, + user: { + name: user, + url: uid, + }, + }; + + return thing; + } + + async syndicate(properties, publication) { + try { + return await atproto({ + identifier: this.options.user, + password: this.options.password, + includePermalink: this.options.includePermalink, + profileUrl: this.#profileUrl && this.#profileUrl, + serviceUrl: this.#serviceUrl && this.#serviceUrl, + }).post(properties, publication.me); + } catch (error) { + throw new IndiekitError(error.message, { + cause: error, + plugin: this.name, + status: error.statusCode, + }); + } + } + + init(Indiekit) { + Indiekit.addSyndicator(this); + } +} diff --git a/packages/syndicator-atproto/lib/atproto.js b/packages/syndicator-atproto/lib/atproto.js new file mode 100644 index 000000000..eaf70c49a --- /dev/null +++ b/packages/syndicator-atproto/lib/atproto.js @@ -0,0 +1,90 @@ +import { AtpAgent, RichText } from "@atproto/api"; +import { isSameOrigin } from "@indiekit/util"; + +import { getStatusText, getPostParts, uriToPostUrl } from "./utils.js"; + +/** + * Syndicate post to an AT Protocol service + * @param {object} options - Syndicator options + * @param {string} [options.profileUrl] - Profile URL + * @param {string} [options.serviceUrl] - Service URL + * @param {string} options.identifier - User identifier + * @param {string} options.password - Password + * @param {boolean} options.includePermalink - Include permalink in status + * @returns {object} Post functions + */ +export const atproto = ({ + profileUrl, + serviceUrl, + identifier, + password, + includePermalink, +}) => ({ + async client() { + const agent = new AtpAgent({ service: serviceUrl }); + + await agent.login({ identifier, password }); + + return agent; + }, + + /** + * Post a like + * @param {string} postUrl - URL of post to like + * @returns {Promise} Mastodon status URL + */ + async postLike(postUrl) { + const client = await this.client(); + + const post = getPostParts(postUrl); + const { uri, cid } = await client.getPost({ + repo: post.did, + rkey: post.rkey, + }); + + const like = await client.like(uri, cid); + + return uriToPostUrl(profileUrl, like.uri); + }, + + /** + * Post to AT Protocol + * @param {object} properties - JF2 properties + * @returns {Promise} URL of syndicated status + */ + async post(properties) { + const client = await this.client(); + + if (properties["like-of"]) { + // Syndicate like of AT Proto URL as a like + if (isSameOrigin(properties["like-of"], profileUrl)) { + return this.postLike(properties["like-of"]); + } + + // Do not syndicate likes of other URLs + return false; + } + + const text = getStatusText(properties, { + includePermalink, + serviceUrl, + }); + + const rt = new RichText({ text }); + await rt.detectFacets(client); + + try { + const post = await client.post({ + $type: "app.bsky.feed.post", + text: rt.text, + facets: rt.facets, + createdAt: new Date().toISOString(), + }); + + return uriToPostUrl(profileUrl, post.uri); + } catch (error) { + const message = error.message; + throw new Error(message); + } + }, +}); diff --git a/packages/syndicator-atproto/lib/utils.js b/packages/syndicator-atproto/lib/utils.js new file mode 100644 index 000000000..eff0a8277 --- /dev/null +++ b/packages/syndicator-atproto/lib/utils.js @@ -0,0 +1,105 @@ +import brevity from "brevity"; +import { htmlToText } from "html-to-text"; + +/** + * Get post parts (UID and CID) + * @param {string} url - Post URL + * @returns {object} Parts + */ +export const getPostParts = (url) => { + const pathParts = new URL(url).pathname.split("/"); + + // Extract DID and post ID from the path + const did = pathParts[2]; + const rkey = pathParts[4]; + + return { did, rkey }; +}; + +/** + * Covert AT Protocol URI to post URL + * @param {string} profileUrl - Profile URL + * @param {string} uri - AT Proto URI + * @returns {string} Post URL + */ +export const uriToPostUrl = (profileUrl, uri) => { + const [, did, , rkey] = uri.split("/"); + return `${profileUrl}/${did}/post/${rkey}`; +}; + +/** + * Get status parameters from given JF2 properties + * @param {object} properties - JF2 properties + * @param {object} [options] - Options + * @param {boolean} [options.includePermalink] - Include permalink in status + * @param {string} [options.serviceUrl] - Service URL + * @returns {string} Status text + */ +export const getStatusText = (properties, options = {}) => { + const { includePermalink, serviceUrl } = options; + + let text; + if (properties.content && properties.content.html) { + text = htmlToStatusText(properties.content.html, serviceUrl); + } + + // Truncate status if longer than 300 characters + text = brevity.shorten( + text, + properties.url, + includePermalink // https://indieweb.org/permashortlink + ? properties.url + : false, + false, // https://indieweb.org/permashortcitation + 300, + ); + + // Show permalink below status, not within brackets + text = text.replace(`(${properties.url})`, `\n\n${properties.url}`); + + return text; +}; + +/** + * Convert HTML to plain text, appending last link href if present + * @param {string} html - HTML + * @param {string} serviceUrl - Service URL, i.e. https://bsky.social + * @returns {string} Text + */ +export const htmlToStatusText = (html, serviceUrl) => { + // Get all the link references + let hrefs = [...html.matchAll(/href="(https?:\/\/.+?)"/g)]; + + // Remove any links to Mastodon server + // HTML may contain Mastodon usernames or hashtag links + hrefs = hrefs.filter((href) => { + const hrefHostname = new URL(href[1]).hostname; + const serverHostname = new URL(serviceUrl).hostname; + return hrefHostname !== serverHostname; + }); + + // Get the last link mentioned, or return false + const lastHref = hrefs.length > 0 ? hrefs.at(-1)[1] : false; + + // Convert HTML to plain text, removing any links + const text = htmlToText(html, { + selectors: [ + { + selector: "a", + options: { + ignoreHref: true, + }, + }, + { + selector: "img", + format: "skip", + }, + ], + wordwrap: false, + }); + + // Append the last link if present + const statusText = lastHref ? `${text} ${lastHref}` : text; + + return statusText; +}; diff --git a/packages/syndicator-atproto/package.json b/packages/syndicator-atproto/package.json new file mode 100644 index 000000000..ffc104a36 --- /dev/null +++ b/packages/syndicator-atproto/package.json @@ -0,0 +1,47 @@ +{ + "name": "@indiekit/syndicator-atproto", + "version": "1.0.0-beta.16", + "description": "AT Protocol syndicator for Indiekit", + "keywords": [ + "indiekit", + "indiekit-plugin", + "indieweb", + "syndication", + "at-protocol", + "atproto" + ], + "homepage": "https://getindiekit.com", + "author": { + "name": "Paul Robert Lloyd", + "url": "https://paulrobertlloyd.com" + }, + "license": "MIT", + "engines": { + "node": ">=20" + }, + "type": "module", + "main": "index.js", + "files": [ + "assets", + "lib", + "index.js" + ], + "bugs": { + "url": "https://github.com/getindiekit/indiekit/issues" + }, + "repository": { + "type": "git", + "url": "https://github.com/getindiekit/indiekit.git", + "directory": "packages/syndicator-atproto" + }, + "dependencies": { + "@atproto/api": "^0.13.23", + "@indiekit/error": "^1.0.0-beta.15", + "@indiekit/util": "^1.0.0-beta.16", + "brevity": "^0.2.9", + "html-to-text": "^9.0.0" + }, + "publishConfig": { + "access": "public" + } +} diff --git a/packages/syndicator-atproto/test/index.js b/packages/syndicator-atproto/test/index.js new file mode 100644 index 000000000..cd9efd33c --- /dev/null +++ b/packages/syndicator-atproto/test/index.js @@ -0,0 +1,56 @@ +import { strict as assert } from "node:assert"; +import { describe, it } from "node:test"; +import { Indiekit } from "@indiekit/indiekit"; +import AtProtoSyndicator from "../index.js"; + +describe("syndicator-atproto", () => { + const atproto = new AtProtoSyndicator({ + password: "password", + url: "https://butterfly.example", + user: "username.butterfly.example", + }); + + it("Gets plug-in environment", () => { + assert.deepEqual(atproto.environment, ["ATPROTO_PASSWORD"]); + }); + + it("Gets plug-in info", () => { + assert.equal(atproto.name, "AT Protocol syndicator"); + assert.equal(atproto.info.checked, false); + assert.equal(atproto.info.name, "@username.butterfly.example"); + assert.equal(atproto.info.uid, "https://username.butterfly.example"); + assert.ok(atproto.info.service); + }); + + it("Returns error information if no server URL provided", async () => { + const result = new AtProtoSyndicator({ + password: "password", + user: "username", + }); + + assert.equal(result.info.error, "Service URL required"); + }); + + it("Returns error information if no username provided", () => { + const result = new AtProtoSyndicator({ + password: "password", + url: "https://username.butterfly.example", + }); + + assert.equal(result.info.error, "User identifier required"); + }); + + it("Initiates plug-in", async () => { + const indiekit = await Indiekit.initialize({ config: {} }); + atproto.init(indiekit); + + assert.equal( + indiekit.publication.syndicationTargets[0].info.name, + "@username.butterfly.example", + ); + }); + + it.todo("Returns syndicated URL"); + + it.todo("Throws error getting syndicated URL if access token invalid"); +});