Files
indiekit-blog/_data/blueskyFeed.js

75 lines
2.1 KiB
JavaScript

/**
* Bluesky Feed Data
* Fetches recent posts from Bluesky using the AT Protocol API
*/
import EleventyFetch from "@11ty/eleventy-fetch";
export default async function () {
const rawHandle = (process.env.BLUESKY_HANDLE || "")
.trim()
.replace(/^@+/, "");
const handle =
rawHandle && !rawHandle.includes(".") && !rawHandle.startsWith("did:")
? `${rawHandle}.bsky.social`
: rawHandle;
if (!handle) {
return [];
}
try {
// Get the author's feed using public API (no auth needed for public posts)
const feedUrl = `https://public.api.bsky.app/xrpc/app.bsky.feed.getAuthorFeed?actor=${handle}&limit=10`;
const response = await EleventyFetch(feedUrl, {
duration: "15m", // Cache for 15 minutes
type: "json",
fetchOptions: {
headers: {
Accept: "application/json",
},
},
});
if (!response.feed) {
console.log("No Bluesky feed found for handle:", handle);
return [];
}
// Transform the feed into a simpler format
return response.feed.map((item) => {
// Extract rkey from AT URI (at://did:plc:xxx/app.bsky.feed.post/rkey)
const rkey = item.post.uri.split("/").pop();
const postUrl = `https://bsky.app/profile/${item.post.author.handle}/post/${rkey}`;
return {
text: item.post.record.text,
createdAt: item.post.record.createdAt,
uri: item.post.uri,
url: postUrl,
cid: item.post.cid,
author: {
handle: item.post.author.handle,
displayName: item.post.author.displayName,
avatar: item.post.author.avatar,
},
likeCount: item.post.likeCount || 0,
repostCount: item.post.repostCount || 0,
replyCount: item.post.replyCount || 0,
// Extract any embedded links or images
embed: item.post.embed
? {
type: item.post.embed.$type,
images: item.post.embed.images || [],
external: item.post.embed.external || null,
}
: null,
};
});
} catch (error) {
console.error("Error fetching Bluesky feed:", error.message);
return [];
}
}