Comments are disabled on DRAFT pages.
:Comments are disabled on DRAFT pages.
+ ) : ( +diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000..70d3ce8 Binary files /dev/null and b/public/favicon.ico differ diff --git a/public/favicon.png b/public/favicon.png deleted file mode 100644 index 156231d..0000000 Binary files a/public/favicon.png and /dev/null differ diff --git a/public/icon-192.png b/public/icon-192.png new file mode 100644 index 0000000..7639ac3 Binary files /dev/null and b/public/icon-192.png differ diff --git a/public/icon-512.png b/public/icon-512.png new file mode 100644 index 0000000..064292b Binary files /dev/null and b/public/icon-512.png differ diff --git a/public/manifest.webmanifest b/public/manifest.webmanifest new file mode 100644 index 0000000..98bd2ba --- /dev/null +++ b/public/manifest.webmanifest @@ -0,0 +1,6 @@ +{ + "icons": [ + {"src": "/icon-192.png", "type": "image/png", "sizes": "192x192"}, + {"src": "/icon-512.png", "type": "image/png", "sizes": "512x512"} + ] +} \ No newline at end of file diff --git a/scripts/puppeteer.ts b/scripts/puppeteer.ts index 3b17437..2fa7418 100644 --- a/scripts/puppeteer.ts +++ b/scripts/puppeteer.ts @@ -44,8 +44,12 @@ async function handleUrl(browser: Browser, url: string) { const pathname = new URL(url).pathname; console.time(pathname); - const path1 = await snap(browser, url, "base"); - const path2 = await snap(browser, url.replace(base, compare), "compare"); + const path1 = await snap(browser, `${url}?snap`, "base"); + const path2 = await snap( + browser, + `${url.replace(base, compare)}?snap`, + "compare", + ); const img1 = readScreenshot(path1); const img2 = readScreenshot(path2); const meta1 = await img1.metadata(); diff --git a/src/components/Shadowbox.astro b/src/components/Shadowbox.astro index 3831fbd..88090d9 100644 --- a/src/components/Shadowbox.astro +++ b/src/components/Shadowbox.astro @@ -2,12 +2,14 @@ interface Props { padding?: number; class?: string | undefined; + id?: string | undefined; } -const { padding = 1, class: className } = Astro.props; +const { padding = 1, class: className, id } = Astro.props; ---
- -
---- - This is another X-Treme Nerd Interlude post. -Last time we announced, mercifully briefly, - - our shiny new blog redesign - - and if you’re a normal human you should read that, nod thoughtfully, say “looks lovely”, and be on your merry way. -The rest of you can frolic deep in the weeds here with - - Nathan Arthur - - aka narthur — the mastermind and architect of everything you’re looking at. -Also it’s a nice primer on what the heck a Static Site Generator (SSG) is, in case you were curious about that. - -
-
- WordPress is the de facto tool for building blogs and, as Danny said last time, it served Beeminder well for many years. -But now we’re using a static site generator called - - Astro - - . -Let’s talk about why and how! -
-- We’ll start with the downsides of WordPress. -WordPress is a server-rendered content management system. -That means that whenever you visit a page on a WordPress website, WordPress retrieves that page’s data from a database, renders it to HTML, and sends it to you. -
-- -
-- This architecture is powerful — it allows websites to tailor each page for the current user. -And in WordPress’s case it allows non-technical admins to build and customize their sites, no coding required. -
-- There’s an alternative — the static site. -A static site is simply a set of HTML, CSS, and JavaScript files, served as-is, without waiting for the server to painstakingly construct them before sending them to you. -
-- -
-- It would be impractical to build most websites this way. -Hand coding HTML files for every post on the Beeminder blog would be painful. -And then a change to the footer, say, would require editing every HTML file again. -
-- But what if we did that WordPress-style page construction all at once, one time? -Then we’d have a set of static files. -We call that one-time painstaking construction the build step. -
-- -
-- Whenever you add a new blog post, or change anything in an old blog post, or change anything at all about the website, you trigger a build. -The Static Site Generator does its WordPressing, as we’ll call it, to every page and blog post on the whole site again. -But, crucially, it’s not doing this in real time with eager blog readers waiting for it. -It regenerates the static site and when the new files are ready, it swaps the old ones out and the new ones in. -End users are only ever asking for and getting served static pages. -
-- Basically we have the best of both worlds: let the build process do all the HTML generating, putting the same footer on every page and all that, but let end users get served simple static pages. -It’s also more secure. -WordPress has a whole admin interface to let you add posts and modify the site — a static site needs none of that. -And it’s drastically cheaper (or even free, depending on how much traffic it gets) to host a static site. -And content delivery networks (CDNs) can cache those files closer to end users which adds to the speed advantage. -
-- The Beeminder blog is particularly well-situated to take advantage of all of those benefits. -Its content is primarily static. -The non-static bits (like the comments) can use third-party embeds. -Beeminder’s staff are technical and not code-shy (so no need for everything to be WYSIWYG). -And Beeminder had been spending a lot on blog hosting. -
-- - (Danny adds: Does this mean the migration paid for itself? -Haha, no. -We initially thought it might — we were paying around $1k/year for WordPress hosting — but then decided we didn’t care. -We really like this new system! -And it might eventually pay off even just money-wise, who knows?) - -
-- Alright, enough why, let’s run through the what and how! -We had some specific boxes that we wanted the new blog to tick. -First, we needed to preserve what the Beeminder team liked about their workflow. -That meant pulling in external markdown source files for each post, and being agnostic about where Beeminder may decide to store that markdown. -And builds needed to be fast enough that fixing a typo wouldn’t be excruciating. -
-- We also wanted the blog to be nice and maintainable — easy for future developers to improve the blog. -And of course not increasing Beeminder’s security exposure or other costs. -Like I said, security and cost should both be improved a lot by a static site generator. -
-- In terms of what users see when read the blog, we needed to - - Pareto-dominate - - the old blog. -So of course all the old posts should stay the same, the RSS feed shouldn’t break. -Beeminder also has a bunch of fancy custom markdown augmentations like footnotes and LaTeX-style references that needed to keep functioning. -
-- And as icing on the cake, we wanted the blog to feel nicer for end users — faster and more modern and more in line with Beeminder’s visual identity. -
-- We needed to take our backlog of content and use a static site generator to convert that content into static HTML, CSS, and JavaScript files. -
-- - Many options exist - - . -I didn’t want to use those that I’d used previously. - - Jekyll - - — not my preferred stack. - - Gatsby - - — too unintuitive. - - Next.js - - — too many unneeded features. -Also Brent Yorgey recommended - - Hakyll - - which seems cool but he recommended it too late and it would have been hard since I’m bad at Haskell. -
-- One I hadn’t used previously was - - Astro - - . -I watched - - a video - - of someone building a simple site in Astro, and I was impressed. -It seemed like our use case was exactly Astro’s happy path, which made me feel much more comfortable that I could spend more time executing and less time fighting with the framework. -I also appreciated that I could use my preferred stack — node, pnpm, typescript, vitest. -I liked how the default Astro templating language is JSX-like, and how it co-locates server JS, client JS, HTML, and CSS in each component file. -I also really liked that it comes with folder structure routing out of the box. -And I liked how easy it makes using external data in static generation, just using the standard fetch API. -
-- Astro had everything we wanted. -It has file-based routing with dynamic route segments and simple route-based data fetching. -It’s written in JavaScript and supports TypeScript out of the box. -It uses file-based components. -The Astro templating format is close enough to JSX to make a React developer feel at home, while sidestepping all the complexity that comes from using React. -It’s built on Vite, making for good build performance. -And it has an RSS plugin which made setting up our feed quite simple. -
-- Additionally Astro proved beneficial in several other ways that we didn’t think to look for. -It has built-in link prefetching. -It includes experimental support for image optimization (still working on setting that up!) and static redirects. - -And it has a hot-reloading local dev server. -
-- Danny suggested Render.com. -I hadn’t used it before, but I’m glad we did. -It was a great experience. -I’ve never enjoyed deploying to AWS or Google Cloud. -In contrast, Render.com integrates with GitHub, providing painless git-triggered deployments. -Also blueprints. -
-- Initially I planned to have GitHub Actions run our builds. -I planned to take the custom PHP code from Beeminder’s WordPress plugin and run it as a stand-alone PHP proxy in front of Etherpad inside the build action. -That way I could avoid reimplementing the custom functionality from the old blog for things like footnotes. -The Continuous Integration (CI) action would then deploy the built blog to whatever static host we settled on. -
-- However Render.com has its own Netlify-style static site builds, so no need for GitHub Actions. -Instead I reimplemented the custom logic in TypeScript. -While this meant more effort in reproducing the features from the old blog, it kept the deploy pipeline much simpler and kept everything in the main TypeScript codebase, which I think was the right call for long-term maintainability. -
-- Beeminder hosts the raw markdown behind the blog posts in an Etherpad instance separate from WordPress, so we didn’t need to worry about exporting post content from WordPress. -But we still needed to export post metadata, including the Etherpad source URLs, as well as some info about the authors, and then use that data in the builds for old posts that aren’t using the new markdown frontmatter. -
-- I used two WordPress plugins to export the data — one for posts and one for users. -I then took this data, imported it into Google Sheets, and exported just the columns we needed into CSV files which are stored in the blog repository. -
-- At build time the code uses these CSV files to look up information about posts that were written before the blog rewrite. -For new posts, this meta information will be stored in frontmatter at the top of the markdown files. -
-- I spent too much time optimizing build performance. -It’s a fun puzzle. -
-- I - - memoized - - liberally. -You don’t want to repeat any expensive computations. -In addition I cached the raw markdown we requested from Etherpad. -In a production build, this is cached in-memory. -Locally (i.e., in development) it’s cached in the filesystem to avoid needing to wait for a bunch of requests to finish before seeing the result of a test build. -
-- There are some fancy markdown features that I needed to use a virtual DOM library to implement. -I started by using - - jsdom - - , -but this was very slow. -I ended up switching to - - happy-dom - - and it was much faster. -
-- When I first started fetching markdown from Etherpad, I made all the requests at once. -This flooded Beeminder’s Etherpad instance and caused our builds to fail. -I switched to doing these requests synchronously, which allowed the builds to complete, but made builds take a long time. -Finally I switched to using - - p-limit - - to send the requests as fast as possible while limiting the number of simultaneously-active requests. -I tested that limit, timing different values, until I found one that was optimal for build speed and reliability. -
-
-
-
-
-
- In conclusion… voila? -Goodbye WordPress, hello Astro. -And it’s being served by Render.com at zero monthly cost. -Here’s the full final stack in case you got bored partway through the above and are jumping here for the punchline: -
-- Danny would also like me to remind everyone that there’s a $10 honey money bounty for any bugs or typos you may find on the blog. -You can report those in the comments or - - in the forum - - ! -
-" -`; - -exports[`body > post astroblog hash f214b52e8804417aa13a661edf7e99c0 1`] = ` -"post predict hash 1c0bdd2bcebb1e1c91531bb0bce392d0 1`] = ` +exports[`body > post predict hash b9ed963e091616cadde1b4326a1ba560 1`] = ` "
post predict hash b9ed963e091616cadde1b4326a1ba560 1`] = ` -" -
- -
-- A fun fact about predicting your own behavior, particularly publicly, is that the act of predicting it changes the prediction. -“I’m 75% likely to maintain my Duolingo streak all year, but now that I’ve said so I’m actually 90% likely, but now that I’ve said - - that - - , …” -Or what happens when the probability starts very low but you add a wager? -It’s like this self-describing xkcd: -
-- -
-- Or like the sentence, “This sentence consists of exactly fifty-seven characters.” - - [1] - -
-- (This, incidentally, is part of what’s hard about predicting whether superhuman AI will accidentally destroy everything humans value in the universe. -If we actually had consensus on the probability being high, we’d be galvanized to change course. -It’s like if you knew you had a high chance of accidentally taking a fatal dose of medicine, you’d double check the dosage and make the probability low again. -Not that I’m arguing against sounding alarm bells.) -
-- Anyway, a fun fact about Beeminder, from way back in the - - Kibotzer - - days, is that it started as essentially a prediction market. -
-- You can hear - - Bethany tell the story - - in an ancient Quantified Self talk from when Beeminder was a side project. -Back then, instead of automatically paying Beeminder when you went off track, Beeminder just showed you a graph of your progress and your commitment. -The commitment contracts themselves were arranged separately. -The inaugural commitment contracts were Bethany and me staking $2,000 each on a gallery of personal fitness and productivity goals during the summer of 2010: -
-- So what did “staking $2,000” mean? -We were each promising to pay some friend or family member that $2k if we didn’t stay on track on all of those goals. -Which friend or family member was determined by auction, coordinated manually by email at first. -We started the bidding at $10, it quickly went to $50, and then we sold them for $65 each the next day. -
-- That meant that our friends were paying $65 for the chance to win $2k. -Which implies market odds of 96.8% that we’d succeed at our goals that summer. -Which we did, phew. -Obviously the probability would not have been nearly so high without the money at stake. -
-- Emboldened by that trial, we set up a slightly more formal auction interface for other people to use: -
-- -
-- That one turns the game theory up to eleven. -The commitment contract there was my sister Melanie’s weight loss goal, with $600 of her money at stake. -The bidders specify in the first row of that table the most they’d pay to be the beneficiary of the commitment contract. -The auction mechanism then does a bunch of math - - [2] - - to decide how much people actually pay (second row) and to divvy the potential booty (third row). -As it turned out Laurie, our mother, got half the contract — half of Melanie’s $600 should she have failed at her goal. - - David Yang - - got 25%, and - - Dan Goldstein - - got the remaining 25%. - - David Reiley - - , - - Bethany - - , and I didn’t bid enough to get anything. -And then, once again, - - Melanie - - did in fact stay on track so those shares didn’t end up paying out. -Each person’s bid implies the odds they assign to Melanie actually staying on track on her weight goal. -The bottom right number — 2.5% — is the aggregated market odds. -
-- Today, Beeminder only offers infinitely bad odds. -You pay for going off track, but don’t get paid for staying on track. -You could think of that as perfectly fair odds if using Beeminder meant a 100% chance of always staying on track. -It doesn’t but these days we view this all very differently, largely because staying on vs going off track is not actually a binary outcome. -You’ll inevitably go off track - - sometimes - - and that’s fine — you’re really just paying for the motivation Beeminder is providing the rest of the time. -We have a whole series of blog posts about this: - - Derailing Is Not Failing - - , - - Paying Is Not Punishment - - , and - - Derailing It Is Nailing It - - . -
-- Probably if you’re reading to the end of Beeminder blog posts you’re already on board with all that. - - [3] - - But if you’re a prediction market person, we’d actually love to convince you to try predicting your own behavior via - - Manifold markets - - . -(Disclosure: I’m an investor in Manifold.) -Here are some examples of people (including us) doing so: -
-- Last but not least, I want to - - launch a new feature - - to allow Manifold users to transfer mana (that’s Manifold’s play-money currency) to honey money (Beeminder’s currency, aka store credit on Beeminder) and vice versa in time to talk about it and use it at - - Manifest - - ! -
-
-
-
-
-
- - [1] - - If you’re wondering how I made that sentence come out right, I just let Mathematica brute-force it: -
-
-
- For[i = 2, i < 1000, i++,
- s = "This sentence consists of exactly "<>
- IntegerName[i]<>" characters.";
- If[StringLength[s] == i, Print[i, ": ", s]]]
-
-
-- I then thought to let GPT-4’s Code Interpreter - - have a shot at it - - . -It gets there, with minimal coaxing. -As of 2023, this feels utterly gobsmacking. -
-- - [2] - - If you really want to know, we have an - - ancient document - - about it. -If I had to justify it, it started with the idea to let everyone buy in at any amount of money they chose. -Call that the pot. -Then you divvy up shares of the contract -(the money you win if the person auctioning off the commitment contract doesn’t do what they committed to) -by just giving each person a fraction of the contract equal to the fraction they contributed to the pot. -The more you put in, the more of the payout you get (if the contract pays out). -It’s like a parimutuel market. -
-- But then! -This is where the wild game theory comes in. -Deciding how much to pay into the pot is a tricky strategic problem that depends pretty delicately on what other people are paying in. -
-- Say two other bidders have put in $100 each on a $1000 contract. -And let’s say the person creating this commitment contract is doomed, as in, you’re 100% certain they’ll be paying that $1000. -So those other bidders are currently getting $500 each by spending $100. -If you put in $100 as well then the three of you will get $333 each. -This is still profitable and you want to put more into the pot, but at about $222 you start losing money. -Each person putting $222.22 into the pot turns out to be the Nash equilibrium of this game. -
-- So the idea of this mechanism is to compute those equilibrium bids for everyone. -You just say how much you think the whole contract is worth, and the mechanism plays the Nash equilibrium for you, under the maybe-false assumption that everyone has said their true value for the whole contract. -Bidders don’t have to understand any of that. -They can just keep bumping up their bids as long as the mechanism is giving them a price and a fraction of the contract that feels fair. -
-- It’s all preposterous amounts of overkill/overcomplication but it did seem to work. -
-- - [3] - - To recap the argument, if you view Beeminder’s stings as punishment, that can -(a) encourage an excuse-making mindset over a results-oriented mindset, and maybe more importantly, -(b) encourage you to make your goals less ambitious. -If you accept that some derailments are inevitable — part of the cost of the service, just nudges or rumble strips keeping you on track — you can dial in a sweet spot where you’re being pushed to do as much as possible (or whatever maximizes the motivational value you get from Beeminder) at minimal cost. -See also - - Yassine Meskhout - - ’s notion of - - taxing vs punitive commitment contracts - - . -
-" -`; - exports[`body > post preeat hash 8e7d083d3033a47cebeb9b41336dcae1 1`] = ` "@@ -72988,7 +72186,7 @@ See especially the case of Pact, whose scheme of paying successful users we̵ " `; -exports[`body > post readwise hash cfc7dd19133c9cf22f76789324c48ebf 1`] = ` +exports[`body > post readwise hash df9779ab68a4458e6036baf279071d00 1`] = ` "
post readwise hash df9779ab68a4458e6036baf279071d00 1`] = ` -" -
- -
-- - Readwise Reader - - is a powerful tool for “power readers”. -It’s like a supercharged read-it-later app, with first-class support for notes and highlights and tags. -Now, you can keep track of your Readwise Reader items using Beeminder. -
-- You save things like web pages, PDFs, YouTube videos, Twitter threads, or aim it at an RSS feed or an email newsletter. -It collects them all and makes them easy to read on their own. -Web pages show just the juicy content you’re looking for. -YouTube videos are shown alongside their transcript, for example, and the transcript highlights as the words are spoken in the video. -You can click on a line in the transcript to jump right to it. -Of course, it wouldn’t be a good read-it-later app if it didn’t save your progress in the video (or anything else you use Readwise Reader for). -
-- As awesome as Readwise Reader is, it’s no surprise that some folks go into a “read-it-later” frenzy. -The save rate tends to be higher than the read rate, and that pile accumulates… -I’d show you some good quotes about this problem, but they’re somewhere in my later pile. -We recently got a note from the Readwise Reader folks that they added some API endpoints for fetching items that are already in your account. -Readwise Reader has been sitting near the top of the candidate autodata integration list for a while — and folks have loved our - - Pocket integration - - for a while, and Adam’s Readwise Reader had gotten a little overwhelming… -
-- You set a limit for a Readwise Reader location, like Inbox, Later, or Shortlist. -If you’re already under the limit, great! -You’ll have to stay under the limit each day or pay a penalty. -If you’re currently over that limit, Beeminder will set up a goal to get you down to that limit, and then stay under it. -
-- If you want to adjust how long it’ll take to get to your limit, you can do that once the goal is created. -
-- Welcome aboard! -You might want to check out our - - getting started guide - - in our Help Docs, and then get yourself signed up. -Assuming you’re on board with the commitment device, paying-money-if-you-go-off-track bit, the beauty of an autodata integration is that you don’t normally need to interact with Beeminder once you’ve gone through the process above of setting up your commitment. -Just work through your items, either deleting them, archiving them, or otherwise moving them out of the location (your Inbox or Later or Shortlist) when Beeminder yells at you and you’re good. -
-- Ready to dive in? -
-
-
-
-
-
diff --git a/src/pages/[slug].astro b/src/pages/[slug].astro
index f0ece14..715200f 100644
--- a/src/pages/[slug].astro
+++ b/src/pages/[slug].astro
@@ -1,5 +1,5 @@
---
-import type { InferGetStaticPropsType } from "astro";
+import type { InferGetStaticPropsType } from "astro";
import Tags from "../components/Tags.astro";
import getPosts from "../lib/getPosts";
import Shadowbox from "../components/Shadowbox.astro";
@@ -12,10 +12,10 @@ import Paginated from "../layouts/Paginated.astro";
export async function getStaticPaths() {
const posts = await getPosts({
includeUnpublished: true,
- sort: true
+ sort: true,
});
- const published = (p: Post) => p.status === "publish"
+ const published = (p: Post) => p.status === "publish";
return posts.map((post: Post, i: number) => ({
params: {
@@ -31,22 +31,9 @@ export async function getStaticPaths() {
type Props = InferGetStaticPropsType Comments are disabled on DRAFT pages. Comments are disabled on DRAFT pages.