<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Christopher Hicks</title>
    <description>The latest articles on DEV Community by Christopher Hicks (@chicks).</description>
    <link>https://dev.to/chicks</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3347194%2F1e374bb0-cd3a-4ed1-906f-c837076e7b50.jpg</url>
      <title>DEV Community: Christopher Hicks</title>
      <link>https://dev.to/chicks</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/chicks"/>
    <language>en</language>
    <item>
      <title>How Rust Had to Save Python From Itself: The uv Revolution</title>
      <dc:creator>Christopher Hicks</dc:creator>
      <pubDate>Thu, 21 May 2026 16:31:51 +0000</pubDate>
      <link>https://dev.to/chicks/how-rust-had-to-save-python-from-itself-the-uv-revolution-3fcb</link>
      <guid>https://dev.to/chicks/how-rust-had-to-save-python-from-itself-the-uv-revolution-3fcb</guid>
      <description>&lt;p&gt;Well, well, well. Here we are in 2025 and Python packaging has finally been fixed. Not by the Python community, mind you - they had their shot for about 20 years. No, it took the Rust folks to come in and show us how it's done.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Long, Painful History
&lt;/h2&gt;

&lt;p&gt;Let me paint you a picture. Back when I was starting out, we had &lt;code&gt;distutils&lt;/code&gt;. That was it. Then came &lt;code&gt;setuptools&lt;/code&gt;, which was supposed to fix everything. Then &lt;code&gt;pip&lt;/code&gt; showed up to handle installation. Then &lt;code&gt;virtualenv&lt;/code&gt; because global package installs were a nightmare. Then &lt;code&gt;pipenv&lt;/code&gt; to combine pip and virtualenv. Then &lt;code&gt;poetry&lt;/code&gt; because pipenv wasn't quite right. Then &lt;code&gt;pip-tools&lt;/code&gt; for deterministic builds. Then &lt;code&gt;conda&lt;/code&gt; for scientific computing. Then...&lt;/p&gt;

&lt;p&gt;You get the idea. Each tool solved one specific problem while creating three new ones. The Python community spent decades bikeshedding over package formats, dependency resolution algorithms, and lock file formats while the rest of the world just wanted to install packages without breaking their system. Some tools even managed to break the ancient Unix standard of shebang lines - you know, that &lt;code&gt;#!/usr/bin/env python&lt;/code&gt; thing that's been working since the dawn of time.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Delusion
&lt;/h2&gt;

&lt;p&gt;Here's the thing that drives me nuts: the Python community kept insisting they could solve this from within. "We just need better standards!" they'd cry. "PEP 517 will fix everything!" Narrator: it didn't. "PEP 621 is the answer!" Still waiting on that revolution.&lt;/p&gt;

&lt;p&gt;Meanwhile, every other modern language figured this out ages ago. &lt;a href="https://doc.rust-lang.org/cargo/" rel="noopener noreferrer"&gt;Cargo&lt;/a&gt; for &lt;a href="https://www.rust-lang.org/" rel="noopener noreferrer"&gt;Rust&lt;/a&gt;. &lt;a href="https://go.dev/ref/mod" rel="noopener noreferrer"&gt;Go modules&lt;/a&gt;. &lt;a href="https://www.npmjs.com/" rel="noopener noreferrer"&gt;npm&lt;/a&gt; (despite its flaws) for &lt;a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript" rel="noopener noreferrer"&gt;JavaScript&lt;/a&gt;. Even Perl got &lt;a href="https://www.cpan.org/" rel="noopener noreferrer"&gt;CPAN&lt;/a&gt; right decades ago! But Python? Python kept spawning new tools that partially solved the problem while maintaining backward compatibility with decades of technical debt.&lt;/p&gt;

&lt;h2&gt;
  
  
  Enter uv: Rust to the Rescue
&lt;/h2&gt;

&lt;p&gt;Then along comes &lt;a href="https://astral.sh/" rel="noopener noreferrer"&gt;Astral&lt;/a&gt; with &lt;a href="https://docs.astral.sh/uv/" rel="noopener noreferrer"&gt;&lt;code&gt;uv&lt;/code&gt;&lt;/a&gt;, written in &lt;a href="https://www.rust-lang.org/" rel="noopener noreferrer"&gt;Rust&lt;/a&gt;, and suddenly everything just... works. It's fast. Like, stupidly fast. It handles virtual environments transparently. It resolves dependencies without making you wait for your coffee to brew. It installs packages so quickly you wonder if it actually did anything.&lt;/p&gt;

&lt;p&gt;The best part? They didn't try to be compatible with every broken decision the Python packaging ecosystem made over the past two decades. They looked at what actually needed to be solved and built a tool that solves those problems.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Irony
&lt;/h2&gt;

&lt;p&gt;The delicious irony here is that it took a systems programming language to fix a scripting language's package management. Rust's obsession with memory safety and performance optimization turns out to be exactly what Python's packaging ecosystem needed. Who could have predicted that?&lt;/p&gt;

&lt;p&gt;It's like watching someone struggle to open a jar for 20 years, refusing help from anyone, only to have their neighbor walk over with the right tool and pop it open in two seconds.&lt;/p&gt;

&lt;h2&gt;
  
  
  What This Means
&lt;/h2&gt;

&lt;p&gt;For those of us who've been dealing with Python packaging hell since the &lt;code&gt;easy_install&lt;/code&gt; days, &lt;code&gt;uv&lt;/code&gt; feels like a miracle. One tool that does everything, does it fast, and doesn't break when you look at it sideways. Release engineers everywhere can finally sleep soundly knowing their builds won't randomly break because of dependency resolution conflicts or virtual environment corruption.&lt;/p&gt;

&lt;p&gt;The Python community should probably feel a little embarrassed about this. Twenty years of "we can fix this ourselves" only to be shown up by a tool written in a different language entirely. But hey, at least we finally have something that works.&lt;/p&gt;

&lt;p&gt;So here's to the Rust community for doing what the Python community couldn't: fixing Python packaging. Sometimes you need an outsider's perspective to see the forest for the trees. Or in this case, to see the solution through all the PEPs.&lt;/p&gt;

&lt;p&gt;Now if you'll excuse me, I need to go update all my projects to use &lt;code&gt;uv&lt;/code&gt;. It only takes about 30 seconds per project, which is another miracle in itself. The best part? My colleagues independently discovered &lt;code&gt;uv&lt;/code&gt; around the same time I did - that's how you know a tool is genuinely solving real problems when multiple people stumble across it organically.&lt;/p&gt;

</description>
      <category>python</category>
      <category>programming</category>
      <category>uv</category>
      <category>rust</category>
    </item>
    <item>
      <title>Announcing gh-observer: Waiting for CI Should Not Be Misery</title>
      <dc:creator>Christopher Hicks</dc:creator>
      <pubDate>Fri, 15 May 2026 07:00:00 +0000</pubDate>
      <link>https://dev.to/chicks/announcing-gh-observer-waiting-for-ci-should-not-be-misery-4me0</link>
      <guid>https://dev.to/chicks/announcing-gh-observer-waiting-for-ci-should-not-be-misery-4me0</guid>
      <description>&lt;p&gt;Look, I know what you're thinking: "Another dev scratched their own itch and now they want to tell me about it." Guilty. But hear me out, because this particular itch has probably been annoying you too, and the scratch is genuinely useful.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Problem Nobody Talks About
&lt;/h2&gt;

&lt;p&gt;Here's the scenario that broke me: I push a PR, immediately run &lt;code&gt;gh pr checks --watch&lt;/code&gt;, and... it bombs out with an error because GitHub Actions hasn't queued anything yet. So I wait. I run it again. Maybe it works, maybe it doesn't. And when it finally does start showing me checks, I'm staring at a list of job names with no idea whether that &lt;code&gt;3m 52s&lt;/code&gt; I've been waiting is normal or a sign that something's silently wedged.&lt;/p&gt;

&lt;p&gt;The standard &lt;code&gt;gh pr checks --watch&lt;/code&gt; has had some real gaps for a while now:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;It doesn't handle startup delay.&lt;/strong&gt; GitHub Actions typically takes 30-900 seconds to queue jobs after a PR is created or pushed to. The built-in watcher just... gives up during that window.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No queue latency visibility.&lt;/strong&gt; You can see a job is "in progress," but you have no idea if it's been sitting in a queue for 2 seconds or 45 seconds before it started.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;No runtime metrics.&lt;/strong&gt; Is that job that's been "running" for a while actually running, or has GitHub just not updated the status yet? Who knows!&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;So naturally, I did what any reasonable developer does when something annoys them enough: I spent way more time building a solution than I would have lost just dealing with the annoyance. Classic.&lt;/p&gt;

&lt;h2&gt;
  
  
  Introducing gh-observer
&lt;/h2&gt;

&lt;p&gt;&lt;code&gt;gh-observer&lt;/code&gt; is a GitHub CLI extension that replaces &lt;code&gt;gh pr checks --watch&lt;/code&gt; with something that actually tells you what's going on. It's a full TUI (terminal UI) that polls GitHub's API every 5 seconds and shows you the information you actually care about.&lt;/p&gt;

&lt;p&gt;Here's what a typical run looks like:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;PR #5: 🔶 [claude] /init 21:04:15 UTC
Updated 0s ago  •  Pushed 43h 8m 11s ago

Startup   Workflow/Job                                Duration

  15s ✗ MarkdownLint / lint                             5s
   .github:13 - Failed with exit code: 1

  15s ✓ Auto Assign / run                               5s
  15s ✓ CUE Validation / verify                         6s
  15s ✓ Checkov / scan                                 27s
  15s ✓ Claude Code Review / claude-review          3m 52s
  15s ✓ Lint GitHub Actions workflows / actionlint      8s
  39s ✓ Checkov                                         2s

Press q to quit
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That &lt;code&gt;15s&lt;/code&gt; in the Startup column? That's how long GitHub sat on the job before actually starting it. The &lt;code&gt;3m 52s&lt;/code&gt; at the end? That's the total runtime. Now you know the Claude review is just slow, not broken.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Startup Phase Thing
&lt;/h2&gt;

&lt;p&gt;This is the part I'm most proud of, honestly. Instead of bombing out when there are no checks yet, &lt;code&gt;gh-observer&lt;/code&gt; shows you a helpful waiting message:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;PR #123: Add new feature

Startup Phase (37s elapsed):
  ⏳ Waiting for Actions to start...
  💡 GitHub typically takes 30-90s to queue jobs after PR creation
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;It tracks how long you've been waiting, reminds you that this is normal, and just... keeps watching. No manual intervention required. No re-running the command. It transitions smoothly into showing actual check status once jobs start appearing.&lt;/p&gt;

&lt;h2&gt;
  
  
  Features Worth Knowing About
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;Queue latency and runtime metrics&lt;/strong&gt; are the headline features, but there's more:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Workflow/Job naming&lt;/strong&gt; — Instead of just the job name, you see "MarkdownLint
/ lint" so you know which workflow the job belongs to. Uses GitHub's GraphQL
API to pull this efficiently in a single query.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Error log integration&lt;/strong&gt; — Failed checks show the first line of their error
output right there in the terminal. No more clicking through to GitHub to
find out &lt;em&gt;why&lt;/em&gt; something failed.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Rate limit awareness&lt;/strong&gt; — When you're getting close to GitHub API limits, it
automatically backs off and polls less frequently. Keeps your rate limit
healthy without any manual configuration.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;CI-friendly snapshot mode&lt;/strong&gt; — When stdout isn't a TTY (like in a script or
pipeline), it prints a plain text snapshot and exits with an appropriate exit
code. So &lt;code&gt;gh-observer &amp;amp;&amp;amp; deploy.sh&lt;/code&gt; actually works.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Configurable colors&lt;/strong&gt; — ANSI 256-color support via
&lt;code&gt;~/.config/gh-observer/config.yaml&lt;/code&gt; if you're particular about your terminal
aesthetics.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Installation
&lt;/h2&gt;

&lt;p&gt;The easiest path is the precompiled binary via GitHub CLI extensions — no Go&lt;br&gt;
toolchain required:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;gh extension &lt;span class="nb"&gt;install &lt;/span&gt;fini-net/gh-observer
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Precompiled binaries exist for macOS (Intel and Apple Silicon), Linux (x86-64&lt;br&gt;
and ARM64), and Windows (x86-64). All binaries include build attestations for&lt;br&gt;
supply chain security verification.&lt;/p&gt;
&lt;h2&gt;
  
  
  How to Use It
&lt;/h2&gt;

&lt;p&gt;Auto-detect your current branch's PR:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;gh observer
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Watch a specific PR number:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;gh observer 123
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Under the Hood (for the curious)
&lt;/h2&gt;

&lt;p&gt;It's built in Go using &lt;a href="https://github.com/charmbracelet/bubbletea" rel="noopener noreferrer"&gt;Bubbletea&lt;/a&gt; for the TUI, which follows the Elm Architecture pattern — if you've done any Elm or Redux, the Model/Update/View pattern will feel familiar. &lt;a href="https://github.com/charmbracelet/lipgloss" rel="noopener noreferrer"&gt;Lipgloss&lt;/a&gt; handles the terminal styling.&lt;/p&gt;

&lt;p&gt;The interesting bit technically is that it uses GitHub's GraphQL API to pull check run data — same approach as &lt;code&gt;gh pr checks&lt;/code&gt;, but in a single query that returns both the workflow name and the job status together. This is why it can show "MarkdownLint / lint" instead of just "lint": it's joining the workflow and job name in one efficient API call.&lt;/p&gt;

&lt;p&gt;Queue latency is calculated as the delta between when you pushed the commit and when the check actually started. Runtime is &lt;code&gt;time.Now() - check.StartedAt&lt;/code&gt; for in-progress checks. Simple math, but surprisingly useful information.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Code
&lt;/h2&gt;

&lt;p&gt;Everything's at &lt;a href="https://github.com/fini-net/gh-observer" rel="noopener noreferrer"&gt;fini-net/gh-observer&lt;/a&gt;. It's open source, and I'm genuinely interested in feedback and contributions. If you hit a weird edge case or have a feature idea, open an issue.&lt;/p&gt;

&lt;p&gt;And yes, before you ask — &lt;code&gt;gh observer&lt;/code&gt; was partially built using &lt;code&gt;gh observer&lt;/code&gt; to watch its own CI. It's turtles all the way down. Or the dog food tastes great. Pick your own adventure.&lt;/p&gt;




&lt;p&gt;&lt;em&gt;If you find it useful, a star on the repo goes a long way. And if you find a&lt;br&gt;
bug, please do tell me so we can fix it rather than quietly suffering.&lt;/em&gt;&lt;/p&gt;

</description>
      <category>github</category>
      <category>cli</category>
      <category>tui</category>
      <category>cicd</category>
    </item>
  </channel>
</rss>
