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");
+});