In praise of link rot

Thanks to my On This Day plugin (not the plugin I was asking ChatGPT for help with in my last post), it’s really easy to “surface” (I hate that as a transitive verb) some really old posts here on my blog, which has now existed for about 24 years… frighteningly close to half of my life.

I clicked on the link to a post I wrote on this day in 2008. It was about links to articles I didn’t want to lose track of.

And it’s interesting for a couple of reasons. First, there was — was — an image at the top of the page. What it was, I have no idea. But that, like all of the other images on the blog before about a year ago, are now gone. I deleted them, because I was tired of getting harassed into paying hundreds of dollars out of the blue for random images that were lurking, unseen by anyone, on ancient blog posts no one reads.

I’ve usually been pretty good about only posting images I created, but when you’ve been at this for two decades, there are going to be a few times when you get lax. And there are a couple of news agencies who have contracted with a company that is absolutely ruthless about shaking down unsuspecting bloggers. What a racket.

But I digress… I wanted to talk about the links on that page.

They’re both dead, too, but they still go somewhere. The domains they linked to are still active, but now they redirect to 404 pages on completely different domains. Whoever owned those domains either sold them or let them lapse, and now they’re something completely different.

A lot of people want to fight “link rot.” But not me. I relish it. The Internet is an Orwellian memory hole, and fixing broken links in blog posts from 2008 just accelerates that. The pages I’m linking to have long since disappeared. But their URLs remain in a zombie state here on my blog as a testament to the fact that these things once existed.

Like the website Veer.com. I don’t even remember exactly what it was, and it had been years since I had last contemplated its existence, but once I saw that link, my immediate reaction was, “Oh yeah, I remember that site. Well… no, actually I don’t, but I remember that was a site, and I used to like checking it out semi-regularly.”

There are so many lost worlds like this. No longer present anywhere on the Internet*, but lingering in the dusty backrooms of Gen Xers’ minds.

*I lied, of course. The Wayback Machine exists. I managed to find a snapshot of that URL from June 10, 2008, about a week before I posted about it.

It even still has the image of the coffee mug I was interested in! But I’m not going to post it here. I know better. Finally.

I actually used ChatGPT to solve a coding problem today

The following originally appeared as a post on my YouTube channel.

Trigger warning: not entirely negative stuff about A.I.

I have the script written for the first half of my “5-String Bass Showdown” video (the second half will be unscripted), but I may not get around to making that video this weekend, because one half of my “day job” (indie WordPress plugin developer) is bleeding over into my video time.

I spent about 10 hours sitting in front of my computer today. The first half of that was beating my head against the wall over an extremely obscure problem I was having with Block Editor development. (I am an old school PHP developer and I have never bothered to learn React, so I keep the dev side of the Block Editor at arm’s length.)

Anyway… after literally 5 hours of almost no progress, beyond homing in on exactly where, but not what, the problem was, and getting nowhere reading 6-year-old GitHub discussion threads and even older StackOverflow threads, I finally resorted to…

Asking ChatGPT. (With DuckDuckGo’s Duck.ai as an intermediary, of course.)

Luckily I had gotten to the point where I could succinctly state in one sentence exactly where I was stuck, and provide ChatGPT with the relevant JavaScript function from my code.

In less than 5 seconds, it returned a cogent explanation of why my code was having an issue, along with a minimally rewritten version of my function that solved the problem.

It even mimicked my somewhat idiosyncratic coding style.

As Dr. Gnome would say, “Uff da.”

After finally taking a break around 3 PM to eat lunch and ponder my existential crisis, I decided to throw a few of my other “back burner” issues at it. The first one, which I thought would be the simplest, ended up being unresolvable. Two others were simple code refactoring (JavaScript is not my strongest suit), and it knocked those out of the park.

So… I’ve come to realize, A.I. like ChatGPT does have a role in my work. It can be a good partner to troubleshoot and optimize the code I write in languages I don’t know like the back of my hand.

But to that end, I think of it in a very similar way to how I’m using the stem splitter in Logic. It’s a tool to streamline the things I already do, and fill in the gaps. But ultimately I still have to be the one doing the thinking.

Thoughts on Doing Better

This is the script from a YouTube video I posted today.

These are not normal times. I can’t just do the same things I normally do, and pretend nothing is happening.

I. Freedom and Responsibility

I believe in the innate freedom of every person to choose how they want to live their lives. (And every person is a person, regardless of their race, religion, sexual orientation, gender expression, or ability.) I also believe that freedom comes with responsibility.

If we choose to live in a society — which is not really a choice in the 21st century — then we also have a responsibility to the others around us, to respect and defend their freedoms. The civil rights movement was born of the principle that no one is free, unless we all are free. Or, to put it another way: injustice anywhere is a threat to justice everywhere.

But it’s not just about individual freedoms. Working together, we can do things that none of us can do alone. Collective action allows us to create a better, more just, more meaningful world, for everyone.

II. Power and Money

The late, great senator from my home state of Minnesota, Paul Wellstone, famously said it best: “We all do better when we all do better.”

Or maybe you prefer the words of Jesus: “So in everything, do to others what you would have them do to you, for this sums up the Law and the Prophets.” (Matthew 7:12 [NIV])

Power and money are corrupting forces. Yes, complex organizations need some kind of hierarchy, to make decisions and maintain order, but everyone should still have a voice. And money is an essential tool for an economy to function, when we remember its fundamental purpose: storing the value of human effort, to facilitate trade.

But there are those among us who ignore the social contract; those who choose to hoard wealth and power. Their insatiable desire for more drives a wedge between us. Great empires of injustice are built on the backs of the masses, whose labor is essential to the construction of those empires, but who do not receive their fair share of the value they produce.

Every large-scale economic system, across the spectrum, from communist totalitarianism to laissez faire capitalism, creates an imbalance of wealth and power. It is the power of the people, the purpose of democracy, to provide a counterbalance, to ensure liberty and justice for all.

III. The Internet

The Internet exploded in the 1990s, at a unique time of international openness and realignment, following the collapse of the Soviet Union. There had never in human history been such a powerful tool for the free exchange of ideas, with anyone in the world, nor such an opportune moment for it to appear.

Fast forward to 2026, and obviously that utopian vision for the Internet has become severely corrupted by power and money… as almost all large-scale human endeavors inevitably do.

Today, misinformation and abuse are everywhere. Corporations have acquired, or squashed — often acquired in order to squash — many of the once-independent voices. Countries that were once exploring a new openness have slammed the door shut. Social media echo chambers encourage people to indulge their worst impulses. Every website seems to be drowning in ads and AI slop.

And yet, individual, self-hosted blogs still exist. Independent voices can still speak freely to the world on YouTube… even if the mysterious algorithm ultimately decides which voices rise to the surface in the vast ocean of “content.”

One word (OK, it’s a portmanteau/neologism) shows, better than any other, that the Internet can still live up to its earliest ideals: Wikipedia.

The Internet hasn’t completely lost its potential for the kind of collective action that allows us to do better — to create a more informed, more interesting, more just world — for everyone. But we have to work harder at it.

IV. Minneapolis

I live in Minneapolis. I love Minneapolis. It is a city that embraces the spirit of Paul Wellstone.

It’s not a perfect place. We all know that. But it’s a place that is trying hard to do better. And that’s made it a desirable place to be, for a lot of people.

A place that celebrates nature, with its award-winning park system. A place that celebrates the arts in their many forms, especially with its thriving theater and music scenes. A place that celebrates the diversity of its people, with a vast array of thriving restaurants and small businesses. A place that celebrates learning and innovation, with the University of Minnesota and numerous smaller colleges.

A place that is in jeopardy, in the present moment.

Minneapolis is a city of immigrants, built on stolen Dakota land.

I say that not because I believe it can somehow be “unstolen,” or as some kind of token white guilt self-flagellation. We live in the world we live in. We can’t erase the past — as much as some of us may try — but we can choose how we write the future. Choosing a future where we do better requires understanding how we got here in the first place.

People from all over the world choose to live in Minneapolis, because it is a place where they are welcome. Where they find opportunities. And where they become a part of the fabric of the community, making the city better for all of us by sharing their talents, their hard work, and the things that make their cultures unique. This is what the city has always been.

But, of course, Minneapolis is not perfect. No place is. Fort Snelling is a painful reminder of that, from its use as a concentration camp after the Dakota War of 1862, to the use of the nearby Whipple Building by ICE in 2026. (The Whipple Building is not part of Historic Fort Snelling, which is managed by the Minnesota Historical Society.)

Another painful reminder is the history of redlining, which still shapes the character of our neighborhoods more than a half century after the practice was banned. (The city has a program for homeowners to remove the now-unenforceable racial covenants from their property deeds, which my wife and I did in 2021.) And just six years ago, we experienced one of the most difficult times we’ve ever faced as a community, with the police murder of George Floyd.

But despite — maybe because of — our flawed history, and our imperfect present, we are always striving to make Minneapolis a place where “we all do better.”

The civic spirit of our city is a threat, to greed and corrupt power. So we’ve become a target. But our humble reputation for “Minnesota nice” belies the fierceness with which we will defend the things that make this place special.

That quote from Paul Wellstone, “we all do better when we all do better,” is not just an empty platitude. It’s a foundational pillar of the kind of progressive politics that has held strong here over the decades. And it’s something that those corrupted by wealth and power will never understand.

Here, we actually believe in something beyond ourselves. Our moral compass has not been skewed by the pull of avarice and cynical nihilism.

The thing is, we know we’re not especially unique in that regard. We believe most people, everywhere, feel the way we do. That everyone deserves a fair chance to live their own lives, and that they should treat their neighbors with care and respect. It’s just something that most of us go about quietly.

But. Now is the time to not be so quiet about it.

V. Do Better

These are not normal times… in Minneapolis, in the United States, in the world. I can’t just do the same things I normally do and pretend nothing is happening. I have to speak up. I have to do better.

I believe in the innate freedom of every person to choose how they want to live their lives. I also believe that freedom comes with responsibility. Individual freedom comes with the responsibility of helping to ensure and defend the individual freedom of others, and to engage meaningfully with others, to do the things we can’t do alone.

“We all do better when we all do better.”

But… what, exactly, does it mean to “do better?”

Is it a state of being? Or a call to action?

It’s both. And you can’t have one without the other.

How to get WordPress REST API self-requests working in a local development environment, i.e. Docker

I’ve only just started using Docker to run a local development environment for testing my WordPress plugin development. (Yeah, don’t get started. I’ve been working successfully for nearly 3 decades by testing my code on a remote development server.)

Using Docker is great! It’s so much faster to be able to just save my code and hit refresh to see the changes immediately, without having to sync the changes to the server… and local load times are lightning-fast!

But there’s a problem. ICS Calendar Pro‘s main functionality is to retrieve and process data from external URLs (specifically iCalendar subscription feeds), using the wp_remote_get() function. Once I added the ability to create and manage similar data locally, the obvious solution to me was to still retrieve and process that local data as if it were an external source, so ICS Calendar Pro generates ICS feeds for the internal event data, and then it makes wp_remote_get() requests to itself.

This was not working in Docker, and it took me several hours of wading through a sea of AI slop on the WordPress “how-to” websites that have spread across the web like a film of toxic algae, to finally find an answer.

Port forwarding.

I access the local Docker site at the URL http://localhost:8888 but it didn’t occur to me immediately that since Docker is running a virtualized Linux system inside a container, it’s Docker that is listening on port 8888 of my Mac system, but then Docker is routing those requests to Apache listening on port 80 inside the container.

Since the plugin’s REST API requests are coming from inside the container, Apache inside the container also needs to be listening on the port Docker is using outside the container.

This involved going into the file system of the container within Docker and editing the /etc/apache2/ports.conf file, to add the second line shown below:

Listen 80
Listen 8888

I restarted the container and boom! Everything was working.

Well… I think that’s because I had already taken another step that was probably essential for this. Since the WP REST API is using URLs with the wp-json/ path, WordPress’s URL routing needs to be activated. By default, the WordPress installation in the Docker container had the Permalink structure set to “Plain.” Changing that to anything else (on the Settings → Permalinks page) makes the REST URLs work.

Building a custom WordPress block without a React/JSX build process

I spent two solid days trying to make this work. And it’s not purely because I’m too stubborn to learn React/JSX. That’s part of it.

It’s more that I’m too stubborn to introduce an arbitrary build process into my development workflow. I have my own idiosyncratic ways of doing things, and I’m coding solo, so I generally don’t need to conform to someone else’s standard practices. As long as what I’m doing has its own internal consistency and safeguards, and produces a reliable product, who cares how I get there?

I actually have introduced a few “build” processes into my own workflow, using Git (out of necessity) for version control of my WordPress Plugin Directory plugins, my own custom versioning process for non-repository plugins, and “minifying” all of my JavaScript and CSS (using online minifiers).

But I am not doing any significant amount of block development. Certainly not enough to learn JSX and implement the arbitrary build process into my workflow, adding thousands of unnecessary dependency files to my development codebase, etc.

For the latest version of ICS Calendar Pro (version 6), currently in development, I just needed to add a simple custom block that lets users insert a calendar (a Custom Post Type) by selecting one from a dropdown list. It’s a more user-friendly option than what they’ve had to do up to this point: copying a small code snippet from the CPT’s admin page and pasting it into a Shortcode block.

Ideally my new block would render a preview of the calendar in the Block Editor, but that’s secondary to just having a way to insert the block itself and pick from the list of saved calendars.

I found the official list of block development examples, and made some progress with the “no build” example. I also found a great article from Cello Expressions (maker of the Sheet Music Library plugin), called Dynamic Blocks with Block.json, Vanilla JS, and No Build Process. I really thought that was going to get me exactly what I needed… until I realized that their particular needs did not involve having any kind of configuration options in the Block Editor itself.

Then I found DahmaniAdame/block.js on GitHub Gist. It was exactly what I needed. A straightforward, no-build example, using Vanilla JavaScript, of a simple block with a dropdown list of posts in the inspector sidebar.

The only problem: it worked perfectly if I stuck with built-in post types, but it would cause a vaguely defined block error when I switched it to using my CPT.

Eventually I realized the problem was here:

const title = post.meta.some_meta1 && post.meta.some_meta1 !== ''
		? post.meta.some_meta1
		: post.title.rendered;

For an as-yet undetermined reason, I’m not able to retrieve meta data about my CPT. (Yes, I changed some_meta1 to the actual key of my meta data field, view.) I decided I didn’t really need to retrieve that meta data, so I just removed the conditional, i.e.:

const title = post.title.rendered;

With that change (and a few others to change the post type I’m querying and remove a few other things I didn’t really need), my block was working perfectly! So much so, in fact, that I decided to try to push it to do more things. There, I ran into some trouble, because my block does a lot of JavaScript stuff when it renders, and I haven’t (yet) figured out how to get that JavaScript to run when the content renders in the Block Editor, so the HTML output of the block does get inserted into the Block Editor when the user selects from the dropdown, but the JavaScript that actually reveals it on the front-end page isn’t running. (Yes, I used enqueue_block_editor_assets to load my JavaScript and CSS files. And I also added logic to try to force my plugin’s JavaScript initialization event to trigger, but something more complicated is going on.)

Anyway… it’s still not 100% perfect, but I really wish I had found that Gist before I spent two full days banging my head against the wall over this!