mirror of
https://github.com/Smaug123/static-site-pipeline
synced 2025-10-24 16:58:40 +00:00
Compare commits
66 Commits
8b2bab9baa
...
main
Author | SHA1 | Date | |
---|---|---|---|
|
71380df561 | ||
|
e38dafbd58 | ||
|
9901b1dfc4 | ||
|
80c7b2e425 | ||
|
0254b0377d | ||
|
19093d968b | ||
|
3b64e06862 | ||
|
d4b79faa07 | ||
|
bcb8a9eb60 | ||
|
ecd774c318 | ||
|
22aca8a3be | ||
|
1475932c43 | ||
|
0408056c13 | ||
|
2cd7f6839e | ||
|
6fb121da71 | ||
|
f4bfd5fbfe | ||
|
ec9323fdd3 | ||
|
d523b2bf2f | ||
|
89f12a2d39 | ||
|
f4e4564125 | ||
|
9d817b7f19 | ||
|
359647e1d2 | ||
|
cd3e6426a1 | ||
|
23ba49ce71 | ||
|
023d31419a | ||
|
1ac4d77cc9 | ||
|
3d0b74d125 | ||
|
12db6adbb5 | ||
|
5523bb7957 | ||
|
975a37a108 | ||
|
bff167f362 | ||
|
22b80e1fc3 | ||
|
8256606ca0 | ||
|
45f1d56d53 | ||
|
8aac808eac | ||
|
98efc23981 | ||
|
c9d197ebb5 | ||
|
f6e28b0b30 | ||
|
4caf4ee1e6 | ||
|
2ce132a914 | ||
|
8a73e10de3 | ||
|
56669036f0 | ||
|
1cd47be72c | ||
|
aba1f13646 | ||
|
eb2780622d | ||
|
4205c95fa0 | ||
|
151b69aeab | ||
|
d4f0837fb9 | ||
|
ac6e52f16d | ||
|
a19aa1c2e3 | ||
|
97ed4d00f3 | ||
|
2fd36b7437 | ||
|
06c71b0da2 | ||
|
95e110408b | ||
|
b76d192516 | ||
|
fa05c57c35 | ||
|
570afed012 | ||
|
a99747fe39 | ||
|
860640283f | ||
|
fc9bcfb85a | ||
|
2517e03143 | ||
|
ce38f6ba39 | ||
|
a680d7badf | ||
|
18314d94f6 | ||
|
da3a4d2e53 | ||
|
11853da860 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -29,3 +29,4 @@ hugo/static/misc/TokyoEntrance2016/
|
||||
|
||||
.idea/
|
||||
.profile*
|
||||
.direnv/
|
||||
|
@@ -7,3 +7,12 @@ https://www.patrickstevens.co.uk/posts/2016-04-13-independence-of-choice/
|
||||
https://www.patrickstevens.co.uk/posts/2016-04-08-another-monty-hall-explanation/
|
||||
https://www.patrickstevens.co.uk/misc/AdjointFunctorTheorems/AdjointFunctorTheorems.pdf
|
||||
https://www.patrickstevens.co.uk/posts/2021-02-20-in-praise-of-dry-run/
|
||||
http://www.patrickstevens.co.uk/wordpress/archives/364
|
||||
http://www.patrickstevens.co.uk/wordpress/archives/379
|
||||
http://www.patrickstevens.co.uk/wordpress/archives/474
|
||||
http://www.patrickstevens.co.uk/wordpress/archives/584
|
||||
https://www.patrickstevens.co.uk/posts/2020-10-23-anki-learning/
|
||||
https://www.patrickstevens.co.uk/sequentially-compact-iff-compact/
|
||||
http://patrickstevens.co.uk/mathematical%20summary/2014/12/19/matrix-puzzle.html
|
||||
https://www.patrickstevens.co.uk/posts/2024-07-25-lob-theorem/
|
||||
https://patrickstevens.co.uk/cauchy-schwarz-proof/
|
||||
|
84
flake.lock
generated
84
flake.lock
generated
@@ -80,11 +80,29 @@
|
||||
"systems": "systems_2"
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1701680307,
|
||||
"narHash": "sha256-kAuep2h5ajznlPMD9rnQyffWG8EM/C73lejGofXvdM8=",
|
||||
"lastModified": 1731533236,
|
||||
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"rev": "4022d587cbbfd70fe950c1e2083a02621806a725",
|
||||
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"flake-utils_3": {
|
||||
"inputs": {
|
||||
"systems": "systems_3"
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1694529238,
|
||||
"narHash": "sha256-zsNZZGTGnMOf9YpHKJqMSsa0dXbfmxeoJ7xHlrt+xmY=",
|
||||
"owner": "numtide",
|
||||
"repo": "flake-utils",
|
||||
"rev": "ff7b65b44d01cf9ba6a71320833626af21126384",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
@@ -104,11 +122,11 @@
|
||||
"scripts": "scripts_2"
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1703798553,
|
||||
"narHash": "sha256-dDym75Eq6TIw9IrokBWwSoto0/l3nxFGpH4/VZkeqrQ=",
|
||||
"lastModified": 1721842442,
|
||||
"narHash": "sha256-bjvLR/KwHToz0/kRcZ8/MHXs3y4YCTtz3UUQ/D/AliA=",
|
||||
"ref": "refs/heads/main",
|
||||
"rev": "d441f6c507ecbfc7e95440ca8f021fcec6cb4767",
|
||||
"revCount": 18,
|
||||
"rev": "cab3ccc58ec12af893c9ea6b87aac0dfeefa3752",
|
||||
"revCount": 24,
|
||||
"type": "git",
|
||||
"url": "file:/Users/patrick/Desktop/website/static-site-images"
|
||||
},
|
||||
@@ -119,19 +137,15 @@
|
||||
},
|
||||
"katex": {
|
||||
"inputs": {
|
||||
"flake-utils": [
|
||||
"flake-utils"
|
||||
],
|
||||
"nixpkgs": [
|
||||
"nixpkgs"
|
||||
]
|
||||
"flake-utils": "flake-utils_3",
|
||||
"nixpkgs": "nixpkgs_2"
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1696151934,
|
||||
"narHash": "sha256-8kihcqdgYjoVuGozfgfcWh81yqMUvns4+C/fgkn+RNQ=",
|
||||
"lastModified": 1704150937,
|
||||
"narHash": "sha256-G6uJKkY5VErgobe51IIbp/ugHDIhVx5e0xNjJ90JEOk=",
|
||||
"owner": "Smaug123",
|
||||
"repo": "KaTeX",
|
||||
"rev": "ac1f9b30441f63ea20216a36ffa7148dc0e9a9b3",
|
||||
"rev": "b74ed701beec2bebd161a0b5ea30c496c5206b96",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
@@ -172,6 +186,21 @@
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"nixpkgs_3": {
|
||||
"locked": {
|
||||
"lastModified": 1741381468,
|
||||
"narHash": "sha256-w2ig7s5Afz0dOta4Sfynz7KnIWiKN4dtEW2SjS6qztU=",
|
||||
"owner": "NixOS",
|
||||
"repo": "nixpkgs",
|
||||
"rev": "79dab6b4451fb66a939e9f894fec0bd22253cf94",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "NixOS",
|
||||
"repo": "nixpkgs",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"pdfs": {
|
||||
"inputs": {
|
||||
"flake-utils": [
|
||||
@@ -183,11 +212,11 @@
|
||||
"scripts": "scripts_3"
|
||||
},
|
||||
"locked": {
|
||||
"lastModified": 1696190787,
|
||||
"narHash": "sha256-bO/NInpwVefs5Iey8WVwPFnXPt/3WN7WvYXTxzLKmGQ=",
|
||||
"lastModified": 1725713640,
|
||||
"narHash": "sha256-JBTpmQn3Gg3scVUUyLuks1gX+EzkctD+ZN3CBsxXuwQ=",
|
||||
"owner": "Smaug123",
|
||||
"repo": "static-site-pdfs",
|
||||
"rev": "a36d3025b9625cc50fc5bd2eca867eacd8a5bcb9",
|
||||
"rev": "e7133d72e6fab800aa3fd91cab47b6bc6824a0ca",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
@@ -203,7 +232,7 @@
|
||||
"flake-utils": "flake-utils_2",
|
||||
"images": "images",
|
||||
"katex": "katex",
|
||||
"nixpkgs": "nixpkgs_2",
|
||||
"nixpkgs": "nixpkgs_3",
|
||||
"pdfs": "pdfs",
|
||||
"scripts": "scripts_4"
|
||||
}
|
||||
@@ -297,6 +326,21 @@
|
||||
"repo": "default",
|
||||
"type": "github"
|
||||
}
|
||||
},
|
||||
"systems_3": {
|
||||
"locked": {
|
||||
"lastModified": 1681028828,
|
||||
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
|
||||
"owner": "nix-systems",
|
||||
"repo": "default",
|
||||
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"owner": "nix-systems",
|
||||
"repo": "default",
|
||||
"type": "github"
|
||||
}
|
||||
}
|
||||
},
|
||||
"root": "root",
|
||||
|
@@ -2,17 +2,16 @@
|
||||
description = "Static site builder for patrickstevens.co.uk";
|
||||
|
||||
inputs = {
|
||||
flake-utils.url = github:numtide/flake-utils;
|
||||
flake-utils.url = "github:numtide/flake-utils";
|
||||
scripts.url = "github:Smaug123/flake-shell-script";
|
||||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-22.11";
|
||||
nixpkgs.url = "github:NixOS/nixpkgs";
|
||||
extra-content = {
|
||||
url = "path:/Users/patrick/Desktop/website/extra-site-content";
|
||||
flake = false;
|
||||
};
|
||||
katex = {
|
||||
url = "github:Smaug123/KaTeX/nix";
|
||||
inputs.nixpkgs.follows = "nixpkgs";
|
||||
inputs.flake-utils.follows = "flake-utils";
|
||||
# inputs.nixpkgs.follows = "nixpkgs";
|
||||
};
|
||||
images = {
|
||||
url = "git+file:/Users/patrick/Desktop/website/static-site-images";
|
||||
@@ -85,7 +84,7 @@
|
||||
mv output $out
|
||||
'';
|
||||
};
|
||||
in rec {
|
||||
in {
|
||||
packages = flake-utils.lib.flattenTree {
|
||||
gitAndTools = pkgs.gitAndTools;
|
||||
default = website;
|
||||
|
@@ -5,7 +5,7 @@ title = "Patrick Stevens"
|
||||
theme = "anatole"
|
||||
buildFuture = false
|
||||
enableEmoji = true
|
||||
paginate = 20
|
||||
pagination.pagerSize = 20
|
||||
|
||||
[params]
|
||||
profilePicture = "/images/AboutMe/profile"
|
||||
@@ -80,6 +80,10 @@ pygmentsCodefences = true
|
||||
name = "Film List"
|
||||
identifier = "films"
|
||||
url = "/films"
|
||||
[[menu.main]]
|
||||
name = "Games"
|
||||
identifier = "games"
|
||||
url = "/games"
|
||||
[[menu.main]]
|
||||
name = "Lifehacks"
|
||||
identifier = "lifehacks"
|
||||
|
@@ -6,6 +6,11 @@ layout: page
|
||||
sidenotes: true
|
||||
---
|
||||
I am Patrick Stevens, a software engineer based in London, England.
|
||||
|
||||
{{< rawhtml >}}
|
||||
<img src="/images/AboutMe/northern-lights.png" alt="Photo of me under the Northern Lights" style="max-width: 500px;">
|
||||
{{< /rawhtml >}}
|
||||
|
||||
I completed my BA+MMath at the University of Cambridge.
|
||||
|
||||
Social media accounts:
|
||||
|
@@ -12,7 +12,7 @@ Any remaining decks here are CC-BY-SA.
|
||||
|
||||
* [Geography] (the deck has a misleading name; it's actually a general Geography deck). You can filter out the `london-tube` tag if you like, or `world-capitals`, or `american-geography`.
|
||||
|
||||
<a href="http://creativecommons.org/licenses/by-sa/3.0/deed.en_GB" rel="license"><img style="border-width: 0;" alt="Creative Commons Licence" src="https://licensebuttons.net/l/by-sa/3.0/88x31.png" /></a>
|
||||
This work by <a href="/anki-decks" rel="cc:attributionURL">Patrick Stevens</a> is licensed under a <a href="http://creativecommons.org/licenses/by-sa/3.0/deed.en_GB" rel="license">Creative Commons Attribution-ShareAlike 3.0 Unported License</a>.
|
||||
[](http://creativecommons.org/licenses/by-sa/3.0/deed.en_GB)
|
||||
This work by [Patrick Stevens](/anki-decks) is licensed under a [Creative Commons Attribution-ShareAlike 3.0 Unported License](http://creativecommons.org/licenses/by-sa/3.0/deed.en_GB).
|
||||
|
||||
[Geography]: /AnkiDecks/CapitalsOfTheWorld.apkg
|
||||
[Geography]: /AnkiDecks/CapitalsOfTheWorld.apkg
|
||||
|
@@ -7,6 +7,11 @@ layout: page
|
||||
|
||||
This page holds a list of films I have watched, spoiler-free, starting from 9th January 2015.
|
||||
|
||||
* [Dungeons and Dragons: Honour Among Thieves](https://www.imdb.com/title/tt2906216/): Really rather good! As they say, "Consider this fan serviced". Light-hearted, plenty of flashy callouts.
|
||||
* [The Grand Budapest Hotel](https://www.imdb.com/title/tt2278388/): This is one of my favourite films. In general I find Wes Anderson a bit hit-or-miss, but this film is sublime. Excellent characterisation, quick deadpan wit. Ralph Fiennes is just glorious here.
|
||||
|
||||
* [Argylle](https://www.imdb.com/title/tt15009428/): Sadly disappointing. This film was at least twice as long as it should have been. A good version would have been a short funny pastiche; what we actually got was a long boring film punctuated by moments of glorious whimsy.
|
||||
|
||||
* [A Haunting in Venice](https://www.imdb.com/title/tt22687790/): Well, I really enjoyed this, and I think I was surrounded by heathens in the cinema. I successfully called precisely none of the plot, and it all tied up so neatly. Ariadne Oliver will always be Zoë Wanamaker to me, but I believed Kenneth Branagh. Top-tier Poirot.
|
||||
|
||||
* [Oppenheimer](https://en.wikipedia.org/wiki/Oppenheimer_(film)): Brilliant. It was a little too long, but I couldn't pick out anything to take away. Great acting, great filming, a bit harrowing.
|
||||
|
60
hugo/content/games/index.md
Normal file
60
hugo/content/games/index.md
Normal file
@@ -0,0 +1,60 @@
|
||||
---
|
||||
lastmod: "2024-04-28T20:51:00.0000000+01:00"
|
||||
title: Games
|
||||
author: patrick
|
||||
layout: page
|
||||
---
|
||||
|
||||
This page holds an incomplete list of games I have played.
|
||||
|
||||
# Video games
|
||||
|
||||
## In progress
|
||||
|
||||
* ibb and obb. Fun two-player co-op, neat and slightly mind-bending physics.
|
||||
* It Takes Two. Amusing but not worth the price, I think. This is more like a constant stream of "oh, what funny one-off mechanic have they implemented next".
|
||||
|
||||
## Not in progress
|
||||
|
||||
* Chants of Sennaar. An extremely impressive 81% of players have the Steam achievement for leaving the first game area, which I think tells you how gripping this game is. It's basically a particular kind of IQ test, but it's really fun. Reminiscent of _Return of the Obra Dinn_. I could have done with a little more variation in the languages; there are a few neat bits, but there was a broad language design space they mostly didn't touch.
|
||||
* Baldur's Gate 3 (hundreds of hours). Glorious game, aesthetically beautiful, absolutely gigantic amount of content, and sometimes succeeds at making you empathise with the characters. Having played this, I wouldn't really want to go back to playing a DnD-style RPG any other way: all the bookkeeping happens in the background.
|
||||
* Factorio, and Dyson Sphere Program (both hundreds of hours). Grouped together because they are basically the same game. I have banned myself from playing this genre of game (I gave myself RSI from it for a while). Between the two, DSP is much prettier and somewhat more forgiving, but also it has the annoying mechanic of laying your factory out on spheres. "Cannot place blueprint across tropic lines" just feels bad; you really want your factory layout not to depend on where it is situated.
|
||||
* Outer Wilds (about half an hour). Everyone raves about this game, but it makes me travel-sick very fast. No further comment.
|
||||
* Human Resource Corporation, and 7 Billion Humans (25 hours total). As I've got older, games which are purely programming have become less appealing (that is, after all, my day job). These ones are visually pretty, but I have no idea how they'll play for someone who is not a programmer.
|
||||
* Among Us (70 hours). I love this game (multiplayer social deduction), and am quite sad that everyone stopped playing it after the pandemic. Your first few games will be extremely hard if you're an impostor, because your plausibility relies on being able to make up what you were doing where (which means you need to know the maps), but once you've got over that it's just really fun.
|
||||
* Antichamber (2 hours). This makes me motion-sick really quickly, and I have only played it in ten-minute bursts. I imagine this would be very interesting if I could play it.
|
||||
* Baba Is You (68 hours). Cute puzzle game, with some extremely difficult puzzles. Mind-blowingly beautiful mechanics as you get towards the end. Tier 1. I played this with my housemates on a big screen and it was a great way to play!
|
||||
* Braid (9 hours). Lovely puzzle game, innovative, coherent; a classic Jonathan Blow game. See [qntm's write-up](https://qntm.org/braid).
|
||||
* Classic Sudoku (135 hours) and Killer Sudoku (107 hours), by the Cracking the Cryptic team. A bunch of these puzzles are just really neat! A few are slogs, but many of them are cute, and I got the impression at the time that it might be quite a good introduction to advanced Sudoku techniques. The over-100-hours is because I have done each puzzle twice.
|
||||
* Death and Taxes (2 hours). Meh. Three years after playing it I can't really remember it at all.
|
||||
* Disco Elysium (17 hours). Some of this game is pretty funny. I haven't completed it, because it was getting to be a bit of a slog.
|
||||
* Dota 2 (thousands of hours). Deeply addictive. You will spend the first thousand hours or so *completely sucking* at this game, and improvement comes from recognising that you suck (thanks Purge for the framing on this). I stopped playing this game many years ago when I discovered that I didn't actually enjoy it. My mechanical skill was pretty mediocre, so my specialist skill, inasmuch as I had one, was to bind together a team and keep morale up. A team of five coherent players will beat a team of five individuals any day of the week, and a team of five players will beat a team of four players plus a griefer, so an extremely valuable role is "keep your team feeling like a unit". But it grinds away at the soul, and eventually I realised it isn't fun.
|
||||
* Elsinore (7 hours). Intriguing and cute puzzle game: you're playing Ophelia, caught in a time loop for several critical days in the setting of Hamlet. You gradually learn what happens when, and how you can act to change what happens. I haven't finished this because it had a bit too much of the [Shlemiel the Painter](https://www.joelonsoftware.com/2001/12/11/back-to-basics/) about it: as you solve milestones, you keep revealing more, but have to keep retracing the same steps anyway.
|
||||
* The Zachtronics games (Exapunks, TIS-100, Shenzhen I/O, Opus Magnum, MOLEK-SYNTEZ; excluding Infinifactory; a couple of hours). Again, like Human Resource Corporation, these games are purely programming, and I'm getting too old for that sort of thing to be a fun way to relax.
|
||||
* The House of da Vinci (and 2 and 3; 20 hours). These are all a bit too point-and-click linear for my liking.
|
||||
* Inscryption (15 hours). Interesting card game mechanic; intriguing metagame.
|
||||
* Islands of Insight (55 hours). Loads and *loads* of small puzzles, many of them Nikoli-like. Some are just quite annoying ("go to this place and hunt around"). Some are deeply absorbing. Very pretty aesthetics.
|
||||
* The Witness (55 hours). Good lord is this game amazing. It's a superstimulus for the sense of discovering an insight. The game mechanics are all taught entirely without words, which is a brilliant trick if you can manage it.
|
||||
* The Looker (2 hours). Parody of The Witness. Some of it laugh-out-loud funny.
|
||||
* Loop Hero (25 hours, not all of which was me). Very playable roguelike.
|
||||
* Obduction (7 hours). I distinctly remember that when I was much younger I enjoyed Myst and friends. This was back in the days when I had the patience to read _The Lord of the Rings_, for example. Nowadays I suspect I just don't have the attention span and/or patience. Obduction in particular requires a *lot* of walking around.
|
||||
* Offworld Trading Company (30 hours). I am *terrible* at this game, and find it deeply stressful, but it's extremely compelling. The market mechanic is either very innovative or very innovatively phrased.
|
||||
* Oxygen Not Included (8 hours). I feel like I'm missing some fundamental part of this game. It feels like it should be some massive Factorio-like infinite world automation thing, but in practice I ended up doing an awful lot of micromanagement. Probably a skill issue.
|
||||
* Patrick's Parabox (10 hours). Cute puzzle game. Mind-bending, innovative. Really good.
|
||||
* Portal (and 2, and the Reloaded mod; 35 hours in total). Classics, of course.
|
||||
* Return of the Obra Dinn (12 hours). Aesthetically beautiful game. When the main mechanic was introduced, I sat back in awe. You gradually get to have quite a good mental map of what's going on here. Highly recommended.
|
||||
* Slay the Spire (hundreds of hours). Deep gameplay (_extremely_ deep - check out Jorbs's streams!). One of those games where it takes a while to realise just how bad at it you are; all your losses feel like luck, but actually this game is at least 80% skill.
|
||||
* Roguebook (23 hours). A variant of Slay the Spire. Aesthetically quite pleasing, but I didn't find it as gripping as StS.
|
||||
* Subnautica (26 hours). Personally I am a wimp and found quite a bit of this game primally terrifying; so much so that I ended up watching a stream of the game past a certain point. I don't much like the choice to phrase the story in terms of time pressure, when everything is actually triggered by reaching certain milestones. It's pretty, though, and if I weren't a wuss I'd definitely have played it to the end.
|
||||
* SUPERHOT (3 hours). This is the first-person shooter for the person like me who is terrible at first-person shooters! It's more like a puzzle game than a shooter, but for the first time it made me feel like I had a game in the shooter genre that I could play!
|
||||
* Taiji (12 hours). This is a puzzle game much like The Witness but a bit worse (but still a great game!).
|
||||
* The Talos Principle (and 2; 35 hours total). Puzzle games with a skin of basic philosophy. Aesthetically beautiful; some very hard puzzles.
|
||||
* Torment: Tides of Numenera (40 hours, because I've played it twice). If you're going to play an RPG, play Baldur's Gate 3 instead. Torment is much less pretty and considerably smaller, but I did have fun playing it.
|
||||
* Understand (3 hours). Tiny puzzle game. Almost precisely an IQ test. Great fun.
|
||||
* We Were Here (and Too, and Together; a few hours total). Two-player co-op games. The in-game chat is really janky, so we just used Discord. I feel like these were nearly great games, but something just felt missing. A Baldur's Gate 3 campaign completely supplanted this.
|
||||
|
||||
# Board games
|
||||
|
||||
(This section is basically a placeholder for now.)
|
||||
|
||||
* Agricola, one of my favourite games.
|
@@ -7,7 +7,7 @@ layout: page
|
||||
If I ever become rich and famous, I'm sure I'll be besieged with requests for "how to do better in life". I hereby head such requests off at the pass, by providing a list of [lifehacks] I am either using or considering the use of.
|
||||
|
||||
* For learning smallish but numerous facts (such as a list of theorems), I use [Anki], which is a [spaced-repetition] learning system, allowing you to enter flashcards and have them shown to you regularly. The time between repetitions of a certain flashcard changes, depending on how well you've been doing on that flashcard - so marking your performance on a particular card as "easy" rather than "hard" tells Anki that you don't want to see that card for a while. It's a bit like the antithesis of cramming, where you see the material exactly once and use it a short time later; Anki is designed for reviewing the material many times (at an optimal spacing) for recall whenever you need it. The idea is to make use of the [spacing effect] - an extremely powerful memory technique that is currently ignored by almost all methods of formal teaching ([Memrise] is a notable exception; I used Memrise until I used Anki).
|
||||
* A surprisingly good way of making myself work when I'm feeling unmotivated is to gather a few like-minded friends and to work in absolute silence with them (possibly on completely unrelated topics). Oddly, I'd not thought of it until reading <a title="Co-working LessWrong post" href="http://lesswrong.com/lw/gwo/coworking_collaboration_to_combat_akrasia/" target="_blank">a LessWrong post on the subject</a>. There's a kind of "all in this together" feeling, as well as the public commitment effect.
|
||||
* A surprisingly good way of making myself work when I'm feeling unmotivated is to gather a few like-minded friends and to work in absolute silence with them (possibly on completely unrelated topics). Oddly, I'd not thought of it until reading [a LessWrong post on the subject](http://lesswrong.com/lw/gwo/coworking_collaboration_to_combat_akrasia/). There's a kind of "all in this together" feeling, as well as the public commitment effect.
|
||||
* I use [f.lux], an application which dims and tints red the computer screen after dusk. I have no idea whether or not it has any effect on wakefulness at night (that is, whether or not being bathed in a standard blue glow keeps me awake), but it certainly feels nicer on the eye.
|
||||
* I am currently in the middle of learning [Dvorak], which is a keyboard layout (QWERTY is the usual one) that is supposedly easier on the hands than QWERTY. It puts vowels all together in easy-to-reach places, and the most common consonants in easy places such that words tend to be made of letters which lie in different hands. (In QWERTY, for example, the word "the" is oddly hard to type, for such a common word - all the characters are away from the home row - but in Dvorak it's just a simple flourish from right to left on the home row.) A friend tells me that [Colemak] is better than Dvorak, but I'd already half-learnt Dvorak by the time ey told me this, and Dvorak interfered heavily with my attempts to learn Colemak. It appears to be much of a muchness, anyway - both are considerably better than QWERTY.
|
||||
* I don't know if it qualifies as a lifehack - more of a biohack or something - but [lucid dreaming] is really cool, and it doesn't take an enormous amount of commitment to learn to do (it just requires the setting up of a few habits throughout the day).
|
||||
|
@@ -12,21 +12,19 @@ title: Stumbled across 9th July 2013
|
||||
---
|
||||
Being bored over the summer holiday, I decided that I would document the cool things I ran across on the Internet. Over the last week, there have been many of these. If I see anything particularly amazing, it'll go in one of these aggregation posts.
|
||||
|
||||
* Neurons are surprisingly beautiful: <http://blog.eyewire.org/gallery/image-gallery/>
|
||||
* A rather neat and very short story: <https://qntm.org/timeloop>
|
||||
* A *bit* less short but just as good a short story: <https://qntm.org/responsibility>
|
||||
* A rant with which students can all identify, in The Cambridge Student magazine: now lost from the Internet.
|
||||
* An Easter Island word "tingo" means "to borrow objects from a friend’s house one by one until there are none left": <link to the Internet Archive>("http://web.archive.org/web/20100516040410/http://blog.web-translations.com/2008/12/toujours-tingo-words-that-dont-exist-in-english/)
|
||||
* Musings on free will: <http://www.mit.edu/people/dpolicar/writing/prose/text/godTaoist.html>
|
||||
* A thing that I just have to share again: <http://nextbigfuture.com/2013/06/technical-hurdles-have-been-overcome.html>
|
||||
* The human brain is a really weird piece of kit: <http://lesswrong.com/lw/20/the_apologist_and_the_revolutionary/>
|
||||
* We *have* to make one of these at some point: <http://www.pimpthatsnack.com/project/302/1">
|
||||
* This is quite soothing in a weird kind of way: <https://thingsfittingperfectlyintothings.tumblr.com/>
|
||||
* It is possible to be deficient in arsenic. (Link to the Soylent Discourse forum is permanently defunct.)
|
||||
* A really useful website for when you don't want to have to spin up Wolfram|Alpha to work out time differences: <http://everytimezone.com/>
|
||||
* Why never to talk to the police (seriously, never talk to the police): <https://www.youtube.com/watch?v=6wXkI4t7nuc>
|
||||
* A fascinating book about the power of positive and negative reinforcement, and why they're often done wrongly: [Don’t Shoot the Dog]
|
||||
* The Church of England really took its time, but at last they've done it: <https://www.bbc.co.uk/news/uk-23215388>
|
||||
* The Hawkeye Initiative, for the liberation of women in comics: <http://thehawkeyeinitiative.com/>
|
||||
|
||||
[Don’t Shoot the Dog]: https://web.archive.org/web/20130206170903/http://www.papagalibg.com/FilesStore/karen_pryor_-_don_t_shoot_the_dog.pdf
|
||||
* [Neurons are surprisingly beautiful](http://blog.eyewire.org/gallery/image-gallery/)
|
||||
* A rather neat and very short story, [Time Loop](https://qntm.org/timeloop) by qntm
|
||||
* A bit less short but just as good a short story: [I don't know, Timmy, being God is a big responsibility](https://qntm.org/responsibility) by qntm
|
||||
* A rant with which students can all identify, in The Cambridge Student magazine: now lost from the Internet.
|
||||
* An Easter Island word "tingo" means "to borrow objects from a friend's house one by one until there are none left": [Link to the Internet Archive](http://web.archive.org/web/20100516040410/http://blog.web-translations.com/2008/12/toujours-tingo-words-that-dont-exist-in-english/)
|
||||
* Musings on free will: [Is God a Taoist?](http://www.mit.edu/people/dpolicar/writing/prose/text/godTaoist.html)
|
||||
* A thing that I just have to share again: [Technical hurdles have been overcome for the first human head transplant](http://nextbigfuture.com/2013/06/technical-hurdles-have-been-overcome.html)
|
||||
* The human brain is a really weird piece of kit: [The Apologist and the Revolutionary](http://lesswrong.com/lw/20/the_apologist_and_the_revolutionary/)
|
||||
* We *have* to make one of these at some point: [the Creme de la Creme Egg](https://www.pimpthatsnack.com/project/the-creme-de-la-creme-egg/)
|
||||
* This is quite soothing in a weird kind of way: [Things Fitting Perfectly Into Things](https://thingsfittingperfectlyintothings.tumblr.com/)
|
||||
* It is possible to be deficient in arsenic. (Link to the Soylent Discourse forum is permanently defunct.)
|
||||
* A really useful website for when you don't want to have to spin up Wolfram|Alpha to work out time differences: [Every Time Zone](http://everytimezone.com/)
|
||||
* Why never to talk to the police (seriously, never talk to the police): [Don't Talk to the Police](https://www.youtube.com/watch?v=6wXkI4t7nuc)
|
||||
* A fascinating book about the power of positive and negative reinforcement, and why they're often done wrongly: [Don't Shoot the Dog](https://web.archive.org/web/20130206170903/http://www.papagalibg.com/FilesStore/karen_pryor_-_don_t_shoot_the_dog.pdf)
|
||||
* The Church of England really took its time, but at last they've done it: [Church of England makes Chichester child abuse apology](https://www.bbc.co.uk/news/uk-23215388)
|
||||
* The Hawkeye Initiative, for the liberation of women in comics: [The Hawkeye Initiative](http://thehawkeyeinitiative.com/)
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-07-29T00:00:00Z"
|
||||
aliases:
|
||||
- /stumbled_across/stumbled-across-2/
|
||||
- /stumbled-across-29-july-2013/
|
||||
- /wordpress/archives/241/
|
||||
title: Stumbled across 29th July 2013
|
||||
---
|
||||
* Hehe: <http://www.pixartheory.com/>
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-07-30T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/on-to-do-lists-as-direction-in-life/
|
||||
- /on-to-do-lists-as-direction-in-life/
|
||||
- /wordpress/archives/268/
|
||||
title: On to-do lists as direction in life
|
||||
---
|
||||
[Getting Things Done](https://en.wikipedia.org/wiki/Getting_Things_Done) has gathered something of a [cult following](http://web.archive.org/web/20130428015707/http://www.wired.com/techbiz/people/magazine/15-10/ff_allen? "Wired article on GTD") [archived due to [link rot][1]] since its inception. As a way of getting things done, it's pretty good - separate tasks out into small bits on your to-do list so that you have mental room free to consider the bigger picture. However, there's a certain aspect of to-do lists that I've not really seen mentioned before, and which I find to be really helpful.
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-08-04T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/new-computer-setup/
|
||||
- /new-computer-setup/
|
||||
- /wordpress/archives/36
|
||||
title: New computer setup
|
||||
---
|
||||
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-08-04T00:00:00Z"
|
||||
aliases:
|
||||
- /stumbled_across/stumbled-across-3/
|
||||
- /stumbled-across-4-august-2013/
|
||||
- /wordpress/archives/267/
|
||||
title: Stumbled across 4th August 2013
|
||||
---
|
||||
* An ad developer has misgivings: <http://seriouspony.com/blog/2013/7/24/your-app-makes-me-fat>
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-08-11T00:00:00Z"
|
||||
aliases:
|
||||
- /stumbled_across/stumbled-across-11th-august-2013/
|
||||
- /stumbled-across-11th-august-2013/
|
||||
- /wordpress/archives/301/
|
||||
title: Stumbled across 11th August 2013
|
||||
---
|
||||
* A thousand times this (EDIT 2022: the link is dead and I have no idea what I was referring to).
|
||||
|
@@ -9,6 +9,7 @@ math: true
|
||||
aliases:
|
||||
- /psychology/thinking-styles/
|
||||
- /thinking-styles/
|
||||
- /wordpress/archives/335/
|
||||
title: Thinking styles
|
||||
---
|
||||
All the way back into primary school (ages 4 to 11 years old, in case a non-Brit is reading this), we have been told repeatedly that "people learn things in different ways". There were two years in primary school when I had a teacher who was very into [Six Thinking Hats](https://en.wikipedia.org/wiki/Six_Thinking_Hats) (leading to the worst outbreak of headlice I've ever encountered) and [mind maps](https://en.wikipedia.org/wiki/Mind_map). I never understood mind maps, and whenever we were told to create a mind map, I'd make mine as linear and boxy as possible, out of simple frustration with the pointless task of making a picture of something that I already had perfectly well-set-out in my mind. I quickly learnt to correlate "making a mind map" with "being slow and inefficient at thinking". (This was back when my memory was still exceptionally good, so I wasn't really learning much at school - having read, and therefore memorised, a good children's encyclopaedia was enough for me - and hence relative to me, pretty much everyone else was slow and inefficient, because I'd already learnt the material.)
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-08-21T00:00:00Z"
|
||||
aliases:
|
||||
- /psychology/flow/
|
||||
- /my-experiences-with-flow/
|
||||
- /wordpress/archives/326/
|
||||
title: My experiences with flow
|
||||
---
|
||||
I'm in the middle of reading [Flow](https://en.wikipedia.org/wiki/Flow_(psychology)), by [Mihály Csíkszentmihályi][1], and so far, I love it. It describes the "[flow state](https://en.wikipedia.org/wiki/Flow_%28psychology%29)" of consciousness, that state of "everything is irrelevant except for the task at hand" in which time flies past without your noticing, and you don't notice hunger or thirst or people moving around you. Flow can be induced when performing a difficult task which lies within your abilities, where immediate feedback is provided. I, at least, feel characteristically exhausted after coming out of a long period of flow - but it's a good kind of mental exhaustion, much as the tiredness after a long swim is a good kind of physical exhaustion (in contrast to tiredness-after-a-long-day-of-doing-nothing, which feels sort of lazier and unwholesome). The Wikipedia page is a good enough explanation of flow that I will not describe it further here.
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-08-22T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/how-to-punt-in-cambridge/
|
||||
- /how-to-punt-in-cambridge/
|
||||
- /wordpress/archives/345/
|
||||
title: How to punt in Cambridge
|
||||
---
|
||||
[When in Cambridge][1]…
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-08-24T00:00:00Z"
|
||||
aliases:
|
||||
- /stumbled_across/stumbled-across-4/
|
||||
- /stumbled-across-24-august-2013/
|
||||
- /wordpress/archives/329/
|
||||
title: Stumbled across 24th August 2013
|
||||
---
|
||||
* The much-vaunted Hyperloop looks really cool, if it could ever be built: <https://arstechnica.com/business/2013/08/hyperloop-a-theoretical-760-mph-transit-system-made-of-sun-air-and-magnets/>
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-09-13T00:00:00Z"
|
||||
aliases:
|
||||
- /stumbled_across/stumbled-across-14th-september-2013/
|
||||
- /stumbled-across-14th-september-2013/
|
||||
- /wordpress/archives/362/
|
||||
title: Stumbled across 14th September 2013
|
||||
---
|
||||
* On the merits of silence (I wholeheartedly agree): <http://www.nytimes.com/2013/08/25/opinion/sunday/im-thinking-please-be-quiet.html>
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-09-21T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/how-to-prove-that-you-are-a-god/
|
||||
- /how-to-prove-that-you-are-a-god/
|
||||
- /wordpress/archives/434/
|
||||
title: How to prove that you are a god
|
||||
---
|
||||
I came across an interesting question while reading the blog of [Scott Aaronson][1] today. The question was as follows:
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-10-10T00:00:00Z"
|
||||
aliases:
|
||||
- /creative/plot-armour/
|
||||
- /plot-armour/
|
||||
- /wordpress/archives/445/
|
||||
title: Plot Armour
|
||||
---
|
||||
*Wherein I dabble in parodic fiction. The title refers to the TV Tropes page on [Plot Armour][1], but don't follow that link unless you first resolve not to click on any links on that page. TV Tropes is the hardest extant website from which to escape.*
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-10-11T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/meaning-what-you-say/
|
||||
- /meaning-what-you-say/
|
||||
- /wordpress/archives/450
|
||||
title: Meaning what you say
|
||||
---
|
||||
In conversation with (say, for the purposes of propagating a sterotype) humanities students, I am often struck by how imprecisely language is used, and how much confusion arises therefrom. A case in point:
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-10-13T00:00:00Z"
|
||||
aliases:
|
||||
- /psychology/training-away-mental-bias/
|
||||
- /training-away-mental-bias/
|
||||
- /wordpress/archives/455
|
||||
title: Training away mental bias
|
||||
---
|
||||
*In which I recount an experiment I have been performing. Please be aware that in this article I am in "[meaning what I say][1]" mode.*
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-10-20T00:00:00Z"
|
||||
aliases:
|
||||
- /creative/the-ravenous/
|
||||
- /the-ravenous/
|
||||
- /wordpress/archives/462/
|
||||
title: The Ravenous
|
||||
sidenotes: true
|
||||
---
|
||||
|
@@ -10,6 +10,8 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/how-to-do-analysis-questions/
|
||||
- /how-to-do-analysis-questions/
|
||||
- /archives/474/
|
||||
- /wordpress/archives/474/
|
||||
title: How to do Analysis questions
|
||||
---
|
||||
This post is for posterity, made shortly after [Dr Paul Russell][1] lectured Analysis II in Part IB of the Maths Tripos at Cambridge. In particular, he demonstrated a way of doing certain basic questions. It may be useful to people who are only just starting the study of analysis and/or who are doing example sheets in it.
|
||||
|
@@ -8,6 +8,7 @@ date: "2013-11-07T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/my-quest-for-a-new-phone/
|
||||
- /my-quest-for-a-new-phone/
|
||||
- /wordpress/archives/486/
|
||||
title: My quest for a new phone
|
||||
---
|
||||
*This post is unfinished, and may never be finished - I have decided that the Nexus 5 is sufficiently cheap, nice-looking and future-proof to outweigh the boredom of continuing the research here, especially given that such research by necessity has a very short lifespan. I am one of those people who hates shopping with a fiery passion.*
|
||||
|
@@ -9,6 +9,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/markov-chain-card-trick/
|
||||
- /markov-chain-card-trick/
|
||||
- /wordpress/archives/500/
|
||||
title: Markov Chain card trick
|
||||
---
|
||||
In my latest lecture on [Markov Chains][1] in Part IB of the Mathematical Tripos, our lecturer showed us a very nice little application of the theorem that "if a discrete-time chain is aperiodic, irreducible and positive-recurrent, then there is an invariant distribution to which the chain tends as time increases". In particular, let \\(X\\) be a Markov chain on a state space consisting of "the value of a card revealed from a deck of cards", where aces count 1 and picture cards count 10. Let \\(P\\) be randomly chosen from the range \\(1 \dots 5\\), and let \\(X_0 = P\\). Proceed as follows: define \\(X_n\\) as "the value of the \\(\sum_{i=0}^{n-1} X_i\\)-th card". Stop when the newest \\(X_n\\) would be greater than \\(52\\).
|
||||
|
@@ -7,6 +7,7 @@ comments: true
|
||||
date: "2013-11-23T00:00:00Z"
|
||||
aliases:
|
||||
- uncategorized/the-jean-paul-sartre-cookbook/
|
||||
- /wordpress/archives/506/
|
||||
title: The Jean-Paul Sartre Cookbook
|
||||
---
|
||||
> Many thanks to the [Guru Bursill-Hall][1] for bringing this tract to my attention through his weekly History of Maths bulletins. It was originally written in 1987 by Marty Smith, according to the Internet.
|
||||
|
@@ -24,7 +24,7 @@ The contents of soil
|
||||
|
||||
* we eat fungi
|
||||
|
||||
* * we don't just eat mushrooms, we also are starting to eat <a title="Quorn Wikipedia page" href="https://en.wikipedia.org/wiki/Quorn">Quorn</a> etc
|
||||
* * we don't just eat mushrooms, we also are starting to eat [Quorn](https://en.wikipedia.org/wiki/Quorn) etc
|
||||
|
||||
* we eat fungi - more specifically, the reproductive organs of the mycelium
|
||||
|
||||
|
@@ -8,6 +8,7 @@ date: "2014-02-16T00:00:00Z"
|
||||
aliases:
|
||||
- /creative/rage-rage-against-the-poets-hardest-sell/
|
||||
- /rage-rage-against-the-poets-hardest-sell/
|
||||
- /wordpress/archives/563/
|
||||
title: Rage, rage against the poet’s hardest sell
|
||||
---
|
||||
I feel that I can write a sonnet well.
|
||||
|
@@ -8,6 +8,7 @@ date: "2014-03-20T00:00:00Z"
|
||||
aliases:
|
||||
- /uncategorized/a-roundup-of-some-board-games/
|
||||
- /a-roundup-of-some-board-games/
|
||||
- /wordpress/archives/576/
|
||||
title: A roundup of some board games
|
||||
---
|
||||
It has been commented to me that it's quite hard to find out (on the Internet) what different games involve. For instance, [Agricola][1] is a game about farming (and that's easy to find out), but what you actually do while playing it is not easy to discover. Here, then, is a brief overview of some games.
|
||||
|
@@ -10,6 +10,8 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/proof_discovery/how-to-discover-the-contraction-mapping-theorem/
|
||||
- /how-to-discover-the-contraction-mapping-theorem/
|
||||
- /archives/584/
|
||||
- /wordpress/archives/584/
|
||||
title: How to discover the Contraction Mapping Theorem
|
||||
---
|
||||
A little while ago I set myself the exercise of stating and proving the [Contraction Mapping Theorem][1]. It turned out that I mis-stated it in three different aspects ("contraction", "non-empty" and "complete"), but I was able to correct the statement because there were several points in the proof where it was very natural to do a certain thing (and where that thing turned out to rely on a correct statement of the theorem).
|
||||
|
@@ -10,6 +10,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/proof_discovery/discovering-a-proof-of-heine-borel/
|
||||
- /discovering-a-proof-of-heine-borel/
|
||||
- /wordpress/archives/589/
|
||||
title: Discovering a proof of Heine-Borel
|
||||
---
|
||||
I'm running through my Analysis proofs, trying to work out which ones are genuinely hard and which follow straightforwardly from my general knowledge base. I don't find the [Heine-Borel Theorem][1] "easy" enough that I can even forget its statement and still prove it (like [I can with the Contraction Mapping Theorem][2]), but it turns out to be easy in the sense that it follows simply from all the theorems I already know. Here, then, is my attempt to discover a proof of the theorem, using as a guide all the results I know but can't necessarily prove without lots of effort.
|
||||
|
@@ -9,6 +9,7 @@ math: true
|
||||
aliases:
|
||||
- /uncategorized/useful-conformal-mappings/
|
||||
- /useful-conformal-mappings/
|
||||
- /wordpress/archives/594/
|
||||
title: Useful conformal mappings
|
||||
---
|
||||
This post is to be a list of conformal mappings, so that I can get better at answering questions like "Find a conformal mapping from \<this domain\> to \<this domain\>". The following Mathematica code is rough-and-ready, but it is designed to demonstrate where a given region goes under a given transformation.
|
||||
|
@@ -10,6 +10,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/sample-topology-question/
|
||||
- /sample-topology-question/
|
||||
- /wordpress/archives/600/
|
||||
title: Sample topology question
|
||||
---
|
||||
As part of the recent series on how I approach maths problems, I give another one here (question 14 on the Maths Tripos IB 2007 paper 4). The question is:
|
||||
|
@@ -9,6 +9,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/cayley-hamilton-theorem/
|
||||
- /cayley-hamilton-theorem/
|
||||
- /wordpress/archives/606/
|
||||
title: Cayley-Hamilton theorem
|
||||
---
|
||||
This is to detail a much easier proof (at least, I find it so) of [Cayley-Hamilton][1] than the ones which appear on the Wikipedia page. It only applies in the case of complex vector spaces; most of the post is taken up with a proof of a lemma about complex matrices that is very useful in many contexts.
|
||||
|
@@ -10,6 +10,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/proof_discovery/sequentially-compact-iff-compact/
|
||||
- /sequentially-compact-iff-compact/
|
||||
- /wordpress/archives/620/
|
||||
title: Sequentially compact iff compact
|
||||
---
|
||||
[Prof Körner][1] told us during the [IB Metric and Topological Spaces][2] course that the real meat of the course (indeed, its hardest theorem) was "a metric space is sequentially compact iff it is compact". At the moment, all I remember of this result is that one direction requires Lebesgue's lemma (whose statement I don't remember) and that the other direction is quite easy. I'm going to try and discover a proof - I'll be honest when I have to look things up.
|
||||
|
@@ -10,6 +10,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/proof_discovery/discovering-a-proof-of-sylvesters-law-of-inertia/
|
||||
- /discovering-a-proof-of-sylvesters-law-of-inertia/
|
||||
- /wordpress/archives/591/
|
||||
title: Discovering a proof of Sylvester's Law of Inertia
|
||||
---
|
||||
*This is part of what has become a series on discovering some fairly basic mathematical results, and/or discovering their proofs. It's mostly intended so that I start finding the results intuitive - having once found a proof myself, I hope to be able to reproduce it without too much effort in the exam.*
|
||||
|
@@ -9,6 +9,7 @@ math: true
|
||||
aliases:
|
||||
- /mathematical_summary/proof-that-symmetric-matrices-are-diagonalisable/
|
||||
- /proof-that-symmetric-matrices-are-diagonalisable/
|
||||
- /wordpress/archives/658/
|
||||
title: Proof that symmetric matrices are diagonalisable
|
||||
---
|
||||
This comes up quite frequently, but I've been stuck for an easy memory-friendly way to do this. I trawled through the 1A Vectors and Matrices course notes, and found the following mechanical proof. (It's not a discovery-proof - I looked it up.)
|
||||
|
@@ -7,6 +7,7 @@ comments: true
|
||||
date: "2018-02-03T00:00:00Z"
|
||||
title: Infinitesimals as an idea that took a long time
|
||||
summary: Answering the question, "Which mathematical ideas took a long time to define rigorously?".
|
||||
math: true
|
||||
---
|
||||
|
||||
*This is my answer to the same [question posed on the Mathematics Stack Exchange](https://math.stackexchange.com/q/2633847/259262). It is therefore licenced under [CC-BY-SA 3.0](https://creativecommons.org/licenses/by-sa/3.0/).*
|
||||
|
@@ -8,6 +8,7 @@ comments: true
|
||||
date: "2021-03-17T00:00:00Z"
|
||||
title: Rewriting the Technical Interview, in Mathematica
|
||||
summary: "An exploration into reaching into the internals of Mathematica to natively evaluate C code."
|
||||
math: true
|
||||
---
|
||||
|
||||
*Cross-posted at the [G-Research official blog](https://www.gresearch.co.uk/article/rewriting-the-technical-interview-in-mathematica/).*
|
||||
|
@@ -9,7 +9,7 @@ title: Crates (existentials in F#)
|
||||
summary: "An introduction to the crate pattern for representing existential quantification in F#."
|
||||
---
|
||||
|
||||
The usual blog post introducing crates in F# is [G-Research's](https://www.gresearch.co.uk/article/squeezing-more-out-of-the-f-type-system-introducing-crates/).
|
||||
The usual blog post introducing crates in F# is [G-Research's](https://web.archive.org/web/20220421204450/https://www.gresearch.co.uk/blog/article/squeezing-more-out-of-the-f-type-system-introducing-crates/).
|
||||
However, I believe that post approaches things from a much more abstract perspective than is useful.
|
||||
Instead, I will approach them by starting from the concrete.
|
||||
|
||||
|
@@ -56,7 +56,7 @@ To discover checks along both a vertical and a diagonal, it's necessary to move
|
||||
* Castling moves two pieces, but we already know the black king has moved, so it wasn't Black castling.
|
||||
* A standard capture does move two pieces, but it leaves the capturing piece in the same place as the captured piece, so it can't reveal two checks.
|
||||
* Moving a piece without capturing moves one piece, and can only reveal one check.
|
||||
* Promoting a pawn could conceivably have had this effect if this were all happening on the back rank: a pawn promoting to a knight can both discover a check from a bishop and cause check with the new knight. But both the pieces which are giving check are in the middle of the board, with neither on the back rank.
|
||||
* Promoting a pawn could conceivably have had this effect if this were all happening on the back rank: a pawn promoting to a knight can both discover a check from a rook and cause check with the new knight. But both the pieces which are giving check are in the middle of the board, with neither on the back rank.
|
||||
|
||||
There is exactly one other kind of move in the game: capturing en passant.
|
||||
This can clear a file and a diagonal at the same time.
|
||||
|
29
hugo/content/posts/2024-01-19-starting-suspended-process.md
Normal file
29
hugo/content/posts/2024-01-19-starting-suspended-process.md
Normal file
@@ -0,0 +1,29 @@
|
||||
---
|
||||
lastmod: "2024-01-19T18:51:00.0000000+00:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2024-01-19T18:51:00.0000000+00:00"
|
||||
title: Starting a suspended process
|
||||
summary: "How to start a process into a suspended state on Linux"
|
||||
---
|
||||
|
||||
(This is not fancy and is probably common knowledge, but I only learned how to do it today.)
|
||||
|
||||
Say you want to start a process, then log its PID, then allow it to run (so that if it starts logging to the console, you're guaranteed to have done all your logging before it starts polluting stdout).
|
||||
|
||||
This is easy on Windows: [`CreateProcess` accepts `CREATE_SUSPENDED`](https://learn.microsoft.com/en-us/windows/win32/procthread/process-creation-flags).
|
||||
But on Linux, it's not obvious.
|
||||
|
||||
1. The parent sets up a signal handler for SIGUSR1 (say), so that it can know when the child is ready.
|
||||
1. The parent forks, then waits for SIGUSR1.
|
||||
1. The child sets up a signal handler for SIGUSR2, so that it can know when the parent is ready.
|
||||
1. The child sends SIGUSR1 to its parent to signal readiness, then waits for SIGUSR2.
|
||||
1. The parent does whatever it wants to do with the now-running child process (whose PID it now knows).
|
||||
1. The parent sends SIGUSR2 to the child.
|
||||
1. The child `exec`s into the process that it actually wanted to start all along.
|
||||
1. The parent unregisters the SIGUSR1 handler.
|
||||
|
||||
Note that [forked processes have the same signal handlers](https://man7.org/linux/man-pages/man7/signal.7.html) as the parent, but those signal handlers are blatted by an `exec`.
|
||||
|
||||
(Good lord is this not investment advice. It's hard to express just how much of a novice I am at signal handling.)
|
65
hugo/content/posts/2024-03-01-no-confusion.md
Normal file
65
hugo/content/posts/2024-03-01-no-confusion.md
Normal file
@@ -0,0 +1,65 @@
|
||||
---
|
||||
lastmod: "2024-03-01T13:51:00.0000000+00:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2024-03-01T13:51:00.0000000+00:00"
|
||||
title: Why does no-confusion use equality rather than a recursive call?
|
||||
summary: "A question about the definition of a no-confusion type."
|
||||
---
|
||||
|
||||
## Question
|
||||
|
||||
Conor McBride defined the no-confusion property of `Nat` on page 11 of [A Polynomial Testing Principle](https://personal.cis.strath.ac.uk/conor.mcbride/PolyTest.pdf) as:
|
||||
|
||||
```agda
|
||||
NoConf : Nat → Nat → Set
|
||||
NoConf ze ze = 1
|
||||
NoConf ze (su y) = 0
|
||||
NoConf (su x ) ze = 0
|
||||
NoConf (su x ) (su y) = x ≡ y
|
||||
```
|
||||
|
||||
Why was this defined that way, rather than the following way which works without dependencies?
|
||||
|
||||
```agda
|
||||
NoConf : Nat → Nat → Set
|
||||
NoConf ze ze = 1
|
||||
NoConf ze (su y) = 0
|
||||
NoConf (su x) ze = 0
|
||||
NoConf (su x) (su y) = NoConf x y
|
||||
```
|
||||
|
||||
## Context
|
||||
|
||||
I still had this question quite near the end of my working through A Polynomial Testing Principle, and it didn't seem to have been answered: I managed to get through most of the paper without serious problems arising from my different definition.
|
||||
|
||||
## Answer
|
||||
|
||||
Conor immediately follows the definition of `NoConf` by defining a canonical way to construct a `NoConf`:
|
||||
|
||||
```agda
|
||||
noConf : {x y : Nat} -> x ≡ y → NoConf x y
|
||||
noConf {zero} refl = record {}
|
||||
noConf {suc n} {.(suc n)} refl = refl
|
||||
```
|
||||
|
||||
If you do it my way instead, you end up unable to use this canonical construction to do proofs by no-confusion.
|
||||
You probably find yourself proving that `succ` is injective manually, and then using that directly instead of via the `NoConf`; which defeats the entire purpose of packaging up the no-confusion property into a type.
|
||||
|
||||
For example:
|
||||
|
||||
```agda
|
||||
+cancel : (a b : Nat) {c d : Nat} -> a ≡ b -> (a + c) ≡ (b + d) -> c ≡ d
|
||||
+cancel zero .zero refl sums_equal = sums_equal
|
||||
+cancel (succ a) (succ .a) refl sums_equal = +cancel a a refl {!!}
|
||||
```
|
||||
|
||||
What should go here?
|
||||
We need something of type `a + c ≡ a + d`, but all we have in scope is the input `succ (a + c) ≡ succ (a + d)`.
|
||||
That is, we need the no-confusion property for `succ`; which suggests that `NoConf` should somehow contain an equality type, so that we can use it!
|
||||
|
||||
## Commentary
|
||||
|
||||
I think I only had this question because `Nat` is such a simple type.
|
||||
If there were more constructors and more cases in each pattern-match, it would have been obvious straight away that I had deleted the entire point of the `NoConf` construction.
|
80
hugo/content/posts/2024-03-13-northern-lights-trip.md
Normal file
80
hugo/content/posts/2024-03-13-northern-lights-trip.md
Normal file
@@ -0,0 +1,80 @@
|
||||
---
|
||||
lastmod: "2024-03-13T12:18:00.0000000+00:00"
|
||||
author: patrick
|
||||
date: "2024-03-13T12:18:00.0000000+00:00"
|
||||
title: Trip to Tromsø
|
||||
summary: "Travelogue of my trip to Norway."
|
||||
gallery: true
|
||||
---
|
||||
|
||||
# Friday (2024-03-08)
|
||||
|
||||
* Wagamama's at Gatwick
|
||||
* Toblerone
|
||||
* Plane to Tromsø!
|
||||
* Landed 2340 local time
|
||||
* About an hour between landing and getting a taxi
|
||||
* First sight of tunnel system under Tromsø mountains
|
||||
{{< foldergallery src="images/galleries/Tromso2024/1-AirportOutbound" >}}
|
||||
|
||||
Northern Lights visible from plane; they turned off the cabin lights so we could see.
|
||||
{{< foldergallery src="images/galleries/Tromso2024/2-NorthernLightsFromPlane" summary="Northern Lights viewed from the plane" >}}
|
||||
|
||||
# Saturday (2024-03-09)
|
||||
|
||||
* Breakfast (great smoked salmon and pickled herring, scrambled eggs, mushrooms, beans, bread, butter, apple jam)
|
||||
* Ran into some lost tourists, resulting in a rainy walk over [bridge to east](https://en.wikipedia.org/wiki/Troms%C3%B8_Bridge)
|
||||
{{< foldergallery src="images/galleries/Tromso2024/3-TromsoWalk" >}}
|
||||
* [Arctic cathedral](https://www.ishavskatedralen.no/en/the-arctic-cathedral/)
|
||||
{{< foldergallery src="images/galleries/Tromso2024/4-Cathedral" >}}
|
||||
* [Cable car](https://www.fjellheisen.no/engelsk/fjellheisen-home) up mountain, vue panoramique
|
||||
{{< foldergallery src="images/galleries/Tromso2024/5-Viewpoint" >}}
|
||||
* Back to hotel, sauna
|
||||
* Cinnamon bun and golden milk at [Pust Kafé](https://www.kafe.pust.io/) (maximum bougie London energy)
|
||||
* [Polar museum](https://en.uit.no/tmu/polarmuseet)
|
||||
{{< foldergallery src="images/galleries/Tromso2024/6-PolarMuseum" >}}
|
||||
* Carrot cake and Lady Grey tea at hotel
|
||||
{{< foldergallery src="images/galleries/Tromso2024/7-Cake" >}}
|
||||
* Tea at hotel, sauna again and jacuzzi outside
|
||||
{{< foldergallery src="images/galleries/Tromso2024/8-Jacuzzi" >}}
|
||||
|
||||
# Sunday (2024-03-10)
|
||||
|
||||
{{< foldergallery src="images/galleries/Tromso2024/9-BreakfastSunday" summary="Breakfast photos" >}}
|
||||
|
||||
* [Snowshoe walk](https://www.wanderingowl.com/tour/tromso/winter-walk-on-the-wild-side-troms%C3%B8/P7YEHB/) at Kattfjordeidet
|
||||
|
||||
{{< foldergallery src="images/galleries/Tromso2024/10-Snowshoes" summary="My photos" >}}
|
||||
{{< foldergallery src="images/galleries/Tromso2024/11-SnowshoesOfficial" summary="Photos from our guide Evgeni" >}}
|
||||
|
||||
* [Aurora tour](https://auroratour.no/about-us/), ending up around [Skibotn](https://en.wikipedia.org/wiki/Skibotn); back around 0230 local time
|
||||
|
||||
{{< foldergallery src="images/galleries/Tromso2024/12-NorthernLights" summary="My photos" >}}
|
||||
{{< foldergallery src="images/galleries/Tromso2024/13-NorthernLightsOfficial" summary="Photos from our guide Marcin" >}}
|
||||
|
||||
# Monday (2024-03-11)
|
||||
|
||||
* [Husky sledding](https://tromsodogsledding.com/)
|
||||
{{< foldergallery src="images/galleries/Tromso2024/14-Huskies" >}}
|
||||
* Jacuzzi again
|
||||
{{< foldergallery src="images/galleries/Tromso2024/15-JacuzziAgain" >}}
|
||||
|
||||
# Tuesday (2024-03-12)
|
||||
|
||||
* [Glass blowing workshop](https://www.blaast.no/)
|
||||
{{< foldergallery src="images/galleries/Tromso2024/18-GlassBlower" >}}
|
||||
* More buns at Pust
|
||||
{{< foldergallery src="images/galleries/Tromso2024/17-Cake" >}}
|
||||
* [Aquarium](https://polaria.no/en/) (featuring seals!)
|
||||
{{< foldergallery src="images/galleries/Tromso2024/16-Aquarium" >}}
|
||||
|
||||
* Plane to Luton
|
||||
* Arrived home 0120 GMT
|
||||
|
||||
# Stats
|
||||
|
||||
* Nikoli-style puzzles solved: 48
|
||||
* Saunas: 7
|
||||
* Jacuzzis: 2
|
||||
* Cardamom buns: 2
|
||||
* Items of clothing lost: 1
|
41
hugo/content/posts/2024-03-14-yaml-superset-json.md
Normal file
41
hugo/content/posts/2024-03-14-yaml-superset-json.md
Normal file
@@ -0,0 +1,41 @@
|
||||
---
|
||||
lastmod: "2024-04-14T00:12:00.0000000+00:00"
|
||||
author: patrick
|
||||
date: "2024-03-14T00:12:00.0000000+00:00"
|
||||
title: YAML is not a superset of JSON
|
||||
summary: "All the reasons I know for why YAML is not a superset of JSON."
|
||||
categories:
|
||||
- programming
|
||||
---
|
||||
|
||||
I keep forgetting, so here we go.
|
||||
Please let me know if you have any others!
|
||||
|
||||
## Treatment of e.g. `1e2`
|
||||
|
||||
JSON treats the value `1e2` a number, of course, because it's not in quote marks.
|
||||
YAML fails to parse it as a number so silently falls back to treating it as a string.
|
||||
|
||||
```python
|
||||
>>> import yaml
|
||||
>>> import json
|
||||
|
||||
>>> yaml.safe_load('{"a": 1e2}')
|
||||
{'a': '1e2'}
|
||||
|
||||
>>> json.loads('{"a": 1e2}')
|
||||
{'a': 100.0}
|
||||
```
|
||||
|
||||
## Tabs as indentation
|
||||
|
||||
YAML does not permit tabs to be used as indentation.
|
||||
|
||||
```python
|
||||
>>> yaml.load ('{\n "list": [\n {},\n\t{}\n ]\n}')
|
||||
# yaml.scanner.ScannerError: while scanning for the next token
|
||||
# found character '\t' that cannot start any token
|
||||
|
||||
>>> json.loads('{\n "list": [\n {},\n\t{}\n ]\n}')
|
||||
{'list': [{}, {}]}
|
||||
```
|
51
hugo/content/posts/2024-03-27-chatgpt-and-programming.md
Normal file
51
hugo/content/posts/2024-03-27-chatgpt-and-programming.md
Normal file
@@ -0,0 +1,51 @@
|
||||
---
|
||||
lastmod: "2024-03-27T12:01:00.0000000+00:00"
|
||||
author: patrick
|
||||
date: "2024-03-27T12:01:00.0000000+00:00"
|
||||
title: ChatGPT's effect on my programming
|
||||
summary: "After a decent while programming with ChatGPT, I'm not sure it's even a net positive on my ability."
|
||||
categories:
|
||||
- programming
|
||||
---
|
||||
|
||||
For almost a year now, I've had access to ChatGPT 4.
|
||||
In that time, I've used it pretty extensively, I have some large custom instructions to try and get it to be precise and accurate, and so forth.
|
||||
After a particularly gruelling weekend [getting my Neovim environment up to scratch](https://github.com/Smaug123/nix-dotfiles/pull/38/files), in which I used ChatGPT for approximately 70 distinct chats (some of which were quite long), I decided to put down some thoughts.
|
||||
|
||||
# Where is ChatGPT good?
|
||||
|
||||
* Language-to-language translation. Bog-standard line-for-line translation of Vimscript to Lua went pretty much perfectly each time. However, ChatGPT never flagged places where I was doing things that were simply unnecessary or wrong, and a human who understood Neovim would have done. (For example, there are Vim options which make no sense in Neovim and are silently ignored.)
|
||||
* Fixing syntax, if you don't have a decent language server doing that for you. It's worse than a proper language server, but better than nothing.
|
||||
* Explaining what code is doing, if you don't want to dig too deep but just want a general gist.
|
||||
|
||||
# On using ChatGPT to write code for me
|
||||
|
||||
This is where ChatGPT is really pernicious.
|
||||
It's *so* tempting to use its output without putting in the work of building a mental model, but this *never* works.
|
||||
|
||||
I think on balance the presence of ChatGPT might actually have slowed down my Neovim migration, because it allowed me to get 80% of the way there without learning the domain at all.
|
||||
To get the final 20%, you need to understand the domain, but it takes a great deal of discipline to do so when ChatGPT is ready to give you implementations that almost work without any thought at all.
|
||||
I don't really have that discipline, and I think I would have been better off simply with a search engine.
|
||||
|
||||
(The [Neovim API docs](https://neovim.io/doc/user/api.html) are generally awful.
|
||||
They'd be adequate if they were documenting an API in a language with a decent type system, but Lua is untyped so the amount of documentation required is at least 3x bigger.
|
||||
The poor quality of the documentation adds extra friction if you want to learn what you're doing: it's more attractive to just not learn, if learning involves reading large amounts of an unfamiliar language in the guts of the implementation of a system you don't know.
|
||||
[Looking](https://github.com/dotnet/dotnet-api-docs/pull/9753) [at](https://github.com/dotnet/dotnet-api-docs/pull/9595) [you](https://github.com/dotnet/runtime/pull/95956) [too](https://github.com/dotnet/docs/pull/35171), [.NET](https://github.com/dotnet/dotnet-api-docs/pull/9394).)
|
||||
|
||||
I can imagine that ChatGPT 5 would be adequate for writing code for me, but right now I think the time it saves on starting you up (and getting you 80% of the way) is more than eaten up by the amount of learning/fixing/debugging that it makes you do in the future.
|
||||
There's a really high cost to not having a good mental model of what you're doing, and ChatGPT makes it far too easy to put yourself in that debt.
|
||||
|
||||
# What about Opus?
|
||||
|
||||
Claude 3 Opus [has had a lot of hype](https://thezvi.substack.com/p/on-claude-30) for its coding abilities, but over the past few days I've been running side-by-side comparisons with ChatGPT and there's simply no contest.
|
||||
ChatGPT gives better answers: they're more likely to be correct, and they're almost always more useful even when wrong.
|
||||
I have not yet got a single correct code snippet out of Opus.
|
||||
|
||||
# What I actually want from such a system
|
||||
|
||||
A really really good search engine, which has consumed all the relevant documentation.
|
||||
Claude 3 Opus's chat interface won't take more than 16% of the [home-manager documentation](https://nix-community.github.io/home-manager/options.xhtml), and the [POSIX spec](https://pubs.opengroup.org/onlinepubs/9699919799.2018edition/) is orders of magnitude too big.
|
||||
(Perhaps I need to suck it up and start using the Claude API.)
|
||||
|
||||
What I need is to ask "Where precisely in the spec should I look to discover the behaviour in this particular case?", "Under what configuration option does one configure this thing?", "Can you show me some code in the wild where someone is successfully using this?".
|
||||
|
148
hugo/content/posts/2024-04-13-yoneda.md
Normal file
148
hugo/content/posts/2024-04-13-yoneda.md
Normal file
@@ -0,0 +1,148 @@
|
||||
---
|
||||
lastmod: "2024-04-13T16:58:00.0000000+01:00"
|
||||
author: patrick
|
||||
date: "2024-04-13T16:58:00.0000000+01:00"
|
||||
title: The Yoneda lemma
|
||||
summary: "Another attempt to explain that the Yoneda lemma is actually intuitive."
|
||||
math: true
|
||||
---
|
||||
|
||||
*Read this post with a pencil and paper! Draw pictures!*
|
||||
|
||||
# What is a diagram?
|
||||
|
||||
A functor \\(\mathcal{C} \to \mathcal{D}\\) is basically a way of identifying a copy of \\(\mathcal{C}\\) inside \\(\mathcal{D}\\).
|
||||
That is, it's a specification of an object of \\(\mathcal{D}\\) for every object of \\(\mathcal{C}\\), and corresponding morphisms inside \\(\mathcal{D}\\) which must go between the right objects for this to be a copy of \\(\mathcal{C}\\).
|
||||
|
||||
If we specialise to the category of sets, a functor \\(\mathcal{C} \to \mathrm{\mathbf{Set}}\\) is a set for every object of \\(\mathcal{C}\\), and functions between those objects.
|
||||
|
||||
Imagine \\(\mathcal{C}\\) as some kind of abstract template of a universe of types; then a functor \\(\mathcal{C} \to \mathrm{\mathbf{Set}}\\) is an *instantation* of that universe of types.
|
||||
For example, perhaps we decide that this particular object of \\(\mathcal{C}\\) will be instantiated to \\(\mathbb{N}\\), and that particular object will be instantiated to \\(\mathrm{Bool}\\), and this arrow between them will be instantiated to \\(n \mapsto \mathrm{isEven}(n)\\).
|
||||
|
||||
Recall what the functor laws in this context are:
|
||||
|
||||
* The functor respects endpoints of morphisms: if \\(f: A \to B\\) in \\(\mathcal{C}\\), then \\(Ff : FA \to FB\\) in \\(\mathrm{\mathbf{Set}}\\). That is, we really are identifying a copy of \\(\mathcal{C}\\) in \\(\mathrm{\mathbf{Set}}\\): we're not actively breaking the structure in translation. If the template \\(\mathcal{C}\\) tells us how to get from \\(A\\) to \\(B\\), then we can do that in our instantiation.
|
||||
* The identity morphism is always taken to the identity function. (This one's fairly self-explanatory.)
|
||||
* The functor respects compositions: \\(F(g \circ f) = F(g) \circ F(f)\\). That is, if the template tells us we can do \\(f\\) then \\(g\\), we can also do that in our instantiation; moreover, if our template tells us two paths are equal, then they're also equal in the instantiation.
|
||||
|
||||
Note what we're *not* requiring of our instantiations: that they're somehow "fully preserving all the structure".
|
||||
If \\(\mathcal{C}\\) has two objects \\(A\\) and \\(B\\), we're perfectly happy to instantiate both of them to the same type, as long as all the arrows keep composing correctly.
|
||||
In particular, for example, our instantiation might squash away arbitrarily much of the category's structure: every category has a trivial instantiation to the universe where there's only one set \\(\emptyset\\), and only one arrow \\(\mathrm{id} : \emptyset \to \emptyset\\).
|
||||
|
||||
# Homomorphisms between diagrams
|
||||
|
||||
Thinking of a diagram as some kind of instantiation of an abstract template, we can define homomorphisms (that is, structure-preserving maps) between them.
|
||||
The concrete example of the "trivial instantiation" in the previous section may help you imagine how these homomorphisms should work.
|
||||
|
||||
If we have two diagrams \\(F, G : \mathcal{C} \to \mathrm{\mathbf{Set}}\\), we can define some particular homomorphism \\(\alpha\\) from the \\(F\\)-instantiation to the \\(G\\)-instantiation by sending every concrete \\(F\\)-type to a corresponding concrete \\(G\\)-type.
|
||||
In order to deserve the name "homomorphism", it had better preserve the structure: that is, if \\(x : FA \to FB\\) is a function in our concrete instantiation of \\(\mathcal{C}\\), then it had better be the case that our homomorphism \\(\alpha\\) takes \\(x\\) to a suitable corresponding function \\(GA \to GB\\) in the \\(G\\)-instantiation.
|
||||
|
||||
In fact, every function in the \\(F\\)-instantiation is an image of a morphism from \\(\mathcal{C}\\) (that is, we defined our instantiations so that we weren't considering any extraneous functions there might be between the sets), so instead of considering the set of all general \\(x : FA \to FB\\), we only need to consider the set of all \\(Ff : FA \to FB\\).
|
||||
Similarly, the functions \\(GA \to GB\\) are all of the form \\(Gg : GA \to GB\\) for some morphism \\(g\\) of \\(\mathcal{C}\\).
|
||||
|
||||
That is, a homomorphism from diagram \\(F\\) to diagram \\(G\\) is:
|
||||
|
||||
* an assignment, for each \\(A : |\mathcal{C}|\\), of some \\(GX\\) corresponding to \\(FA\\) (but note that in fact this \\(GX\\) must be \\(GA\\) to meet the later requirements, so there was no choice of target here);
|
||||
* an assignment, for each \\(\mathcal{C}\\)-morphism \\(f : A \to B\\), of some \\(Gg\\) corresponding to \\(Ff\\) (but note that this must be \\(Gf\\) to meet the later requirements, so similarly there's no choice of target here);
|
||||
* proofs that the homomorphism respects the basic structure of the category: if \\(f : A \to B\\) in \\(\mathcal{C}\\), then the map \\(Ff : FA \to FB\\) gets taken to the map \\(Gf : GA \to GB\\);
|
||||
* proofs that the homomorphism "respects moving around within the diagram": "moving around in \\(F\\) and then taking the homomorphism over to \\(G\\)" should be the same as "taking the homomorphism to \\(G\\) and then making the same movement in \\(G\\)".
|
||||
|
||||
The homomorphism \\(\alpha\\) is pretty constrained: the only freedom is to decide exactly *how* \\(\alpha\\) sends \\(FA\\) to \\(GA\\) for each \\(A : |\mathcal{C}|\\).
|
||||
|
||||
We have a special name for these structure-preserving maps between diagrams: we call them *natural transformations*.
|
||||
They provide us with a way of mapping between instantiations of the abstract theory specified by \\(\mathcal{C}\\).
|
||||
|
||||
# There are some very special examples of diagrams
|
||||
|
||||
We saw already that there are some pretty degenerate diagrams: the one which has every object going to the empty set, for instance.
|
||||
That diagram has somehow "thrown away all the information" from the category.
|
||||
|
||||
Diagrams can also be pretty complex.
|
||||
For example, take the diagrams for the very simple category that contains just one object and one arrow.
|
||||
Literally every set in the universe (together with the corresponding identity function on that set) is a diagram for this category!
|
||||
Most of those diagrams involve us ignoring tons of structure.
|
||||
|
||||
For example, there are \\(\mathbb{R}\\)-many functions \\(\mathbb{N} \to \mathbb{N}\\), so we're having to ignore uncountably many functions in the category of sets if we take the diagram consisting of "\\(\mathbb{N}\\) and its sole identity function".
|
||||
This diagram wants to have way more structure than there was in \\(\mathcal{C}\\), and it's only by shutting our eyes and ignoring the set-structure we don't care about that we recover anything that looks remotely like \\(\mathcal{C}\\)!
|
||||
|
||||
It turns out there's a sweet spot of diagrams with "exactly the amount of structure \\(\mathcal{C}\\) specified, and no more".
|
||||
The construction is: start with only one element in \\(FA\\) (for some \\(A\\)), and then chase through everything else that \\(\mathcal{C}\\) says should exist (which may involve adding in more elements of \\(FA\\) if there are morphisms telling us there should be more).
|
||||
|
||||
What does \\(\mathcal{C}\\) say should exist?
|
||||
If \\(f : A \to B\\) is a morphism of \\(\mathcal{C}\\), and if \\(a \in FA\\) is an element of our concrete instantiation of the object \\(A\\), we really want there to be a distinct object \\((Ff)(a) \in FB\\) so that we've preserved the information "\\(f\\) was a morphism \\(A \to B\\)".
|
||||
(If there weren't any such member of \\(FB\\), then our diagram has lost information: we might not be able to distinguish any more whether there was a morphism \\(f\\) in \\(\mathcal{C}\\) or not, just by looking at our instantiation \\(F\\).)
|
||||
|
||||
So if we assume there's an element \\(a \in FA\\), then for every \\(B \in |\mathcal{C}|\\), we can deduce the existence of elements \\((Ff)(a) \in B\\) for every \\(f : A \to B\\) in \\(\mathcal{C}\\); and they'd better all be distinct if we couldn't prove them to be equal in \\(\mathcal{C}\\).
|
||||
|
||||
Formally: for each object \\(A\\) of \\(\mathcal{C}\\), we can define a diagram \\(\mathrm{Rep}_A : \mathcal{C} \to \mathrm{\mathbf{Set}}\\) given by sending each \\(B\\) to the set of all morphisms in \\(\mathcal{C}\\) which go from \\(A\\) to \\(B\\).
|
||||
This somehow "captures exactly all the structure that \\(\mathcal{C}\\) said \\(A\\) has".
|
||||
(I've used the symbol \\(\mathrm{Rep}\\) to denote these diagrams, because the category-theoretic term for a diagram isomorphic to one of these is "*representable functor*".)
|
||||
|
||||
Note that I haven't yet written down the functions in these concrete instantiations of \\(\mathcal{C}\\); there's only one thing it could plausibly be.
|
||||
Given \\(f : B \to C\\) a morphism of \\(\mathcal{C}\\), the corresponding function \\(\mathrm{Rep}_A(f) : \mathrm{Rep}_A(B) \to \mathrm{Rep}_A(C)\\) (that is, the function \\(\mathrm{Rep}_A(f) : \langle \text{morphisms $A \to B$ in $\mathcal{C}$} \rangle \to \langle\text{morphisms $A \to C$ in $\mathcal{C}$}\rangle\\)) is defined to be given by composing with \\(f\\): we send \\(g : A \to B\\) to \\(f \circ g : A \to C\\).
|
||||
|
||||
These particular diagrams, the *representable functors* (one for every object in \\(\mathcal{C}\\)), together tell you everything there is to know about the category.
|
||||
(That is kind of intuitive, by their definition as "the sets of morphisms": we can list out every morphism in the category, just by writing down every element of every object in each of these concrete instantiations.)
|
||||
What might be less intuitive and requires a bit more thinking is their characterisation as "define \\(F\\) by supposing there's some single member \\(a \in FA\\), and then seeing what else is forced to exist"; you might find it helps to write out a concrete example of performing this operation for some particular tiny categories.
|
||||
|
||||
# Homomorphisms of the representable functors
|
||||
|
||||
The representable functors contain so little structure that their homomorphisms are forced to be quite simple.
|
||||
|
||||
Given any diagram \\(G : \mathcal{C} \to \mathrm{\mathbf{Set}}\\), and given any \\(A \in |\mathcal{C}|\\), we can precisely characterise the homomorphisms \\(\mathrm{Rep}_A \to G\\).
|
||||
Remember, \\(\mathrm{Rep}_A\\) was defined by supposing there's just one "free generator" element \\(a \in \mathrm{Rep}_A(A)\\), and then throwing into every object all the other elements that are forced to exist by morphisms of \\(\mathcal{C}\\).
|
||||
So intuitively speaking it should be the case that a homomorphism \\(\mathrm{Rep}_A \to G\\) is precisely defined by where that one element \\(a\\) is sent; once we've defined that, *everything* else in our instantiation \\(\mathrm{Rep}_A\\) should be forced by the "preserves morphism structure" property of homomorphisms.
|
||||
|
||||
This reasoning is made formal in the *Yoneda lemma*:
|
||||
|
||||
> Pick an arbitrary diagram \\(G : \mathcal{C} \to \mathrm{\mathbf{Set}}\\). Then for every object \\(A : |\mathcal{C}|\\), the homomorphisms from the diagram \\(\mathrm{Rep}_A\\) to \\(G\\) are precisely in correspondence with the elements of \\(G(A)\\). Moreover, this correspondence is natural in both \\(A\\) and \\(G\\).
|
||||
|
||||
Intuition: the homomorphisms are precisely defined by what happens to the "free generator", and we can get such a homomorphism from any choice of where the "free generator" goes.
|
||||
Moreover,
|
||||
* this construction is respected by homomorphisms out of \\(G\\) (that's "naturality in \\(G\\)": if we have a homomorphism \\(h : G \to H\\), and we perform the construction on both \\(G\\) and \\(H\\), we find that every homomorphism \\(\mathrm{Rep}_A \xrightarrow{\text{Yoneda corresponding to $x \in GA$}} G \xrightarrow{h} H\\) is equal to \\(\mathrm{Rep}_A \xrightarrow{\text{Yoneda corresponding to $h(x) \in HA$}} H\\));
|
||||
* this construction is respected by morphisms \\(A \to B\\) (that's "naturality in \\(A\\)"): this is currently entirely an exercise because I've ground to a halt.
|
||||
|
||||
The precise construction is like so:
|
||||
|
||||
* Given an element \\(a \in G(A)\\), define a homomorphism \\(\mathrm{Rep}_A\\) to \\(G\\) by sending \\(f : A \to B\\) to \\((Gf)(a)\\).
|
||||
* Given a homomorphism \\(\alpha : \mathrm{Rep}_A\\) to \\(G\\), define an element of \\(G(A)\\) by \\(\alpha(\mathrm{id}_A : A \to A)\\).
|
||||
* Show that these are inverse to each other.
|
||||
* Prove the naturality conditions.
|
||||
|
||||
Exercise: do those formally, as follows!
|
||||
|
||||
1. Prove that the homomorphism \\((f : A \to B) \mapsto (Gf)(a)\\) is indeed a homomorphism (that is, a natural transformation), by writing out the necessary properties which define a natural transformation and showing that they each hold.
|
||||
1. Prove (by writing out the equations) that the two Yoneda maps are inverse to each other.
|
||||
1. Write out the equations for both naturality conditions (that is, in \\(A\\) and in \\(G\\)) in full, and prove them.
|
||||
|
||||
# The Yoneda embedding
|
||||
|
||||
Given any (locally-small) category \\(\mathcal{C}\\) and an object \\(A : |\mathcal{C}|\\), we've seen a way to construct an instantiation of \\(\mathcal{C}\\) in \\(\mathrm{\mathbf{Set}}\\) in a canonical way, representing exactly the structure of \\(\mathcal{C}\\) that was available at \\(A\\).
|
||||
|
||||
## Definition of "fully faithful"
|
||||
|
||||
Recall the definitions of "full" and "faithful" for a functor \\(F : \mathcal{C} \to \mathcal{D}\\):
|
||||
* The functor is *faithful* if it is injective on parallel arrows: for all \\(A, B : |\mathcal{C}|\\), no two morphisms \\(A \to B\\) get mapped to the same target arrow.
|
||||
* The functor is *full* if it is surjective on objects which came from the functor: for all \\(A, B : |\mathcal{C}|\\), and for all \\(g : FA \to FB\\), there is \\(f : A \to B\\) with \\(Ff = g\\).
|
||||
|
||||
A functor \\(\mathcal{C} \to \mathcal{D}\\) is *fully faithful* if it is both full and faithful.
|
||||
While it may collapse away some of \\(\mathcal{C}\\)'s structure, and while it might not hit all of \\(\mathcal{D}\\), it does "locally" preserve all of \\(\mathcal{C}\\)'s structure and introduces no new structure locally: if we fix two objects \\(A, B\\) in \\(\mathcal{C}\\) and restrict ourselves to only looking at \\(A, B\\) and \\(FA, FB\\), the restricted \\(F\\) is a bijection.
|
||||
|
||||
## The Yoneda embedding is fully faithful
|
||||
|
||||
Notice that the collection of representable functors (that is, the "sweet spot" instantiations of \\(\mathcal{C}\\)) has got the right number of objects to try and be a copy of \\(\mathcal{C}\\): there's a representable functor for each object of \\(\mathcal{C}\\) already (because that's how we defined the representable functors).
|
||||
So what happens if we try and complete this into a copy of \\(\mathcal{C}\\) with all its categorical structure, inside the much larger space of instantiations of \\(\mathcal{C}\\)?
|
||||
This would be a nice thing to have, because the space of instantiations of \\(\mathcal{C}\\) is very well-behaved (they're all just sets!).
|
||||
|
||||
So to view it as a copy of \\(\mathcal{C}\\), we need for any \\(A, B : |\mathcal{C}|\\) and any morphism \\(f : A \to B\\) to find a homomorphism from \\(\mathrm{Rep}_A\\) to \\(\mathrm{Rep}_B\\).
|
||||
Take the Yoneda lemma and specialise it by setting \\(G := \mathrm{Rep}_B\\); then we have that the homomorphisms \\(\mathrm{Rep}_A \to \mathrm{Rep}_B\\) are naturally in bijection with the elements of \\(\mathrm{Rep}\_B(A)\\), which is by definition \\(\mathrm{Hom}\_{\mathcal{C}}(B, A)\\).
|
||||
|
||||
That's… not actually what we wanted!
|
||||
It's very close, but the arrows are going the wrong way: we have homomorphisms \\(\mathrm{Rep}\_A \to \mathrm{Rep}\_B\\) corresponding naturally to \\(\mathrm{Hom}\_{\mathcal{C}}(B, A)\\).
|
||||
|
||||
What we've actually done is built a canonical copy of the *opposite* category of \\(\mathcal{C}\\) inside the space of instantiations of \\(\mathcal{C}\\).
|
||||
This canonical copy is called the *Yoneda embedding*, and you can prove that it is full and faithful.
|
||||
|
||||
That was the *contravariant Yoneda embedding*, which takes \\(\mathcal{C}^{\mathrm{op}}\\) and embeds it fully faithfully in \\(\mathrm{Nat}(\mathcal{C} \to \mathrm{\mathbf{Set}})\\), the space of all \\(\mathrm{\mathbf{Set}}\\)-instantiations of \\(\mathcal{C}\\).
|
||||
(That space is more precisely a category, with morphisms being precisely the instantiation homomorphisms, or natural transformations.)
|
||||
By flipping all the arrows around, we also get the *covariant Yoneda embedding*, which takes \\(\mathcal{C}\\) and embeds it fully faithfully in \\(\mathrm{Nat}(\mathcal{C}^{\mathrm{op}} \to \mathrm{\mathbf{Set}})\\), the space of all \\(\mathrm{\mathbf{Set}}\\)-instantiations of \\(\mathcal{C}^{\mathrm{op}}\\) (also known as the space of all *presheaves* over \\(\mathcal{C}\\)).
|
25
hugo/content/posts/2024-05-01-i-notice-that-i-am-confused.md
Normal file
25
hugo/content/posts/2024-05-01-i-notice-that-i-am-confused.md
Normal file
@@ -0,0 +1,25 @@
|
||||
---
|
||||
lastmod: "2024-05-01T11:38:00.0000000+01:00"
|
||||
author: patrick
|
||||
date: "2024-05-01T11:38:00.0000000+01:00"
|
||||
title: The phrase "I Notice That I Am Confused"
|
||||
summary: "To me it's got a specific meaning, but I've seen it used much more generally, and I think its meaning should not be polluted."
|
||||
---
|
||||
|
||||
In LessWrong-speak, "[I notice that I am confused](https://www.lesswrong.com/s/zpCiuR4T343j9WkcK)" is a pretty standard phrase.
|
||||
I've seen [Zvi](https://thezvi.wordpress.com/) in particular use that phrase in a great many contexts where my understanding of it doesn't match.
|
||||
So here's my understanding of it.
|
||||
|
||||
> "I Notice That I Am Confused" is the phrase that accompanies/triggers the motion "pause, attach the debugger to your world modelling system, and replay the last explanation you accepted".
|
||||
|
||||
Here are a couple of possible reasons I might want to attach the debugger:
|
||||
|
||||
* I can feel my world model contorting itself around something. (Maybe I can only feel the fact of a contortion and not the nature of it.)
|
||||
* Some part of my world modelling system appears to be actively working on something, but I can see no reason for it to do so. An idle process should not be consuming 20% CPU. Maybe it's still shaking out the implications of something (which means by definition I'm still confused).
|
||||
* Or maybe what it's working on is "make sure I never see this flaw in the model"! That actually happens! The modelling system wants to feel that its model is correct, and it's only of secondary importance that the model actually *be* correct. I am very familiar with the mental motion "there's no need to look over here, and there's *definitely* nothing over here that would invalidate the explanation".
|
||||
|
||||
Those all feel quite adversarial; here's one which is more of a cooperative process:
|
||||
|
||||
* Maybe there's just a nagging sense that not all the data is explained. An explanation feels complete, but the model is highlighting that some piece of data doesn't quite fit (possibly I haven't yet consciously identified what doesn't fit or why).
|
||||
|
||||
The point of the phrase "I notice that I am confused" is to indicate "this explanation feels compelling, but *something is off* and I need to promote this to consciousness to discover why".
|
21
hugo/content/posts/2024-07-24-philosophy-of-code.md
Normal file
21
hugo/content/posts/2024-07-24-philosophy-of-code.md
Normal file
@@ -0,0 +1,21 @@
|
||||
---
|
||||
lastmod: "2024-07-24T17:38:00.0000000+01:00"
|
||||
author: patrick
|
||||
date: "2024-07-24T17:38:00.0000000+01:00"
|
||||
title: Code having "the right philosophy"
|
||||
summary: "Those who would give up essential safety, to purchase a little temporary simplicity, deserve (and will get) neither safety nor simplicity."
|
||||
---
|
||||
|
||||
[Hacker News](https://news.ycombinator.com/item?id=41032806):
|
||||
|
||||
> Time library can be simple, it's just rust libraries tend to be philosophic for some reason, but it's only one of many design approaches.
|
||||
|
||||
Certainly a true statement.
|
||||
|
||||
I claim that the "some reason" here is that this "philosophic" design approach is correct, and the other ones aren't.
|
||||
Most standard libraries (.NET, Golang, Python) don't care about correctness enough that they'd actually *model the domains* in question, so they don't, so they make it trivial to write completely unnecessary bugs.
|
||||
|
||||
As evidence, I submit the fact that I have twice had to rewrite someone else's .NET code to fix time-related bugs in it, whose ultimate direct cause was "the .NET `DateTime` type is an extremely bad model for dates-and-times in the world".
|
||||
In one of those two cases, I ended up completely rewriting the entire application, using NodaTime to drive the date-time computations, because that's a library that actually attempts to model dates-and-times.
|
||||
|
||||
Those who would give up essential safety, to purchase a little temporary simplicity, deserve (and will get) neither safety nor simplicity.
|
28
hugo/content/posts/2024-07-25-lob-theorem.md
Normal file
28
hugo/content/posts/2024-07-25-lob-theorem.md
Normal file
@@ -0,0 +1,28 @@
|
||||
---
|
||||
lastmod: "2024-07-25T19:48:00.0000000+01:00"
|
||||
author: patrick
|
||||
date: "2024-07-25T19:48:00.0000000+01:00"
|
||||
title: Learning plan for "Program Equilibrium in the Prisoner's Dilemma"
|
||||
summary: The questions whose answers I don't know, and the things I intend to learn, on the way to understanding a paper.
|
||||
---
|
||||
|
||||
## Resources
|
||||
The paper is [Program Equilibrium in the Prisoner's Dilemma](https://intelligence.org/files/ProgramEquilibrium.pdf).
|
||||
|
||||
A prerequisite is [Löb's theorem](https://en.wikipedia.org/wiki/L%C3%B6b's_theorem), for which Yudkowsky has a [cartoon guide](https://www.lesswrong.com/posts/ALCnqX6Xx8bpFMZq3/the-cartoon-guide-to-loeb-s-theorem).
|
||||
|
||||
One of the foundational papers is by Smullyan: [Logicians who reason about themselves](https://dl.acm.org/doi/pdf/10.5555/1029786.1029818).
|
||||
|
||||
## Questions I intend to answer
|
||||
|
||||
* Why can't we similarly prove that we're going to defect? "If it's provable that we defect, then we defect"?
|
||||
* I had [an objection about "parametric bounded Löb"](/misc/ParametricBoundedLoeb2016/ParametricBoundedLoeb2016.pdf) ages and ages ago; do I still believe it?
|
||||
* The paper currently works relative to a provability oracle; is it possible in principle for us to precompute and hand over proofs along with our source code? What would such a proof look like?
|
||||
* Understand the [diagonal lemma](https://en.wikipedia.org/wiki/Diagonal_lemma).
|
||||
* Understand Löb's theorem.
|
||||
* Why does Wikipedia say that Smullyan refers to an agent as "modest" when 'such a reasoner can never believe "my belief in P would imply that P is true", without also believing that P is true'? Why that word?
|
||||
* Can I get an intuition for what it would actually feel like to be an ideal agent going through the thought process of FairBot? (Or what it would mean to *be* PA computing the truth of a statement via Löb?)
|
||||
* Fully and precisely write out the definition of FairBot and of PrudentBot in some pseudocode (permitting calls to `is_provable(expr)`).
|
||||
* Comprehend the sentence: It is important that we look for proofs of `X(DB) = D` in a stronger formal system than we use for proving `X(PB) = C`; if we do otherwise, the resulting variant of PrudentBot would lose the ability to cooperate with itself.
|
||||
* Are there any stronger conditions that will let us do somehow "better", by successfully defecting against more opponents, than PrudentBot? (Does this come down to tradeoffs about how easy it is for opponents to prove that you're going to cooperate with them?)
|
||||
* Understand the scope of [MIRI's implementation](https://github.com/machine-intelligence/provability).
|
18
hugo/content/posts/2024-08-15-github-workflows.md
Normal file
18
hugo/content/posts/2024-08-15-github-workflows.md
Normal file
@@ -0,0 +1,18 @@
|
||||
---
|
||||
lastmod: "2024-08-15T17:51:00.0000000+01:00"
|
||||
author: patrick
|
||||
date: "2024-08-15T17:51:00.0000000+01:00"
|
||||
title: New GitHub workflows
|
||||
summary: I've made a GitHub workflow to assert that all required GitHub checks are complete, and one to publish and attest NuGet packages.
|
||||
---
|
||||
|
||||
# [all-required-checks-complete](https://github.com/Smaug123/all-required-checks-complete-action)
|
||||
|
||||
The problem this solves is that GitHub's "required" checks are not actually required.
|
||||
A "required" step which is skipped because it depends on a failed step will count as successful.
|
||||
(This is folklore, but [here's one writeup](https://emmer.dev/blog/skippable-github-status-checks-aren-t-really-required/).)
|
||||
|
||||
# [publish-nuget](https://github.com/Smaug123/publish-nuget-action)
|
||||
|
||||
NuGet [does not lend itself](https://github.com/NuGet/NuGetGallery/issues/10026) to [GitHub artefact attestation](https://docs.github.com/en/actions/security-for-github-actions/using-artifact-attestations/using-artifact-attestations-to-establish-provenance-for-builds), so there's some annoying work to do to make that happen.
|
||||
This action performs that work!
|
56
hugo/content/posts/2024-08-26-woofware-arg-parser.md
Normal file
56
hugo/content/posts/2024-08-26-woofware-arg-parser.md
Normal file
@@ -0,0 +1,56 @@
|
||||
---
|
||||
lastmod: "2024-08-26T12:26:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- uncategorized
|
||||
date: "2024-08-26T12:26:00.0000000+01:00"
|
||||
title: WoofWare.Myriad.Plugins learns to parse args
|
||||
summary: "My F# source generators have some new features, including an argument parser."
|
||||
---
|
||||
|
||||
This post is about [WoofWare.Myriad.Plugins](https://github.com/Smaug123/WoofWare.Myriad), a set of [F#](https://en.wikipedia.org/wiki/F_Sharp_(programming_language)) source generators ([see it on NuGet](https://www.nuget.org/packages/WoofWare.Myriad.Plugins)).
|
||||
Go and read [the README on GitHub](https://github.com/Smaug123/WoofWare.Myriad/blob/main/README.md) if you're interested.
|
||||
|
||||
They are particularly intended for `PublishAot` [ahead-of-time compilation](https://learn.microsoft.com/en-us/dotnet/core/deploying/native-aot/) contexts, in which reflection is heavily restricted, but also for anyone who doesn't want reflection for whatever reason (e.g. "to obtain the ability to step through code in a debugger", or "for more predictable speed").
|
||||
|
||||
Since [my last post], I've implemented the following:
|
||||
|
||||
* `[<JsonSerialize>]` (to stamp out `toJsonNode : 'T -> JsonNode` methods)
|
||||
* `[<CreateCatamorphism>]` (to build a non-stack-overflowing [catamorphism](https://fsharpforfunandprofit.com/posts/recursive-types-and-folds/) for an algebraic data type)
|
||||
* `[<ArgParser>]` (to stamp out an argument parser).
|
||||
|
||||
## `ArgParser`
|
||||
|
||||
Example:
|
||||
|
||||
```fsharp
|
||||
[<ArgParser>]
|
||||
type Foo =
|
||||
{
|
||||
[<ArgumentHelpText "Enable the frobnicator">]
|
||||
SomeFlag : bool
|
||||
A : int option
|
||||
[<ArgumentDefaultFunction>]
|
||||
B : Choice<int, int>
|
||||
[<ArgumentDefaultEnvironmentVariable "MY_ENV_VAR">]
|
||||
BWithEnv : Choice<int, int>
|
||||
C : float list
|
||||
// optionally:
|
||||
[<PositionalArgs>]
|
||||
Rest : string list // or e.g. `int list` if you want them parsed into a type too
|
||||
}
|
||||
static member DefaultB () = 4
|
||||
```
|
||||
|
||||
Features:
|
||||
|
||||
* Optional arguments of type `'a option`.
|
||||
* `[<ArgumentDefaultFunction>]` and `[<ArgumentDefaultEnvironmentVariable>]` to auto-populate default arguments which are not supplied. Default values are modelled as `Choice<'a, 'a>`, with `Choice1Of2` meaning "the user gave me this", and `Choice2Of2` meaning "this was populated from a default source".
|
||||
* Help text with `[<ArgumentHelpText>]`, summoned with `--help` in any position where the parser is expecting an argument, and also summoned during certain failures to parse.
|
||||
* Detailed control over `TimeSpan` parsing with `[<ParseExact>]` (and `[<InvariantCulture>]` if desired).
|
||||
* Accumulation of arguments supplied repeatedly: for example, `Path : FileInfo list` is populated with `--path /foo/bar --path /baz/quux`.
|
||||
* Handling of `--foo=bar` and `--foo bar` equivalently.
|
||||
* Booleans have arity 1 or 0, whichever leads to a successful parse: `--flag` and `--flag=true` and `--flag true` are equivalent.
|
||||
* Positional arguments can appear anywhere, although if they start with a `--` then it is best to put them after a trailing `--` separator: `--named-arg=blah -- --pos-arg1 pos-arg2 --pos-arg3`.
|
||||
|
||||
[my last post]: {{< ref "2023-12-31-woofware-myriad-plugins" >}}
|
28
hugo/content/posts/2024-09-07-massage.md
Normal file
28
hugo/content/posts/2024-09-07-massage.md
Normal file
@@ -0,0 +1,28 @@
|
||||
---
|
||||
lastmod: "2024-09-07T11:47:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- uncategorized
|
||||
date: "2024-09-07T11:47:00.0000000+01:00"
|
||||
title: Lessons from a massage course
|
||||
summary: "I went to a one-day intro to massage taster course, and it was fun and interesting!"
|
||||
---
|
||||
|
||||
I recently attended the [Bodyology one-day intro to massage taster](https://bodyologymassagecourses.co.uk/courses/intro-course).
|
||||
Recommended!
|
||||
The instructor Dror was excellent, with a teaching style correctly based around telling you why things are done the way they are, and then giving you lots of time to practise finding how your own hands best achieve the goals.
|
||||
|
||||
# Why did I try massage?
|
||||
|
||||
1. The ever-present desire to find meaning (c.f. applying for the RAF reserves); it's common knowledge that [directly and viscerally helping others](https://80000hours.org/career-guide/job-satisfaction/) is something that people find very meaningful.
|
||||
2. Wouldn't that be a neat skill to be able to reveal that you're bizarrely, impossibly good at! ("Sometimes, magic is just someone spending more time on something than anyone else might reasonably expect.")
|
||||
|
||||
# Interesting lessons
|
||||
|
||||
* Move from the legs. This is obvious every single time it comes up anywhere, and it comes up all the time, but somehow I *still* never generalise the lesson so that I know it without being told in any given situation.
|
||||
* I find it *really* stressful to have no real-time feedback on what I'm doing. Pretty much everything I do has a tight feedback loop, but the feedback loop for massage is really long: it breaks the immersion to be constantly using your mouth-words during the session, and if the victim is face-down then your only source of feedback is literally just "try to read someone's back", unless you are much more empathic than I am (which would certainly come with practice). I was a lot more relaxed when my subject and I were maintaining pretty constant verbal communication.
|
||||
* This is even worse for head massage, where the subject tends to fall asleep (or get very close to it), which I guess is a positive signal but also is a complete lack of any other signals.
|
||||
* Massage also gives you no *objective* feedback, so it falls into the category of "things where my System 1 expects explicit positive feedback to be false". I can't stop myself interpreting other people's feedback as "this person is trying to make you feel better about your terrible performance", at least without a bit of time getting to know that person first.
|
||||
* Cultural hangups around grabbing a woman by the neck. There was an entire session on neck/traps/shoulder massage, and it took me a solid two minutes before I was able to grip the spinal erectors hard enough to have any massage-related effect at all. This is hard to do when for thirty years you've had a culture screaming "YOU WILL NOT GRAB A WOMAN BY THE NECK" at you! (I also suspect I was under more internal strain than I explicitly felt, because I choked up slightly when I gave a sentence explaining my light touch afterwards.)
|
||||
* A massage often begins with a simple laying-on-of-hands. This is actually a communication channel! Take it slow: you're greeting the subject, setting the terms of engagement, letting yourself adjust to their rhythm (mainly their breathing). A therapist who moves their hands around feels indecisive, and remember that time will feel like it's passing faster for the therapist because they're upright and in control. Different people communicate different things this way, which surprised me a lot. I really felt (for example) that, while I was the demo subject, the instructor was saying hello to me through his hands, and that I was replying through my body's breathing movement. (One breathing cycle lasts many seconds; if the therapist moves their hands before you've even finished one breath, that actually feels pretty rude: they're literally interrupting your communication.) By contrast, during a practice session, someone gave me a very firm and assertive greeting, "I am in control" (which is certainly not bad, it's just very different!).
|
||||
* Sunflower oil works just fine as a massage oil; in fact the instructor recommended it over other oils.
|
99
hugo/content/posts/2024-09-27-software-rant.md
Normal file
99
hugo/content/posts/2024-09-27-software-rant.md
Normal file
@@ -0,0 +1,99 @@
|
||||
---
|
||||
lastmod: "2024-09-27T18:14:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2024-09-27T18:14:00.0000000+01:00"
|
||||
title: Unhinged rant about software
|
||||
summary: "Modern software practices and their sadness."
|
||||
---
|
||||
|
||||
*Epistemic status: wail of anguish*
|
||||
|
||||
I've given this rant twice recently, so will unwisely write it down to avoid giving it a third time.
|
||||
|
||||
# Most people seem bent on self-flagellation
|
||||
|
||||
Subtitle: And I Don't Think They Realise How Hard They're Making It For Themselves.
|
||||
|
||||
## Python
|
||||
|
||||
Python is touted as "simple".
|
||||
What people really mean by is that library developers put in large amounts of effort to make it *concise*.
|
||||
Python itself is pretty complicated (indeed, by design you essentially can't make *any* static guarantees), and its extreme dynamism makes it in practice *extremely complex*.
|
||||
|
||||
I recently mentioned [Pyright](https://github.com/microsoft/pyright) to a professional Python developer.
|
||||
This person tried it out, and stopped, saying "it's giving me all these false positives".
|
||||
He then described two such broad classes of false positives, *both* of which were in fact true positives: they were indicating bugs in the code.
|
||||
(I'm not talking about logic bugs here, revealed by an attempt to impose types; these are bugs arising due to the use of actual language features.)
|
||||
|
||||
A professional full-time Python developer did not know that they were bugs!
|
||||
This is not his fault: Python is really hard to understand!
|
||||
|
||||
(I shall not mention the dependency management situation because everyone already knows what a flaming mess that is.)
|
||||
|
||||
## .NET and ASP.NET
|
||||
|
||||
Idiomatic .NET code seems to be bizarrely full of galaxy-brained object orientation.
|
||||
ASP.NET is a *particularly* bad example.
|
||||
Throughout the standard libraries and chunks of the ecosystem, Microsoft and everyone else goes to extreme lengths to construct a system that's as hard as possible to understand.
|
||||
For example:
|
||||
|
||||
* ASP.NET specialises in converting all compile-time errors to runtime errors. Its total reliance on runtime dependency injection means you simply have to run your server to discover whether it even has a chance of working. This is the part of working on ASP.NET stuff that causes me literal physical pain (in my legs, if you're interested). As far as I know, you can't opt out of this insanity. [There's a correct way to do dependency injection](https://www.bartoszsypytkowski.com/dealing-with-complex-dependency-injection-in-f/), which is type-checked and compile-time-safe, but of course C# can't express it.
|
||||
* Convention Over Configuration, i.e. "don't worry your pretty little head about how this thing works; just trust Daddy Microsoft". All abstractions leak, and Microsoft's approach makes it almost completely impossible to work out what's happening when ASP.NET leaks.
|
||||
* No human is capable of writing an ASP.NET service correctly. The middleware system is notoriously brittle: it depends fundamentally and silently on chaining together all the middleware in the correct order. GitHub [is](https://github.com/aspnet/SignalR/issues/2316) [full](https://github.com/dotnet/AspNetCore.Docs/issues/28467) [of](https://github.com/IzyPro/WatchDog/issues/46) [issues](https://github.com/dotnet/aspnetcore/issues/15313) [describing](https://github.com/dotnet/aspnetcore/issues/34146) [people](https://github.com/dotnet/aspnetcore/issues/14049#issuecomment-533190098) [getting](https://github.com/dotnet/aspnetcore/issues/52295) [this](https://github.com/dotnet/AspNetCore.Docs/issues/14221#issuecomment-530436736) [wrong](https://github.com/dotnet/AspNetCore.Docs/issues/19957#issuecomment-698572231), [or](https://github.com/domaindrivendev/Swashbuckle.AspNetCore/issues/2641) [asking](https://github.com/dotnet/AspNetCore.Docs/issues/17031) [for](https://github.com/dotnet/aspnet-api-versioning/issues/704) [better](https://github.com/dotnet/aspnetcore/issues/45302) [documentation](https://github.com/dotnet/aspnet-api-versioning/issues/979). All common .NET languages are strongly- statically- typed; why did they decide to throw away all the help the computer can give you?
|
||||
* ASP.NET has painted itself into such a corner that they have had to start putting some *seriously* ridiculous features into C#. I'm thinking here of [interceptors](https://devblogs.microsoft.com/dotnet/new-csharp-12-preview-features/#interceptors), which they describe as "enabl[ing] exciting code patterns". I do not want exciting code patterns! If you want to generate source code, then do so; give us a macro system if you like; but don't bake this stuff into your language - you're not a Lisp!
|
||||
* Also by the way you're not Ruby. When I discovered [Harmony](https://github.com/pardeike/Harmony) in a dependency recently, I had several minutes of existential terror.
|
||||
* Why do the Microsoft identity libraries perform 80ms of reflection before decoding a JWT (we timed this)?
|
||||
|
||||
For a system which eschews compile-time safety in favour of "convention", .NET sure has terrible docs.
|
||||
Why are they so big, and why do they contain so little useful information?
|
||||
Why are all these systems built to be impossible to use without a comprehensive knowledge of their implementation, such that I have to open the .NET source code *every time* I have a question about what something does?
|
||||
Why do I [keep](https://github.com/dotnet/dotnet-api-docs/pull/9753) [on](https://github.com/dotnet/dotnet-api-docs/pull/9595) [having](https://github.com/dotnet/runtime/pull/95956) [to](https://github.com/dotnet/docs/pull/35171) [raise](https://github.com/dotnet/dotnet-api-docs/pull/9394) [docs](https://github.com/dotnet/docs/pull/42637) [PRs](https://github.com/dotnet/aspnetcore/pull/57779) adding the most basic information?
|
||||
So many questions!
|
||||
|
||||
## Golang
|
||||
|
||||
Another example of a "simple" language which it's impossible to write correctly without [a large suite of linters](https://github.com/golangci/golangci-lint) you don't get out-of-the-box.
|
||||
[Fasterthanli.me](https://fasterthanli.me/articles/i-want-off-mr-golangs-wild-ride) has written most of what I want to say about this.
|
||||
|
||||
I particularly enjoy the fact that you can identify a Golang program by the extremely cursed error messages you get out of it, where it dumps uninitialised memory to the console and that sort of thing.
|
||||
This happens with considerable regularity.
|
||||
I've even observed it in `kubectl`, so it's not just that I've been unlucky; even the best Golang programmers can't reliably avoid thinking garbage memory is a string.
|
||||
But apparently Golang error handling is great, so what would I know.
|
||||
|
||||
[Boats](https://without.boats/blog/let-futures-be-futures/):
|
||||
|
||||
> You might also decide your language should have other classic features like null pointers, default constructors, data races and GOTO, for reasons known only to you.
|
||||
|
||||
Why did the creators not put algebraic data types into the language, if they were going to do errors-as-return-values?
|
||||
|
||||
## Summary
|
||||
|
||||
The Jonathan Blow/Casey Muratori sphere of the Internet probably expresses this the most strongly, but they are completely correct.
|
||||
I think people are using *really hard environments* to solve *usually very simple problems*, and it makes their lives so much harder than necessary.
|
||||
Their work environment is absolutely full of [incidental complexity](https://en.wikipedia.org/wiki/No_Silver_Bullet).
|
||||
|
||||
To lightly paraphrase someone very senior at $WORK describing me, my schtick (a decent chunk of why I'm a "Principal Engineer" there) is to get annoyed at the existing solution to a problem and write a dumb replacement in F#, in four hours, that's orders of magnitude faster, orders of magnitude less code, and requires less than one PR of maintenance a month (and usually *much* less).
|
||||
I've done this several times now.
|
||||
(There's one thing, "Kraken" for those who know the G-Research internal environment, which doesn't fit that pattern in that I did actual extended work to create it; I think that last received a PR nearly a year ago, which is to say that it is also essentially complete. That used engineering techniques no more complicated than [diligently making sure all state was reified][dryrun] and using discriminated unions to tightly constrain the domains. None of this is rocket science!)
|
||||
|
||||
I believe I'm not a "10x engineer" but that most people are following practices which hold them back *so much*.
|
||||
|
||||
# Copilot etc
|
||||
|
||||
As an encore, I believe I have an answer to "why does everyone seem to like the Copilot-like things so much, whereas I have to turn them off within five minutes of trying them?".
|
||||
|
||||
Most people work in environments which have been built to resist understanding, for which you can't get a good mental model without vast expenditure of effort, so most people are forced to write software by vibes.
|
||||
ChatGPT and friends are astonishingly powerful vibes engines, and in any given case they can often (usually?) vibe better than a human can; in particular, they can vibe code at a human level.
|
||||
Since humans are only vibing when they write code, ChatGPT is good enough to do a large chunk of the human job.
|
||||
A human can't reliably do better because almost no human (very much including me!) can develop the superhuman systems understanding that is required to *engineer* a solution.
|
||||
|
||||
I am thinking of things like:
|
||||
|
||||
* ASP.NET (see earlier).
|
||||
* Python (see earlier).
|
||||
* CSS, which is essentially a vast collection of special cases.
|
||||
* Javascript, about whose shortcomings enough words have certainly been written elsewhere.
|
||||
|
||||
[dryrun]: {{< ref "2021-02-20-in-praise-of-dry-run" >}}
|
65
hugo/content/posts/2024-10-25-marys-room.md
Normal file
65
hugo/content/posts/2024-10-25-marys-room.md
Normal file
@@ -0,0 +1,65 @@
|
||||
---
|
||||
lastmod: "2024-10-25T23:30:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- philosophy
|
||||
date: "2024-10-25T23:30:00.0000000+01:00"
|
||||
title: Mary's Room
|
||||
summary: "Last weekend I was in a discussion ostensibly about whether the mind is nonphysical. Much of the discussion ended up as a rather polite shouting match about Mary's Room. Here is what I actually believe about it."
|
||||
---
|
||||
|
||||
[Mary's Room](https://en.wikipedia.org/wiki/Knowledge_argument) is a standard thought experiment purporting to argue that "however much you know about the physical world in a forever-black-and-white environment, you still learn something new when you step out of the room and experience redness for the first time".
|
||||
I, as a physicalist, do not accept this conclusion.
|
||||
Here is why.
|
||||
|
||||
# A new quale is a new low-level label
|
||||
|
||||
If you've been thinking about something for a while, and have come to a detailed understanding of it, you might decide to write down a name for it in your notebook (or even in the dictionary).
|
||||
That doesn't mean you've *discovered* a new word, or discovered anything new about the world in the act of writing it down.
|
||||
|
||||
Similarly, putting your world-simulating engine into a position where it simulates a phenomenon it's never simulated before (in Mary's Room the chosen phenomenon is traditionally "redness") doesn't necessarily cause you to "learn" anything new.
|
||||
It's just that the same existing knowledge has been assigned a first-class primitive in the underlying mental architecture, so it's much easier to access.
|
||||
|
||||
Analogously to how you made it easier to think about a concept by giving it a word, your brain made it easier to think about redness by assigning it a primitive as soon as it first encountered something red.
|
||||
(Or possibly that primitive was already assigned and waiting to be activated; perhaps redness is hard-coded specifically in the brain. If you think redness might be hard-coded, consider the smell of ammonia or similar instead, and substitute that learned quale wherever I talk about redness.)
|
||||
|
||||
That primitive "always existed" in some mathematical sense, latent in the structure of the simulator, but it was previously not accessed or used.
|
||||
|
||||
Note that the set of available primitives is not fixed up front; the brain turns out to have a *very large* supply of available slots!
|
||||
|
||||
# Why we have qualia
|
||||
|
||||
The new first-class primitive actually makes your reasoning *more sloppy* - things can "seem red" even though they "are not red" (in the sense that they don't emit the right wavelengths of light when irradiated with standard-test-compliant light) - but reasoning with this primitive is *much* faster than reasoning without it.
|
||||
Natural selection rewards speed, so your brain's architecture is predisposed to assign a primitive to "redness" instantly on first encounter.
|
||||
|
||||
The term "[System 1](https://en.wikipedia.org/wiki/Thinking,_Fast_and_Slow)" is a shorthand for some of the reasoning that takes place using *only* these efficient primitives of the architecture.
|
||||
It's much faster, because it's much less general: it can only use primitives that are already represented efficiently, and it can't take account of the ways in which those primitives are only approximations of the world (which is why optical illusions still "feel wrong" even when you know how they work).
|
||||
|
||||
# Qualia can also be learned the hard way
|
||||
|
||||
There are some concepts which *don't* get insta-compiled like redness does: for example, the notion of a group in mathematics is one we're simply not hard-wired to convert instantly into a compatible form for System 1.
|
||||
For a fuzzier example, the notion of "prime number" is one that people can *learn* a primitive for: a number can feel "more prime" or "less prime", and the more you work with prime numbers, the more you start "actually feeling" whether a number is prime and the more System 1 starts having opinions about primality.
|
||||
|
||||
On the other hand, it was very necessary in the ancestral environment to be able to reason quickly and intuitively about things like "*precisely* what chemicals are around me" or "what vibrations are happening near me"; so natural selection eventually predisposed the brain to go absolutely wild in immediately assigning unique highly-accessible identifiers to smells and sounds.
|
||||
|
||||
# Mary is a just-in-time compiler
|
||||
|
||||
Mary carefully built up a knowledge of redness through years of study.
|
||||
She defined a symbolic representation - an [opcode](https://en.wikipedia.org/wiki/Opcode) - and became able to manipulate it through her System 2 reasoning process.
|
||||
|
||||
When Mary stepped out of the room, it's a bit like that opcode was suddenly [JIT](https://en.wikipedia.org/wiki/Just-in-time_compilation)-compiled into machine code, making it *extremely* accessible to System 1.
|
||||
Her brain already came preloaded with a large unused space of appropriate instructions in the machine code architecture; the brain selected one of those instructions (perhaps at random, or perhaps in a way that's common to most humans thanks to shared genetic coding for that specific structure).
|
||||
That JIT didn't happen until her brain actually received the input for the first time.
|
||||
|
||||
I have a more speculative claim, which isn't central to the argument but if true would simply refute Mary's Room entirely, by showing that the experiment could never be performed: if she were well practised with meditation and dream-control and so forth, I claim she could make this JIT process happen without ever seeing red in the world.
|
||||
As evidence, I gesture towards the fact that humans can enter the [jhanas](https://en.wikipedia.org/wiki/Dhyana_in_Buddhism) and find genuinely new experiences this way.
|
||||
(However, it's possible that the human brain architecturally just can't induce these low-level hacks on itself. My bet is that it can, but if she can't, I think that's a reflection of her bounded access to her own hardware, rather than showing that there were some fundamentally non-physical truths.)
|
||||
|
||||
# Inverted spectrum
|
||||
|
||||
["Is my red the same as your red?"](https://en.wikipedia.org/wiki/Inverted_spectrum)
|
||||
|
||||
I don't know the answer to this empirical question.
|
||||
|
||||
* Is the "redness" representation hardcoded, or is it randomly assigned from the wide available pool of qualia devoted to colour? (We could perhaps prove it to be hardcoded if we were able to identify specific parts of the genome which result in specific nerves carrying the red cones' signals to specific structures within the visual processing centre, and using techniques from mechanistic interpretability to identify that the presence of this structure causes redness to manifest the same way in all humans. Or perhaps it's not hardcoded after all!)
|
||||
* Are the reasoning algorithms so perfectly isomorphic between humans that it's even meaningful to say that my quale *is the same as* your quale? (Again partly an empirical question; we could probably show this to be *false* if we could trace the execution of two brains finely enough and their representations turned out to be wildly divergent!) Perhaps two humans can simply have architectures which so fundamentally differ in how they model the world that it's *not* meaningful to say that they are "experiencing the same thing" when exposed to the same input. (Is the `ADD` instruction "the same" in [aarch64](https://en.wikipedia.org/wiki/AArch64) vs [x86-64](https://en.wikipedia.org/wiki/X86-64): do the two processors experience "the same operation", or does the wildly differing architecture make that a meaningless or false statement? Does it even matter, since they have the same result?)
|
28
hugo/content/posts/2024-11-28-net9-aot.md
Normal file
28
hugo/content/posts/2024-11-28-net9-aot.md
Normal file
@@ -0,0 +1,28 @@
|
||||
---
|
||||
lastmod: "2024-11-28T20:10:00.0000000+00:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- philosophy
|
||||
date: "2024-11-28T20:10:00.0000000+00:00"
|
||||
title: NativeAOT in .NET 9 and Nixpkgs/darwin
|
||||
summary: "How to get a NativeAOT build using nixpkgs"
|
||||
---
|
||||
|
||||
When nixpkgs was at commit [af51545ec9a44eadf3fe3547610a5cdd882bc34e](https://github.com/NixOS/nixpkgs/tree/af51545ec9a44eadf3fe3547610a5cdd882bc34e), the following sufficed for `dotnet publish --self-contained --configuration Release` to Just Work to build a `<PublishAot>true</PublishAot>` executable on aarch64-darwin:
|
||||
|
||||
```nix
|
||||
{
|
||||
outputs = { nixpkgs, ... }:
|
||||
let pkgs = nixpkgs.legacyPackages.aarch64-darwin; in
|
||||
{
|
||||
devShell = pkgs.mkShell {
|
||||
buildInputs = [ pkgs.dotnetCorePackages.sdk_9_0 ];
|
||||
packages = [
|
||||
pkgs.clang
|
||||
pkgs.darwin.ICU
|
||||
pkgs.darwin.binutils
|
||||
];
|
||||
}
|
||||
}
|
||||
}
|
||||
```
|
77
hugo/content/posts/2025-03-08-consciousness.md
Normal file
77
hugo/content/posts/2025-03-08-consciousness.md
Normal file
@@ -0,0 +1,77 @@
|
||||
---
|
||||
lastmod: "2025-03-08T13:30:00.0000000+00:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- philosophy
|
||||
date: "2025-03-08T13:30:00.0000000+00:00"
|
||||
title: A Bitcoin analogy for Multiple Drafts theory of consciousness
|
||||
summary: "I've never actually seen anyone write down precisely the way I think about consciousness, so here it is: it's (gasp!) like a blockchain."
|
||||
---
|
||||
|
||||
*This is basically Dennett's Multiple Drafts theory of consciousness as laid out in _Consciousness Explained_, Metzinger's self-model theory that the perception of a "self" is just another part of a model of the world, and Andy Clark's predictive processing theory; but with an additional trendy analogy to indicate a real-world proof-of-concept that Multiple Drafts systems can exist. Epistemic status: it's all vibes, bro. I think Claude thinks this post is bad.*
|
||||
|
||||
## Headline
|
||||
|
||||
There's no inherent single thread of consciousness; instead, it emerges as a best-effort attempt to record after-the-fact a coherent thread out of a much more confused jumble.
|
||||
This coherent thread actually models *what you were* pretty well in a behavioural sense, though, so as a simplifying assumption it's reasonable to say that it's functionally "real", and we call it your "consciousness".
|
||||
It's not a precise record of what you were experiencing, because it's only formed after the fact by discarding a great deal of information, but it's close enough that it's intuitive and useful to keep around in your memory.
|
||||
|
||||
## Background: Multiple Drafts, self-models, and predictive processing
|
||||
|
||||
The brain is a powerful prediction engine, with a single job: to model the world faithfully, [subject to certain constraints](https://en.wikipedia.org/wiki/Free_energy_principle) hard-wired by natural selection.
|
||||
For example, the prediction that "my body is experiencing pain" is hard-wired to have low probability, with the result that the prediction engine's connections to the body cause the body's machinery to take actions that reduce the pain.
|
||||
([One person's modus ponens is another person's modus tollens](https://gwern.net/modus): a strong prediction that the body is in a different place can cause the body to move to a different place, as the machinery which provides the evidence of location adjusts itself to validate the strong prediction coming in from the brain.)
|
||||
This is Clark's predictive processing explanation for how a "mere" prediction engine can take actions in the world.
|
||||
|
||||
One of the things that's present in the world, and therefore within the purview of the brain, is the brain itself, and the body that it's located in.
|
||||
Very early in development, or possibly in a move hard-coded by natural selection, the brain discovers that a pretty parsimonious explanation for what this brain and body tend to do is "this is an agent with desires", so the brain adopts that explanation pending strong evidence to the contrary.
|
||||
(I have no opinion on where this "agents exist" modelling simplification/assumption comes from; I think it would be aesthetically neater for the new brain to learn from scratch that agents are a good model of the world, but when has natural selection ever chosen the neat answer? Claude says "Research by Meltzoff and others suggests that infants begin with innate mechanisms for social cognition rather than discovering agency". Anyway, this point is not central to my argument; I'm sure
|
||||
the field of developmental psychology has an actual answer.)
|
||||
This is basically part of Metzinger's self-model theory, that the existence of a "self" is something that is natural to derive simply by looking at the world and asking how to explain the data you receive from the world.
|
||||
|
||||
To model accurately an agent with desires, it's necessary to model that agent's desires and reasoning processes.
|
||||
Any attempt to model *anything*, though, is subject to revision in the light of new incoming evidence, and in particular the brain's model of its own reasoning is constantly in flux.
|
||||
Some of its models will prove to be pretty accurate, and some will not; the ones which are not are discarded, and the ones which are accurate get enshrined in memory.
|
||||
These models are the "multiple drafts" of Dennett.
|
||||
|
||||
Importantly, the brain has no single one-true-model of itself at the time that it's making predictions; there are *only* the many models arising within the prediction engine, competing as all explanations do to explain the data.
|
||||
Most of the models agree in many ways, because the brain is pretty good at its job, and those things get committed to memory quickly and confidently.
|
||||
There are also many fringe parts of the predictions where the models disagree (because there's less evidence that would prove or refute noncentral parts of the model), and the memory system doesn't commit those quickly or at all.
|
||||
|
||||
The intuition of a single thread of personhood arises from the "intersection of all the models" being committed to memory: those features of the competing models of oneself which agree and explain the behaviour of this fictional "agent that exists in the world" that is you.
|
||||
But this intersection is itself just the memory subsystem's best attempt to record a consistent model; there are many other possible such records, in principle, and it's down to simple accident which one gets "written down".
|
||||
Moreover, memory is mutable, so even the one which got written down is constantly changing (usually only in little ways, but sometimes in huge ways).
|
||||
|
||||
## Speculative aside
|
||||
|
||||
*Epistemic status: Claude believes this section is false, because "research by Tore Nielsen and others on micro-awakenings and thought formation during sleep transitions (Nielsen, 2000, in Behavioral and Brain Sciences) suggests that the phenomenology of losing thoughts during sleep onset relates to reduced activity in the prefrontal cortex and default mode network, not to failures of memory commitment per se". This appears to be referencing
|
||||
[A review of mentation in REM and NREM sleep: “Covert” REM sleep as a possible reconciliation of two opposing models](https://www.dreamscience.ca/en/documents/publications/_2000_Nielsen_BBS_23_851-866_c-rem.pdf). Perplexity believes that the paper does not support Claude's objection. The paper is pretty lengthy and dense, and I wanted to get this post out.*
|
||||
|
||||
You can sometimes actually feel the negative space when you *fail* to commit a memory, when you're falling asleep.
|
||||
If you've had a train of thought which you simply lose because you're falling asleep, that's what it feels like when the intersection of some of these drafts fails to be committed.
|
||||
You feel that there *was* a coherent self (the memory subsystem would normally have committed a record of what the simulated agent that is "you" was doing, and if it had done so, it would have become the basis of the coherent future "you"), but you have completely lost access to what that self was (the memory subsystem did not in fact commit anything).
|
||||
This is probably a good thing, because pre-sleep thoughts can get pretty weird (i.e. contradictory with the usual self), and you probably don't want them to become foundational parts of yourself!
|
||||
|
||||
## The blockchain analogy
|
||||
|
||||
There's actually a Multiple Drafts system operating in the wild!
|
||||
Perhaps by musing on what it might feel like to "be the blockchain", we can intuitively feel Multiple Drafts in action.
|
||||
|
||||
There is no single necessary consciousness, just as there is no single necessary history of the Bitcoin blockchain.
|
||||
In the case of Bitcoin, a coherent history arises from the many validator nodes collectively agreeing on that history; but on short time horizons as the transactions are taking place, there is *no* coherent history.
|
||||
Instead, there are leaves of the Merkle tree constantly springing into being (one for every validator node) and being discarded (when the node becomes convinced that it has lost the race to commit its suggestion of what the history should be).
|
||||
These ephemeral leaves are equally "valid" as branches of history, but most of them are simply forgotten due to quirks of the distributed consensus algorithm: the nodes decide that *other* nodes are likely going to reject their attempts at history, so they abandon them quickly.
|
||||
|
||||
Similarly, in the brain there are many models of the world being postulated to explain current sense data, and in particular there are many models of the engine that is doing the prediction; and some of these models successfully continue predicting the world over short horizons, while some do not (and are discarded).
|
||||
The memory subsystem tries to collect a coherent intersection of the successful models, and records this history, because it's valuable evidence about what the prediction engine (the self) is going to do in the future.
|
||||
But there's no specific time when this happens; some detail gets cemented into history when the memory subsystem is confident in it, but this can be mutated years later, when you replay the memory and reconstruct what you think you must have been feeling at the time.
|
||||
The models which the memory subsystem believes aren't correctly cohering with the "self" get discarded, just as the failed leaves of the Merkle tree are forgotten.
|
||||
|
||||
The brain is probably not as neat as the blockchain: we don't end up with (say) a thousand clearly-delineated models which are then filtered based on how well they predict incoming evidence, and explicitly chosen or rejected.
|
||||
But I find this a compelling proof-of-concept that a coherent "history of the self" can arise from a wild cacophony of competing attempts to commit predictions of the self.
|
||||
|
||||
Is that history "real" despite being "merely emergent"?
|
||||
Sure, if you like, although it's certainly very mutable.
|
||||
For a start, there's the possibility of a "hard fork" (e.g. if you suddenly achieve religion and reinterpret your entire self).
|
||||
But more commonly, because your memory isn't perfect, you have to rerun your model of yourself to some extent when replaying a memory, to fill in the gaps; and your model of yourself *now* is a bit different to your model of yourself *then*, so the replay isn't quite faithful (this is the "false memory" phenomenon studied by Elizabeth Loftus, where rehearsing history rewrites history).
|
||||
That's not something that happens with the blockchain, of course, so in some sense the blockchain is "more real" as a persistent object than my consciousness is: the blockchain won't mutate its history as I try and study it.
|
52
hugo/content/posts/2025-06-16-llms-redux.md
Normal file
52
hugo/content/posts/2025-06-16-llms-redux.md
Normal file
@@ -0,0 +1,52 @@
|
||||
---
|
||||
lastmod: "2025-06-16T09:28:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2025-06-16T09:28:00.0000000+01:00"
|
||||
title: LLM effect on my programming (2025 edition)
|
||||
summary: "Since my last post about this in 2024-03, LLMs have become a distinctly positive addition to my ability."
|
||||
---
|
||||
|
||||
I wrote in a [previous post] that ChatGPT 4 was not clearly having a positive effect on my ability to code.
|
||||
|
||||
We're over a year later now, and we have larger models and reasoning models, and those *are* definitely an improvement over not having them.
|
||||
|
||||
# As documentation-knower
|
||||
|
||||
[Opus 4](https://claude.ai) has memorised how .NET works to a terrifying level of detail.
|
||||
In general, I can ask it minutely specific questions about the .NET runtime and it will simply answer them correctly.
|
||||
|
||||
I found Gemini 2.5 Pro to be fairly solid as an engine for searching the .NET spec (which I uploaded to its context), although still quite prone to making things up.
|
||||
|
||||
# As bug-finder
|
||||
|
||||
Opus 4 is also a powerful debugging engine, the first one that I actually trust to tell me the cause of a bug without handholding.
|
||||
|
||||
My workflow is:
|
||||
|
||||
* Put a whole load of tracing into my code
|
||||
* Run the code to obtain the tracing
|
||||
* Dump the trace into Opus 4 (using the ordinary chat interface) along with the code (usually from working in a Project, but sometimes also supplying the `git diff` against `main` if I believe that the diff introduced the bug), and ask it what is wrong.
|
||||
|
||||
This Just Works alarmingly frequently.
|
||||
|
||||
# As sanity-checker
|
||||
|
||||
At work, I feed many (non-sensitive) diffs automatically on `git push` through o3-mini-high with a prompt asking for a sanity check, obvious mistakes, etc.
|
||||
This catches a bunch of really dumb stuff like typos, and is particularly valuable for the cursed dynamic languages like Groovy where you can't know whether anything is wrong without running it.
|
||||
|
||||
I've found o3-mini-high less valuable for "proper" code reviews (the sort that you'd have an engineer do to identify design problems, for example).
|
||||
|
||||
# As LeetCode solver
|
||||
|
||||
o3 is better than I am at solving well-defined tightly-scoped LeetCode-style problems.
|
||||
(Fortunately for me, I basically never have to do this in real life.)
|
||||
|
||||
# Wrapup
|
||||
|
||||
Many of my points in the previous post are still correct: I still don't trust any of the models to write code for me, and for the same reasons.
|
||||
But now the utility of the models in other areas is much higher - I now have what I described in the "What I actually want from such a system" section of the previous post, for example - and I would absolutely recommend a Claude Pro subscription.
|
||||
(I'm even considering Claude Max; Opus consumes a *lot* of quota.)
|
||||
|
||||
[previous post]: {{< ref "2024-03-27-chatgpt-and-programming" >}}
|
19
hugo/content/posts/2025-06-17-woofware-expect.md
Normal file
19
hugo/content/posts/2025-06-17-woofware-expect.md
Normal file
@@ -0,0 +1,19 @@
|
||||
---
|
||||
lastmod: "2025-06-17T00:00:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2025-06-17T00:00:00.0000000+01:00"
|
||||
title: Announcing WoofWare.Expect
|
||||
summary: "A basic but functional expect-testing framework for F#."
|
||||
---
|
||||
|
||||
This is a linkpost for [WoofWare.Expect](https://github.com/Smaug123/WoofWare.Expect), which implements [expect testing](https://blog.janestreet.com/the-joy-of-expect-tests/) in an F# [computation expression](https://learn.microsoft.com/en-us/dotnet/fsharp/language-reference/computation-expressions).
|
||||
Go and read [the README on GitHub](https://github.com/Smaug123/WoofWare.Expect/blob/main/README.md) if you're interested.
|
||||
|
||||
As of this writing, the following are implemented:
|
||||
|
||||
* Comparison with existing inline snapshot
|
||||
* Custom formatting of the snapshot (fully general `'T -> string` to override the default `.ToString()`)
|
||||
* Specific support for JSON snapshotting using [FSharp.SystemTextJson](https://github.com/Tarmil/FSharp.SystemTextJson), including overriding the default ser/de settings
|
||||
* "Bulk update" mode for mass regenerating snapshots in a test fixture (which is only slightly unwieldy).
|
52
hugo/content/posts/2025-06-20-dotnet-runtime-learnings.md
Normal file
52
hugo/content/posts/2025-06-20-dotnet-runtime-learnings.md
Normal file
@@ -0,0 +1,52 @@
|
||||
---
|
||||
lastmod: "2025-06-20T00:00:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2025-06-20T00:00:00.0000000+01:00"
|
||||
title: Things I've learned about the .NET runtime
|
||||
summary: "While writing an MSIL interpreter, I discovered a bunch of unexpected things about the .NET runtime."
|
||||
---
|
||||
|
||||
# The JIT is actually required
|
||||
|
||||
(Or, in NativeAOT, other mechanisms that aren't just what's in System.Private.CoreLib.)
|
||||
|
||||
Some methods are JIT intrinsics, but I had previously thought the JIT was just an optimisation.
|
||||
This is false!
|
||||
For example, [Type.TypeHandle throws](https://github.com/dotnet/runtime/blob/ec11903827fc28847d775ba17e0cd1ff56cfbc2e/src/libraries/System.Private.CoreLib/src/System/Type.cs#L467-L471) and *must* be swapped out by the JIT.
|
||||
I don't know how this works on NativeAOT.
|
||||
|
||||
# `Console.WriteLine` is *incredibly* complicated
|
||||
|
||||
`Console.WriteLine` exercises at least the following:
|
||||
|
||||
* [Locking](https://github.com/dotnet/runtime/blob/4314ca34f3c620c180ef8f7fc28973af7ff47ad4/src/libraries/System.Private.CoreLib/src/System/SR.cs#L52)
|
||||
* [`Ldtoken` and type lookup](https://github.com/dotnet/runtime/blob/4314ca34f3c620c180ef8f7fc28973af7ff47ad4/src/libraries/System.Private.CoreLib/src/System/SR.cs#L84)
|
||||
* [Exception handling](https://github.com/dotnet/runtime/blob/4314ca34f3c620c180ef8f7fc28973af7ff47ad4/src/libraries/System.Private.CoreLib/src/System/SR.cs#L50)
|
||||
* [Byrefs](https://github.com/dotnet/runtime/blob/4314ca34f3c620c180ef8f7fc28973af7ff47ad4/src/coreclr/System.Private.CoreLib/src/System/Runtime/CompilerServices/RuntimeHelpers.CoreCLR.cs#L174)
|
||||
* [Delegates](https://github.com/dotnet/runtime/blob/4314ca34f3c620c180ef8f7fc28973af7ff47ad4/src/libraries/System.Private.CoreLib/src/System/Type.cs#L737)
|
||||
|
||||
# Top-level no-exception-handler behaviour change in net9
|
||||
|
||||
In earlier versions of .NET, `finally` blocks and `Dispose` methods were liable not to run when an unhandled exception terminates program execution.
|
||||
This was documented as implementation-defined behaviour [in the C# language spec](https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/language-specification/exceptions#214-how-exceptions-are-handled), though it doesn't appear to be documented elsewhere.
|
||||
|
||||
That implementation-defined behaviour means my programs basically all contain this boilerplate:
|
||||
|
||||
```fsharp
|
||||
let reallyMain argv =
|
||||
0
|
||||
|
||||
let main argv =
|
||||
try
|
||||
reallyMain argv
|
||||
with
|
||||
| _ -> reraise ()
|
||||
```
|
||||
|
||||
However, in .NET 9 (I think) this behaviour appears to have been changed, and now the .NET runtime does the obvious thing - `Dispose` methods run and `finally` blocks execute even when an exception causes shutdown!
|
||||
|
||||
# The .NET runtime doesn't actually model uint32
|
||||
|
||||
Neither [on the eval stack](https://github.com/Smaug123/WoofWare.PawPrint/blob/c747d6eb3a2d138debce9c6cca763287bb21a7a9/WoofWare.PawPrint/EvalStack.fs#L60) nor [as a CLI intrinsic type](https://github.com/Smaug123/WoofWare.PawPrint/blob/c747d6eb3a2d138debce9c6cca763287bb21a7a9/WoofWare.PawPrint/BasicCliType.fs#L39) are `uint32` or `uint64` modelled; they're stored as `int32` and `int64` with two's complement.
|
71
hugo/content/posts/2025-06-21-python-pitfalls.md
Normal file
71
hugo/content/posts/2025-06-21-python-pitfalls.md
Normal file
@@ -0,0 +1,71 @@
|
||||
---
|
||||
lastmod: "2025-06-21T00:00:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2025-06-21T00:00:00.0000000+01:00"
|
||||
title: Some Python surprises
|
||||
summary: "I am forever astonished that people describe Python as a simple language. Here are some of the things I found very surprising about it."
|
||||
---
|
||||
|
||||
# An assertion that can pass
|
||||
|
||||
The following code can succeed without throwing, although it's structurally very unintuitive why:
|
||||
|
||||
```python
|
||||
try:
|
||||
with Foo():
|
||||
result = blah
|
||||
except ValueError:
|
||||
result.baz()
|
||||
else:
|
||||
assert False
|
||||
```
|
||||
|
||||
This is an entry in the "you need to know how everything is implemented before you can understand it" registry: desugaring this makes it clear how it could pass (if the `__exit__` method on the context manager `Foo` throws `ValueError`), but the syntax is practically purpose-built to make it hard to see that it can.
|
||||
|
||||
# You can't monkey-patch `__add__` on an instance
|
||||
|
||||
Special methods like `__add__` are looked up on the *class*, not the instance.
|
||||
(See [Special method lookup](https://docs.python.org/3/reference/datamodel.html#special-lookup).)
|
||||
|
||||
Again, `a + b` *looks* simple, but is in fact weirdly complex and its semantics are highly unintuitive.
|
||||
|
||||
# Integers are sometimes ref-equal and sometimes not
|
||||
|
||||
```python
|
||||
a = 10000 # big enough to escape the small integer cache
|
||||
b = 10000
|
||||
a is b # False
|
||||
10000 is 10000 # True
|
||||
```
|
||||
|
||||
We're seeing a cpython optimisation in the `10000 is 10000` check: instances of literals in the same statement may be reused within that statement.
|
||||
|
||||
Personally I expected some amount of referential transparency around the `is` statement, but it turns out not!
|
||||
|
||||
# Dataclasses are implemented with strings and `eval`
|
||||
|
||||
I was astonished - it's literally [just weaving some strings together](https://github.com/python/cpython/blob/4eab9da960d6944546baa76e3eed56b809ea8ec0/Lib/dataclasses.py#L496) and [calling `exec`](https://github.com/python/cpython/blob/4eab9da960d6944546baa76e3eed56b809ea8ec0/Lib/dataclasses.py#L498)!
|
||||
I don't know it to be *incorrect*, but the sheer bluntness of the instrument amazed me.
|
||||
|
||||
# The six builtins
|
||||
|
||||
There are [six builtins in Python](https://docs.python.org/3/library/constants.html).
|
||||
`True`, `False`, and `None` are unsurprising; `__debug__` is a bit odd to be one of the six privileged constants, but fine, I guess.
|
||||
|
||||
`NotImplemented` is bizarre and suggests whole areas that you should ignore to maintain your sanity.
|
||||
|
||||
Then there's `Ellipsis`, which is the object for which `...` is sugar.
|
||||
It can, for some reason, be reassigned (when `True` etc can't)?
|
||||
(This reassignment doesn't actually have an effect, because the syntax `...` somehow manages to evaluate to `Ellipsis` even when the name `Ellipsis` itself is being used for a global. I still don't really understand what's happening here.)
|
||||
|
||||
# Multiprocessing
|
||||
|
||||
Oh *lord* the number of problems I've seen with the passing of objects across `multiprocessing` boundaries.
|
||||
|
||||
# Annoying wart: violation of Liskov substitution principle with keyword args
|
||||
|
||||
When overriding a method on a class, you can't rename the arguments - even to underscores - because argument names are part of a method's usable API surface.
|
||||
(Pyright warns about this.)
|
||||
The safe way to mark a variable as unused in an overridden method is to have the first line of the method be a useless assignment, e.g. `_ = arg_name`.
|
98
hugo/content/posts/2025-08-29-boolean-blindness.md
Normal file
98
hugo/content/posts/2025-08-29-boolean-blindness.md
Normal file
@@ -0,0 +1,98 @@
|
||||
---
|
||||
lastmod: "2025-08-29T10:23:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2025-08-29T10:23:00.0000000+01:00"
|
||||
title: Boolean blindness
|
||||
summary: "Why booleans are an advanced technique that should be used with care."
|
||||
---
|
||||
|
||||
*Written off the back of [That Boolean Should Probably Be Something Else](https://ntietz.com/blog/that-boolean-should-probably-be-something-else/).*
|
||||
|
||||
Booleans often throw away information.
|
||||
Consider *not* throwing away that information unless you have some good reason to do so.
|
||||
|
||||
I know of two main use cases for booleans: indicating that some condition was met, or indicating that some state is desired.
|
||||
Here's what it means to "throw away information" in those two cases.
|
||||
|
||||
# Boolean to indicate that some condition was met
|
||||
|
||||
A boolean means "the condition was met" or "the condition was not met".
|
||||
But why are you justified in making a judgement either way on the statement "the condition was met"?
|
||||
It's usually because you ran some check that gave you evidence that the condition was met (that is, `true`), or that you have no evidence that the condition was met, or even counter-evidence (that is, `false`).
|
||||
|
||||
So why are you telling me that you ran a check and it gave you evidence?
|
||||
Just give me the evidence instead!
|
||||
|
||||
## Timestamps
|
||||
|
||||
The article linked at the top uses the example of storing timestamps in a database, instead of simply a `true` value, indicating "this event happened".
|
||||
That's an optimised case of this pattern.
|
||||
To go purely into the "give me the evidence" mode, you would actually store not a timestamp but a complete representation of the event - who did it, when, and why, for example.
|
||||
But in reality, that's just quite a lot of information, and some of it might be sensitive; so we truncate it.
|
||||
|
||||
In the boolean world, we truncate it all the way, throwing away *all* the information beyond the bare fact that the event happened.
|
||||
|
||||
But in the timestamps world, we truncate it down to just the timestamp component.
|
||||
Now, when someone asks us whether the event happened, we can give them (a small amount of) evidence that it did happen, not just a yes/no answer.
|
||||
|
||||
## Enums
|
||||
|
||||
The article then discusses the question "is this user an admin?".
|
||||
In the booleans world, again we can only say "yes" or "no".
|
||||
But if we start replying with evidence instead of brute fact, we can give the type of the user: we can say "this user is a guest" and allow the caller to deduce that the user is not an admin.
|
||||
|
||||
That is, we can answer not with a boolean but with an enum.
|
||||
|
||||
## Thresholds
|
||||
|
||||
Claude commented on an earlier draft of this essay, suggesting the case of `x > 5` as a boolean result that doesn't have rich intermediate state worth capturing.
|
||||
For such a simple check, though, I'd at least *consider* not passing the boolean around, but instead simply passing `x` and allowing the callee to perform the check explicitly.
|
||||
Perhaps you don't need to anticipate what the callee needs from you, but instead give them *everything* and let them choose.
|
||||
|
||||
# Boolean to indicate that some state is desired
|
||||
|
||||
Config objects are often just big bags of ints and booleans and things.
|
||||
This can be a decent representation, but be aware that it is often artificially restrictive: booleans are often the shadow of more general features that you might be able to offer with minimal extra effort.
|
||||
The boolean causes the user to throw away information about their intent, in conforming to your API.
|
||||
|
||||
For example, `enableLogging = false` is OK, but `logThreshold = LogLevel.None` both expresses the same intent and allows more use cases.
|
||||
|
||||
Or a boolean `putSpacesAtEnd : bool` could instead be `suffix : string`: the boolean API is a constraint on what the user can supply, when you might instead prefer to offer the more general feature.
|
||||
|
||||
Whether a more general representation is *desirable* in any particular case is another matter - API design is hard - but be aware that it's an available option, that makes the meaning much more explicit!
|
||||
|
||||
# Philosophical point
|
||||
|
||||
The type `bool` is logically equivalent to `Option<unit>`.
|
||||
If your usage of `bool` is one for which `Option<unit>` is a natural model, then it's fairly obvious that you've thrown stuff away: `unit` is a very unusual type because it's almost useless.
|
||||
In this light, it's more clearly a candidate for an optimisation that has been *prematurely* applied.
|
||||
|
||||
Here are two equivalent type signatures, one making it clear that information has been thrown away:
|
||||
|
||||
```fsharp
|
||||
let hasPermission (userId : int) : bool = ...
|
||||
```
|
||||
|
||||
```fsharp
|
||||
let hasPermission (userId : int) : Option<unit> = ...
|
||||
```
|
||||
|
||||
And here is where we *don't* throw away that information:
|
||||
|
||||
```fsharp
|
||||
let permission (userId : int) : Option<UserPermissions> = ...
|
||||
```
|
||||
|
||||
The "boolean-equivalent" example has effectively just shoved the result of `permission` through `Option.map ignore`; and `ignore` is a function that should always provoke a *little* unease.
|
||||
|
||||
# Good reasons to use booleans
|
||||
|
||||
Throwing stuff away is a standard performance technique, and indeed booleans can be good for performance: for example, you might perform some computation over the evidence, rendering the result as a boolean, and then repeatedly refer to the result of that computation.
|
||||
Like any optimisation, it's often easier to understand if it's constrained in its scope: creating a boolean internally to a function, that doesn't escape, is almost always completely fine and natural.
|
||||
|
||||
Sometimes you might want to throw away information for other reasons: perhaps you specifically need not to store that user's identifying information.
|
||||
|
||||
Booleans also give you a chance to name some condition explicitly, if that would help: instead of `if fooBar x then ...`, it might be more readable to have `let isFrobnicated = fooBar x; if isFrobnicated then ...`.
|
||||
There may be other ways to do this that don't throw away as much information, but they're often much more heavyweight.
|
61
hugo/content/posts/2025-10-06-llm-concerns.md
Normal file
61
hugo/content/posts/2025-10-06-llm-concerns.md
Normal file
@@ -0,0 +1,61 @@
|
||||
---
|
||||
lastmod: "2025-10-06T22:59:00.0000000+01:00"
|
||||
author: patrick
|
||||
categories:
|
||||
- programming
|
||||
date: "2025-10-06T22:59:00.0000000+01:00"
|
||||
title: LLMs sanding off the edges
|
||||
summary: "I'm a bit concerned about a failure mode that is increasingly arising with LLMs: the divorce between how something looks, and its quality."
|
||||
---
|
||||
|
||||
*This post is in the form of several distinct facts, and then a synthesis of vague creeping unease.*
|
||||
|
||||
# A sense of taste may be trained to match style
|
||||
|
||||
It used to be the case that you could get some sense of something's quality by seeing whether it "looked OK".
|
||||
This was certainly not a perfect way to judge something - indeed, proverbially one must not judge a book by its cover - but the years have granted me what I'll call a somewhat idiosyncratic "sense of taste", and my System 1 feels that things are "tasteful" or not.
|
||||
|
||||
System 1 isn't great at thinking about complicated things, so I suspect a lot of what it's picking up on is style; sometimes literally in the sense of "which words were used", but sometimes of the form "this design is composed of primitives that feel a bit grating", and sometimes "there's a bug or inconsistency here which I think you wouldn't have written if you truly understood the platonically correct form of the problem".
|
||||
|
||||
# LLMs can translate style
|
||||
|
||||
LLMs are excellent at translating between styles.
|
||||
|
||||
* "This is how I'd describe to a friend what I want to say; please convert it to [Dangerous Professional](https://x.com/patio11/status/1162561822248992768?lang=en-GB)."
|
||||
* "Please select your words and phrasing with artistry, using elegant and precise learnèd language, picking the right words for the job assuming the reader has an excellent vocabulary and will understand any cultural or scientific allusions you make."
|
||||
* "Condense this fluffy blog into three sentences."
|
||||
|
||||
# Sonnet 4.5 hits a specific style
|
||||
|
||||
The first LLM I've interacted with which really nails the "experienced and competent software engineer" vibe is Sonnet 4.5.
|
||||
I still don't trust it to write code without supervision - I've not been super happy with its output - but it's the first model which has convincingly and (I think) correctly helped with the architecture of my current side project.
|
||||
It speaks with a strong air of competence, where previous models spoke with the air of someone trying to project competence; it's actually coming close to a superstimulus for me in that regard (although I have observed it making design errors, so System 2 at least knows I need to keep fact-checking).
|
||||
|
||||
# GPT-5 is great at spotting bugs
|
||||
|
||||
For a while now at work I've been using a little PR-review bot that basically feeds stuff into GPT-5.
|
||||
It's really excellent, and I've started using something very similar at home.
|
||||
I might even say GPT-5 is in the top 5% of code reviewers for the specific sub-slice of code review that is "find the localised non-architectural bugs in this code, and proof-read it for typos, omissions, and documentation inconsistencies".
|
||||
|
||||
I do *not* trust GPT-5 to have any design taste (I trust Sonnet 4.5 a little more on that axis).
|
||||
|
||||
# "The Most Forbidden Technique"
|
||||
|
||||
Zvi Mowshowitz uses the term "The Most Forbidden Technique" to refer to the AI alignment strategy of identifying some signal of misalignment, and then training your model to stop displaying that signal, in the hope that this will train away the misalignment.
|
||||
(Naturally the expected effect is actually to train the model to hide the misalignment from you, all while greatly improving your "alignment metrics"; hence the "Most Forbidden" name.)
|
||||
|
||||
# The ominous synthesis
|
||||
|
||||
But I've been getting nervous about using GPT-5 as a reviewer in conjunction with using Claude Code to *write* code.
|
||||
Naively adopting a loop of "Claude writes some code; GPT-5 reviews it; repeat", you can eventually end up with code that both Claude and GPT-5 are happy with.
|
||||
But I fear that the act of using GPT-5 to "sand away the rough edges", to catch the obvious bugs, may be effectively resulting in a "style translation", from "original Claude output" to "code that's sanitised to remove any bugs GPT-5 can see".
|
||||
Since GPT-5 doesn't have good taste, I expect code that has been iteratively mashed into GPT-5's image to be kind of bad: weird designs that don't scale to future features, or whatever.
|
||||
|
||||
And the act of removing the bugs that were visible to GPT-5 is the act of *making the code look more plausible*.
|
||||
I think a loop like this is liable to produce code that's wrong, or badly designed, but that's palatable to my human taste.
|
||||
Is this loop "Most Forbidden"?
|
||||
Will it produce good code, or is it instead incrementally iterating towards hiding the badness?
|
||||
|
||||
The history of computing has showed us again and again that what we think of as "human intelligence" is actually many properties that correlate in humans but are not intrinsically linked.
|
||||
(Solving chess, solving differential equations, writing short stories that win competitions - these all used to be correlated skills, but now we solve them all individually with very different systems.)
|
||||
I am concerned that another set of these properties is becoming apparent where the link is coming adrift: "the code looks plausible" and "the code is correctly designed".
|
@@ -1,5 +1,5 @@
|
||||
---
|
||||
lastmod: "2023-09-02T13:31:58.0000000+01:00"
|
||||
lastmod: "2024-09-27T19:31:58.0000000+01:00"
|
||||
title: My reading list
|
||||
author: patrick
|
||||
layout: page
|
||||
@@ -12,17 +12,19 @@ This page holds a list of the books I am reading, and a list of books I have rea
|
||||
* [Don't Shoot the Dog] - Nonfiction book on animal training and operant conditioning: the art of getting obedience (from animals and humans alike) without using punishment. It has clarified my understanding of a whole variety of phenomena related to motivation and so on.
|
||||
* [Inadequate Equilibria](https://equilibriabook.com/), by Eliezer Yudkowsky. Fascinating look at why everything is broken and nobody can fix anything. I'm a sucker for ideas presented as Socratic dialogue.
|
||||
* [A Person Paper on Purity in Language][Person Paper], by Douglas Hofstadter (linked to the Wayback Machine version) - it is pretty shocking to realise just how much the English language discriminates against females, and how routine it is. Whether or not you think it's an issue, this is an excellent satire.
|
||||
* [Gödel, Escher, Bach], by Douglas Hofstadter (an incredible book on pretty much everything - possibly the most meta thing I've ever read). Revisiting it now, I realise that I already know quite a lot of the mathematical content from different sources, but then to the extent that I specialised in anything during Part III, I specialised in logic.
|
||||
* The [LessWrong] entry on [Cached Thoughts] - of the many fascinating LessWrong entries, this is probably the one that had the most profound and immediate effect on my thinking. I don't know if that effect was just a culmination of my previous readings, but this was the moment that I really got the idea that "human thinking was not designed, it's a bit of a miracle that we can think at all, and there are steps we can take to make our thinking less sloppy".
|
||||
* [Harry Potter and the Methods of Rationality][HPMoR] - it's a very long fanfic about what might happen if Harry came from a different intellectual background and if every character were actually *trying* at life. It successfully manages to find coherent explanations for many of the random inconsistencies of the original. Give it a try - if you don't like it by the time we reach Hogwarts, feel free to stop. It's got its own subreddit, of which the most pertinent post is probably [what HPMoR is about][HPMoR subreddit]. People either find HPMoR horribly dull or absolutely incredible, as far as I can tell.
|
||||
* Another LessWrong entry, this one on [effective altruism][EA LessWrong] ([factual summary][EA Wikipedia] at Wikipedia) - the holy grail of charitable giving is surely to get maximum bang for your buck. (On a vaguely related note, a link that for ethical reasons doesn't make it onto the list but seems worryingly insightful is a post from some random blog describing a horrible unit of currency, the "dead child", being the amount of money required to save the life of one child's life through charity - it makes very uncomfortable reading, so consider yourself duly warned that you might not want to read it. The idea is to make numbers like "£250,000 spent on a luxury dog kennel" make *sense* to us. The link is [this post on a horrible unit of currency][dead child].)
|
||||
* [The Name of the Wind] by Patrick Rothfuss is the fantasy book with the best story I've ever read. It's the start of a trilogy (the Kingkiller Chronicle) and, if you like fantasy (and maybe even if you don't), is a stupendously well-told book. I gave it to a friend who professes never to read, and I didn't see em again for another week. It's not uniformly liked, though: another friend described it as an obvious self-insert wish-fulfillment fantasy from beginning to end.
|
||||
* [Three Men in a Boat] - the funniest book I've read. After the first two pages, I can barely move after having dissolved in laughter, and it just gets better. I was banned from reading this in the presence of other people, probably because I was having too much fun.
|
||||
* [Midnight's Children] - justly famous mystical fiction, set in twentieth century India. Wonderful book, beautiful turns of phrase and awe-inspiring plot.
|
||||
* [Flowers for Algernon] - possibly the saddest book I've ever read, and at least one other person has agreed with me.
|
||||
* [Turn The Ship Around!] - book on the nature of leadership. Central thesis: if a great commander leaves a ship and the ship falls apart straight away, in what sense is it reasonable to say they were a "great" commander? Its teachings are weirdly similar, in fact to [Never Split the Difference], though they're approaching two very different problems.
|
||||
* The Debt to Pleasure, by John Lanchester - fiction/cookbook. One of the most erudite books I've read; this book really benefits from being read on a Kindle, with its inbuilt dictionary. A joy to read, excellent characterisation of the narrator. I'd advise not Googling the book before you read it (and ideally don't even read the blurb); there are spoilers to be found, and I think the best experience of this book would be to go in completely blind. Do read it, though; it has many beautiful turns of phrase, and is just generally extremely well written.
|
||||
* [Jonathan Strange & Mr Norrell], by Susanna Clarke. This book alternates between hilarious, whimsical, and deeply ominous or wild (like you're seeing the tiniest cross-section of a huge incomprehensible thing). In general style it's like E Nesbit if she were writing extremely competently for non-children. I was glued to it, and laughed out loud at regular intervals.
|
||||
* [The Name of the Wind] by Patrick Rothfuss is the fantasy book with the best story I've ever read. It's the start of a trilogy (the Kingkiller Chronicle) and, if you like fantasy (and maybe even if you don't), is a stupendously well-told book. I gave it to a friend who professes never to read, and I didn't see them again for another week. It's not uniformly liked, though: another friend described it as an obvious self-insert wish-fulfillment fantasy from beginning to end.
|
||||
* [Gödel, Escher, Bach], by Douglas Hofstadter (an incredible book on pretty much everything - possibly the most meta thing I've ever read). Revisiting it now, I realise that I already know quite a lot of the mathematical content from different sources, but then to the extent that I specialised in anything during Part III, I specialised in logic. For some reason many people don't like this book, and it is certainly very long, but decades later I still remember e.g. the Contracrostipunctus as being just mind-blowing.
|
||||
* [Harry Potter and the Methods of Rationality][HPMoR] - it's a very long fanfic about what might happen if Harry came from a different intellectual background and if every character were actually *trying* at life. It successfully manages to find coherent explanations for many of the random inconsistencies of the original. Give it a try - if you don't like it by the time we reach Hogwarts, feel free to stop. It's got its own subreddit, of which the most pertinent post is probably [what HPMoR is about][HPMoR subreddit]. People either find HPMoR horribly dull or absolutely incredible, as far as I can tell.
|
||||
|
||||
[Jonathan Strange & Mr Norrell]: https://en.wikipedia.org/wiki/Jonathan_Strange_%26_Mr_Norrell
|
||||
[Gödel, Escher, Bach]: https://en.wikipedia.org/wiki/G%C3%B6del,_Escher,_Bach
|
||||
[LessWrong]: http://www.lesswrong.com
|
||||
[Cached Thoughts]: http://lesswrong.com/lw/k5/cached_thoughts/
|
||||
@@ -40,13 +42,13 @@ This page holds a list of the books I am reading, and a list of books I have rea
|
||||
|
||||
# Currently reading
|
||||
|
||||
* In Praise of Shadows, by Junichiro Tanizaki
|
||||
* Building Secure and Reliable Systems, by the Google SRE team. This is genuinely a textbook, so it's quite slow going. A lot of this is *very* Google-centric, where it's assumed that everything is a microservice and any given query to a service will require hundreds of RPC calls. However, the general lessons seem to be valuable.
|
||||
* High Output Management, by Andy Grove
|
||||
|
||||
# Bought and ready to read
|
||||
|
||||
* The Attention Merchants, by Tim Wu
|
||||
* American Gods, by Neil Gaiman
|
||||
* 84, Charing Cross Road, by Helene Hanff
|
||||
|
||||
# To read
|
||||
|
||||
@@ -57,32 +59,41 @@ This page holds a list of the books I am reading, and a list of books I have rea
|
||||
* Linear Algebra Done Right, by Sheldon Axler (non-fiction, apparently the definitive textbook on linear algebra)
|
||||
* Reinventing Discovery, by Michael Nielsen (non-fiction: how modern-day collaboration could remake science)
|
||||
* The Strategy of Conflict, by Tom Schelling (book about game theory by a professional game theorist)
|
||||
* Slaughterhouse Five, by Kurt Vonnegut (depressing fiction)
|
||||
* The Quincunx, by Charles Palliser (Wilkie Collins-esque Dickensian fiction)
|
||||
* Genesis, by Bernard Beckett (future ?dystopian fiction)
|
||||
* Small Gods, by Terry Pratchett (fiction)
|
||||
* Linked, by Albert Lazlo Barabassi (study of networks)
|
||||
* Libriomancer, by Jim C Hines (fantasy fiction)
|
||||
* Breakfast of Champions, by Kurt Vonnegut (fiction of some sort)
|
||||
* Nature via Nurture, by Matt Ridley (study of interaction of genes and environment and development)
|
||||
* The Culture series, by Iain M Banks
|
||||
* A Place of Greater Safety, by Hilary Mantel
|
||||
* So Good They Can't Ignore You, by Cal Newport
|
||||
* 84, Charing Cross Road, by Helene Hanff
|
||||
* Black Box Thinking, by Matthew Syed
|
||||
* Impro, by Keith Johnstone
|
||||
|
||||
# Read some of and then put down
|
||||
|
||||
* [Legal Systems Very Different From Ours](http://web.archive.org/web/20220806200919/http://www.daviddfriedman.com/Legal%20Systems/LegalSystemsContents.htm): it's all in the title, really. Case studies of different legal systems, by the economist David Friedman. I read the first third or so of this, and it was a very interesting set of things to think about. How can a civilisation run itself without our systems of law courts? There have been many answers through the centuries.
|
||||
* [Extraordinary Popular Delusions and the Madness of Crowds](https://en.wikipedia.org/wiki/Extraordinary_Popular_Delusions_and_the_Madness_of_Crowds), by Charles Mackay - nonfiction about the ways in which the crowd can get things wrong. This book is very, very long; I made it through the first half of the first volume before giving up. A pretty eye-opening book, though: Mackay's description (from the year 1841!) of the South Sea Bubble is almost indistinguishable from the recent cryptocurrency crazes, right down to an extensive list of some of the crazy companies that were eventually made illegal ("for improving the art of making soap", "for a wheel of perpetual motion", "for extracting silver from lead", "for a grand American fishery") - all very reminiscent of the various shitcoins.
|
||||
* [Quantum Computing for the Very Curious](https://quantum.country/qcvc): essay to teach the fundamentals of quantum computing, with embedded spaced repetition cards.
|
||||
* True Names, by Vernor Vinge
|
||||
* Diaspora, by Greg Egan
|
||||
|
||||
# Have read
|
||||
|
||||
* The Diamond Age (A Young Lady's Illustrated Primer), by Neal Stephenson. After so long having the Primer in my vocabulary without having read the source of the meme, I finally did it when encountering Andy Matuschak's [Exorcising Us of the Primer](https://andymatuschak.org/primer/) and decided that actually reading the book was a prerequisite. Good book, maybe 7/10. Loved the presentation of how globalised societies can factor into tribes; loved the central idea of a device for rapidly teaching a child to be a fully rounded adult, and the periodic demonstrations that Nell is learning real skills in the real world thanks to her study. Slightly perturbed by the strange sex cult that appears to be obligatory in this kind of book. I'm excited to receive my [Daylight Computer](https://daylightcomputer.com/) because I hope it (plus [Claude](https://claude.ai)) can be some shadow of the Primer, although I've actually had some trouble finding a prompt that has Claude come close to enacting the Primer.
|
||||
* We Could Be So Good, by Cat Sebastian. Heartwarming and cute! I've never actually read anything that explored this period of gay history, and it was very gripping.
|
||||
* A Fire Upon The Deep, by Vernor Vinge. Really enjoyed this. Lots of little throwaway bits that just stayed with me ("of course our culture has spent vast amounts of leisure time working out the shortest route to bootstrapping Civilisation on a new planet", the Tines' physical construction, the idea that some cultures are impossible in some regimes of physics).
|
||||
* In Praise of Shadows, by Jun'ichirō Tanizaki. Obviously heavily romanticised, but I enjoyed this a lot. (In fact I have recently been using an audiobook of this, narrated by David Rintoul, to fall asleep to; his voice and the content are both great for that.) Some of this is actually rather sad: an old man, seeing a Western future overtake the culture of which he is a part. The aesthetic pictures his words paint are often beautiful.
|
||||
* Slaughterhouse Five, by Kurt Vonnegut. Indeed pretty depressing! I actually found this quite gripping, like you were really getting to know the protagonist, which I wasn't expecting. Felt like it did a good job of portraying a broken numbed soldier.
|
||||
* Accelerando, by Charles Stross. Weird book, which I guess is the point. I had to struggle through the first few decades, because I found the style very offputting. (Even later, the gratuitous use of jargon like "ackles": I know what an [ACL](https://en.wikipedia.org/wiki/Access-control_list) is, and perhaps such words will indeed be common in the future, but it really felt like the author was "just showing off" some basic computing knowledge.) The general idea was pretty cool, though!
|
||||
* Red Side Story, by Jasper Fforde, the second in the Shades of Grey series. I'm afraid I didn't enjoy this as much as the first book. There's a bunch of gripping worldbuilding here, but the first book was full of endearing naïvité from all characters, and this one was more "serious characters who happen to be in a whimsical setting".
|
||||
* Now It Can Be Told, by General Groves. I wasn't as taken with this book as I was with _The Making of the Atomic Bomb_; this book contained a bunch more individual callouts and "this particular person did a brilliant job" than are really necessary fifty years later. Still a bunch of excellent quotes, e.g. around how the US government paid him $35mm personally to get around procurement rules; _that's_ how you really try to win when you have trust in your employees!
|
||||
* [Planecrash](https://www.projectlawful.com/). It turns out that self-insert sex fiction is not actually much better if you are self-aware about it. There's some excellent readable story in here, but you have to wade through hours and hours of sex and sex-politics to find it. Absolutely not worth reading from start to finish; just find someone to link the "dath ilan society description and basic coordination classes" highlights. These authors *really* need an editor; the work is literally at least 10x too long.
|
||||
* Tao Te Ching, by Lao Tzu, interpreted by Ursula Le Guin. Her poetry is *beautiful*. Not an easy read: I felt I was rushing at two hours for thirty-odd chapters to get the bare minimum of comprehension, but it's deep enough that each chapter would reward an hour of study. I'm currently going to improv classes, and this feels like it is secretly a book about improv!
|
||||
* Piranesi, by Susanna Clarke. Wonderful. Ethereal (but in a sort of grounded way), beautiful.
|
||||
* Impro, by Keith Johnstone. The first half of this book is very interesting, easy reading, and I suspect rather useful; I believe it's been helpful in my introductory improv lessons. (I don't know how useful it will be without practice of some sort, but I also suspect it's possible to practise without going full improv-classes.) The second half is an exploration of mask work, Commedia dell'arte, trance states, and possession, and I found that much less interesting; the first half captured me, and the second half did not.
|
||||
* Diary of a Provincial Lady, by E. M. Delafield. Hilarious. I was laughing out loud every few pages. Strong recommend. This was so like real life.
|
||||
* [Folding Beijing](https://www.uncannymagazine.com/article/folding-beijing-2/), by Hao Jingfang. Interesting short story, although really I think it could have been half the length without losing much; the premise is not exactly complicated, and the story is really just a brief exploration of the premise.
|
||||
* Permutation City, by Greg Egan. Cool book! It's interesting to think about why I find the core premise implausible, but it certainly seemed novel to me. Pretty gripping. It seems to pair quite nicely with Robin Hanson's _The Age of Em_, which would be a nonfiction accompaniment.
|
||||
* Building Secure and Reliable Systems, by the Google SRE team. This is genuinely a textbook, so it's quite slow going. A lot of this is *very* Google-centric, where it's assumed that everything is a microservice and any given query to a service will require hundreds of RPC calls. However, the general lessons seem to be valuable.
|
||||
* A Handful of Dust, by Evelyn Waugh. Good Lord this is bleak. The kind of book which would definitely reward rereading; I feel like I got about a quarter of what was going on, there was so much subtext. Oddly a fun read, though!
|
||||
* The Remains of the Day, by Kazuo Ishiguro. Short, sweet, and tragic. Simple throwaway lines become retrospectively rich in meaning as you read on. Spoiler in rot13: gur haeryvnoyr aneengbe tvirf guvf fgenatr frafr bs grafvba guebhtubhg.
|
||||
* A Pocketful of Rye, by Agatha Christie. One of the Marples. Christie is generally excellent light reading, perhaps partly because her writing is all so familiar. This particular book I first read as an early teenager, but recently reread as part of a marathon in which we watched two adaptations and listened to a radio version; the exercise was actually quite interesting in seeing what the various adapters (and actors) chose to pull out.
|
||||
* The Hornblower series, by CS Forester. I read some of these over and over as a child, and they bear rereading as an adult. The later books become less interesting to me, though, as Hornblower is promoted and gets further from the action.
|
||||
* The Hornblower series, by CS Forester. I read some of these over and over as a child, and they bear rereading as an adult for some naval action from the point of view of a competent man in generally miserable circumstances. The later books become less interesting to me, though, as Hornblower is promoted and gets further from the action.
|
||||
* Station Eleven, by Emily St. John Mandel.
|
||||
* Sea of Tranquility, by Emily St. John Mandel. Short and enjoyable mystery book set across pretty large tracts of space and time. I agree with a GoodReads reviewer, though, that Cloud Atlas does a very similar thing but much better.
|
||||
* Cloud Atlas, by David Mitchell. Beautiful book. Each story is gripping in its own way. Imaginative.
|
||||
@@ -93,7 +104,7 @@ This page holds a list of the books I am reading, and a list of books I have rea
|
||||
* The SRE handbook, by the Google Site Reliability Engineering team. Super interesting. A bunch of this stuff is slowly seeping into how I think about software, with the main obvious effect on my work being that I am now much more strongly inclined to notice when I'm doing a task that could in principle be automated. However, if you're not Google, some of this stuff is probably not aimed at you; they have unique problems. I think this is worth reading for any software engineer.
|
||||
* Leadership Is Language, by David Marquet (sequel to Turn The Ship Around, discussed above). Writing this review some time after reading the book, I don't actually retain anything specific from it, but I have certainly cargo-culted some of what I know was in it - e.g. "the way you phrase things can have a strong effect on the responses you get, so think about what you're implicitly telling people when you choose to phrase something a particular way". I suspect this was worth reading.
|
||||
* The Elephant In The Brain, by Robin Hanson. This is one of those books that colours everything you think about afterwards. The premise is that all aspects of humans can be explained, and more specifically that disparate things like art and modern medicine are in large part about social signalling. Very mind-expanding, very cynical-economist. Recommended.
|
||||
* The Making Of The Atomic Bomb, by Richard Rhodes (non-fiction, about what it says on the tin). Absolutely brilliant book. Strong layman's introduction to the nuclear physics relevant to building the bomb, and plenty of humanising stories. I was not expecting to read so many pages of _Mein Kampf_ quoted in this book (I think most readers probably accept the "Hitler evil" premise). There was a chapter devoted to eyewitness accounts of the destruction of Hiroshima, which was literally sickening to read. Highly recommended.
|
||||
* The Making Of The Atomic Bomb, by Richard Rhodes (non-fiction, about what it says on the tin). Absolutely brilliant book. Strong layman's introduction to the nuclear physics relevant to building the bomb, and plenty of humanising stories. I was not expecting to read so many pages of _Mein Kampf_ quoted in this book (I think most readers probably already accept the "Hitler evil" premise without pages of explanation). There was a chapter devoted to eyewitness accounts of the destruction of Hiroshima, which was literally sickening to read. Highly recommended.
|
||||
* If on a winter's night a traveller, by Italo Calvino. I couldn't really put this down, but I also wouldn't describe it as "gripping". There are some lovely vignettes in here, and many of the stories I'd love to read more of. Recommended.
|
||||
* Infinite Jest, by David Foster Wallace. This was an infuriating book. It's so, so long, and it's so nearly excellent throughout, but honestly I got to the end and thought it wasn't worth the time I'd put into it. Some lovely bits of wordplay, a fairly good plot, excellent characterisation; if it were half the length this would be a snap recommendation.
|
||||
* To Be Taught If Fortunate, by Becky Chambers. Basically four vignettes about extrasolar worlds, each very different. Its core premise is rather different from any sci-fi I have ever read before, and the whole book is gentle.
|
||||
@@ -140,6 +151,13 @@ This page holds a list of the books I am reading, and a list of books I have rea
|
||||
* Consciousness Explained, by Daniel Dennet (explaining the Multiple Drafts view of consciousness) - very interesting book, thought-provoking, and I've absorbed some of its contents into my world-view.
|
||||
* The Mind's I, by Douglas Hofstadter and Daniel Dennet (various musings on consciousness and the nature of the self) - this is a collection of pieces by other people, with commentary by Hofstadter and Dennet, which wasn't quite what I expected. I'd read several of the pieces before, although Hofstadter sparkled just as he usually does. I got bored after about the tenth excerpt, but certainly up to the eighth my attention was held.
|
||||
|
||||
# Read some of and then put down
|
||||
|
||||
* [The Complete Cosmicomics](https://en.wikipedia.org/wiki/The_Complete_Cosmicomics), by Italo Calvino. Whimsical indeed, but oddly enough I started finding them rather samey after about a third of the way through. Worth reading a couple of them (no particular ones seemed better than the others to me).
|
||||
* [Legal Systems Very Different From Ours](http://web.archive.org/web/20220806200919/http://www.daviddfriedman.com/Legal%20Systems/LegalSystemsContents.htm): it's all in the title, really. Case studies of different legal systems, by the economist David Friedman. I read the first third or so of this, and it was a very interesting set of things to think about. How can a civilisation run itself without our systems of law courts? There have been many answers through the centuries.
|
||||
* [Extraordinary Popular Delusions and the Madness of Crowds](https://en.wikipedia.org/wiki/Extraordinary_Popular_Delusions_and_the_Madness_of_Crowds), by Charles Mackay - nonfiction about the ways in which the crowd can get things wrong. This book is very, very long; I made it through the first half of the first volume before giving up. A pretty eye-opening book, though: Mackay's description (from the year 1841!) of the South Sea Bubble is almost indistinguishable from the recent cryptocurrency crazes, right down to an extensive list of some of the crazy companies that were eventually made illegal ("for improving the art of making soap", "for a wheel of perpetual motion", "for extracting silver from lead", "for a grand American fishery") - all very reminiscent of the various shitcoins.
|
||||
* [Quantum Computing for the Very Curious](https://quantum.country/qcvc): essay to teach the fundamentals of quantum computing, with embedded spaced repetition cards.
|
||||
|
||||
[1]: https://en.wikipedia.org/wiki/Flow_%28psychology%29 "Flow Wikipedia page"
|
||||
[Worm]: https://parahumans.wordpress.com/ "Worm (fiction)"
|
||||
[Pact]: https://pactwebserial.wordpress.com/
|
||||
|
15
hugo/content/wordpress/index.html
Normal file
15
hugo/content/wordpress/index.html
Normal file
@@ -0,0 +1,15 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en-US">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>Redirecting…</title>
|
||||
<link rel="canonical" href="/index.html">
|
||||
<script>location="/index.html"</script>
|
||||
<meta http-equiv="refresh" content="0; url=/index.html">
|
||||
<meta name="robots" content="noindex">
|
||||
</head>
|
||||
<body>
|
||||
<h1>Redirecting…</h1>
|
||||
<a href="/index.html">Click here if you are not redirected.</a>
|
||||
</body>
|
||||
</html>
|
@@ -1,18 +1,26 @@
|
||||
<details>
|
||||
{{ $summary := default "Click to view gallery" (.Get "summary") }}
|
||||
<summary>{{ $summary }}</summary>
|
||||
<div class="gallery">
|
||||
{{ $base := .Get "src" }}
|
||||
{{ $path := print "static/" (.Get "src") }}
|
||||
{{ $galleryName := default "gallery" (.Get "galleryName") }}
|
||||
|
||||
{{ range (readDir $path) }}
|
||||
{{- $thumbext := "-thumb" }}
|
||||
{{- $isthumb := .Name | findRE ($thumbext | printf "%s\\.") }}<!-- is the current file a thumbnail image? -->
|
||||
{{- $isimg := lower .Name | findRE "\\.(gif|jpg|jpeg|tiff|png|bmp)" }}<!-- is the current file an image? -->
|
||||
{{- if and $isimg (not $isthumb) }}
|
||||
{{- $thumb := .Name | replaceRE "(\\.)" ($thumbext | printf "%s.") }}
|
||||
{{- $thumb := .Name | replaceRE "(\\.[^.]+)" ($thumbext | printf "%s.jpg") }}
|
||||
{{- $alttextfile := .Name | replaceRE "(\\..+)" ".md" }}
|
||||
{{- $alttext := printf "%s/%s" $path $alttextfile | os.ReadFile }}
|
||||
<a data-fancybox="gallery" href="/{{ $base }}/{{ .Name }}">
|
||||
<a data-fancybox="{{ $galleryName }}" data-caption="{{ $alttext }}" href="/{{ $base }}/{{ .Name }}">
|
||||
<img src="/{{ $base }}/{{ $thumb }}" width=100 height=100 alt={{ $.Page.RenderString $alttext }}> <br/>
|
||||
</a>
|
||||
{{- end }}
|
||||
{{ end }}
|
||||
</div>
|
||||
<script>document.addEventListener('DOMContentLoaded', function() {
|
||||
Fancybox.bind('[data-fancybox="{{ $galleryName }}"]', {});
|
||||
});</script>
|
||||
</div>
|
||||
</details>
|
1
hugo/layouts/shortcodes/rawhtml.html
Normal file
1
hugo/layouts/shortcodes/rawhtml.html
Normal file
@@ -0,0 +1 @@
|
||||
{{.Inner}}
|
@@ -314,7 +314,7 @@ a:active {
|
||||
}
|
||||
|
||||
.post .post-content a {
|
||||
text-decoration: none;
|
||||
text-decoration: underline;
|
||||
letter-spacing: 1px;
|
||||
color: #2660ab;
|
||||
}
|
||||
@@ -915,4 +915,4 @@ print {
|
||||
.page-top {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@@ -1,15 +1,12 @@
|
||||
{{- if (isset .Params "gallery") -}}
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"
|
||||
integrity="sha384-tsQFqpEReu7ZLhBV2VZlAu7zcOV+rXbYlF2cqB8txI/8aZajjp4Bqd+V6D5IgvKT"
|
||||
crossorigin="anonymous"></script>
|
||||
{{- $url := "https://cdnjs.cloudflare.com/ajax/libs/fancybox/3.4.0/jquery.fancybox.min.js" -}}
|
||||
{{- $hash := "sha384-fPPwDm9Mfp/tF9MgxjA4B33PPUlNmlXdP3oKSJgdAywm+cfAEYllAIRl/D3SBlpc" -}}
|
||||
{{- $url := "https://cdnjs.cloudflare.com/ajax/libs/fancyapps-ui/5.0.35/fancybox/fancybox.umd.js" -}}
|
||||
{{- $hash := "sha512-Jbl+VuuqPnKsz0QxpW0eFun5z3ftzXwFOGxA75I5+o7zlY5uFu+FmGp9mY3BEso+f3Ej4YtFtnUGeKwHU/YVvA==" -}}
|
||||
<script defer
|
||||
src="{{- $url -}}"
|
||||
integrity="{{- $hash -}}"
|
||||
crossorigin="anonymous"></script>
|
||||
{{- $url := "https://cdnjs.cloudflare.com/ajax/libs/fancybox/3.4.0/jquery.fancybox.min.css" -}}
|
||||
{{- $hash := "sha384-Ewh0nI9dXkThksh2nOTHmr8wAk5UlhiE2FiGSuZSnYkQ/G1W5wPuXGBnXodqD8GC" -}}
|
||||
{{- $url := "https://cdnjs.cloudflare.com/ajax/libs/fancyapps-ui/5.0.35/fancybox/fancybox.min.css" -}}
|
||||
{{- $hash := "sha512-iYdUaD/DKwJeYpOBlNLwBcGtJj/GSrFDzbPAqye/+IB/Fz6vwRekM2Wqi53sIgxCfaKp3uXpdT666ehkGP1qhw==" -}}
|
||||
<link rel="stylesheet"
|
||||
href="{{- $url -}}"
|
||||
integrity="{{- $hash -}}"
|
||||
|
@@ -19,6 +19,14 @@
|
||||
</a>
|
||||
</td>
|
||||
{{ end }}
|
||||
<td>
|
||||
<!-- RSS -->
|
||||
{{ with .OutputFormats.Get "rss" -}}
|
||||
<a href="{{ printf .Permalink }}" aria-label="RSS feed" rel="me">
|
||||
<span class="fas fa-rss fa-2x" aria-hidden="true"> </span>
|
||||
</a>
|
||||
{{ end -}}
|
||||
</td>
|
||||
</tr>
|
||||
</table>
|
||||
<div class="footer">
|
||||
|
Reference in New Issue
Block a user