29 comments

  • diablevv 1 day ago
    Nice — the use case for this is immediately obvious. I've spent more time than I care to admit running `lsof -i :PORT` or `netstat -tulpn | grep PORT` trying to figure out what's squatting on a port before running a dev server. Having a clean CLI for this with kill support is the kind of small tool that earns a permanent spot in your toolkit. Added to my dotfiles setup.
    • zephyreon 23 hours ago
      Ditto. Though I always take the lazy route and just change port numbers until I find an open one. My Mac is probably running like 20 different localhost apps at any given time.
    • raskrebs 1 day ago
      Great to hear! Please add any issues or feature request if you find anything missing :)
  • raskrebs 1 day ago
    I always have a bunch of local projects running, particularly during the weekend where I'm rarely working on one thing at a time. A big pain of mine was constantly running into port: Redis from one project blocking another, orphaned dev servers from old worktrees, Docker containers I forgot about. The usual fix is lsof -iTCP | grep ..., then figuring out what the PID actually is, then killing it. But I always forget the command, and it doesn’t really include all the information that I like.

    So I built this lightweight CLI. Single binary, no dependencies. It shows everything listening on localhost with process names, Docker container info, clickable URLs etc.

    Sure there are workarounds, but none that satisfied my need for a short, easily rememberable command. Also nothing really has the same satisfaction as running sonar kill 3000 — it just feels nice. I’ve already been approached by a few agent orchestration tools that have been struggling with the same thing. It's really useful when you have multiple agents running, but it's not built for just that use case, I have also find it handy when killing off all containers after a failed cleanup and so on. Also know that MCPs are dead and CLIs are the new thing in agentic coding, this might be a useful tool for Claude, particularly when a compose process exits before all containers are stopped.

    Open for contributions, ideas and feedback.

    • embedding-shape 1 day ago
      > I’ve already been approached by a few agent orchestration tools that have been struggling with the same thing

      Wow, this says more about the agent orchestration tool ecosystem than what you might think, that they're unable to kill child processes they themselves spawn makes it seem like they have zero clue about what they're doing.

      Probably why my impression always end up with "Wow, what a vibe-coded mess" when I look through the source of all these harnesses, they don't seem engineered at all.

      • rtpg 13 hours ago
        This is a stupidly annoying problem because it's _very easy_ to accidentally spawn children that won't get killed up in many kill situations because the distinction between processes and process groupes papered over by the fact that shells will be nice enough to kill via process group.

        But if your program is some TUI in raw mode its ctrl+c handler is often just killing itself... leaving its children along! Process groups in Unix are a stupid mess and the default being "a process can go away with the subprocess sticking around" rather than the inverse has just caused so many of these long-standing issues.

      • msdz 23 hours ago
        > that they're unable to [manage and] kill child processes they themselves spawn makes it seem like they have zero clue about what they're doing.

        Yeah, at the bare minimum these projects could also use something like portless[1] which literally maps ports to human- (and language model-)readable, named .localhost URLs. Which _should_ heavily alleviate assignment of processes to projects and vice versa, since at that point, hard-to-remember port numbers completely leave the equation. You could even imagine prefixing them if you've got that much going on for the ultimate "overview", like project1-db.localhost, project1-dev.localhost, etc.

        [1] https://port1355.dev/

        • embedding-shape 10 hours ago
          Well, or just use port 0 like we've done for decades, read what port got used, then use that. No more port collisions ever. I thought most people were already aware of that by now, but judging from that project even existing, seems I was wrong.
      • raskrebs 23 hours ago
        [dead]
    • pluc 1 day ago
      • 0cf8612b2e1e 1 day ago
        Sold on the name alone. It also has the API I never realized I needed

          puts 'usage:'
          puts 'murder 123    # kill by pid'
          puts 'murder ruby   # kill by process name'
          puts 'murder :3000  # kill by port'
      • raskrebs 1 day ago
        Will check it out
  • derodero24 18 hours ago
    Can never remember the right lsof incantation for this and end up googling it every time. The Docker container detection is a nice touch — half the time whatever's squatting on port 3000 is some container I forgot to stop two days ago.
    • spl757 17 hours ago
      ctop is great for docker containers
  • maciejj 1 day ago
    Nice! I always forget the lsof flags and end up googling them every time. Would be cool if it could run in the system tray and show what's running on your ports at a glance. Also, the name had me thinking SonarQube at first, might be worth considering a rename to avoid the confusion.
  • oweiler 7 hours ago
    https://github.com/helpermethod/pk/blob/main/pk

    I have a tiny shell function to kill whatever is listening on that port

  • mrbonner 1 day ago
    Hey, thanks for sharing this. Your app inspires me to take a look at Go, again! I've been searching for another primary language to learn. My primary used to be Java at $day$ job and now Python for ML/AI. I love Python but still feel insecure given the lack of static typing. I look at TypeScript as well, especially in the context of Bun runtime. I decided it may not be for me, not the language, but the ecosystem around it.
    • raskrebs 1 day ago
      I quite like go for how lean it is, really nice for local projects with lots of constant changes. Also easy to learn. The biggest most annoying thing is how bad error handling is done, not sure why they built it like that.

      I primarily write typescript and python for work. But have dabbled in bunch of other languages at different jobs and for different tasks. Yet to pick up rust, but have been wanting to. But tend to pick what seems most right for the task, while also considering what i most want to be working with

      • mrbonner 1 day ago
        I forgot that I also "try" Rust as well. But, I feel like it may not be suitable for my use cases and not simple enough for non-intelligent person like me :-). I agree error handling in Go could be better. But, comparing to Java I don't think I would feel salty.
  • klaushardt 1 day ago
    Would be nice to have a flag to customize the URL displayed for Docker containers. I connect to my host via Tailscale, but I can’t open links with localhost. It would be helpful to have a parameter that allows us to choose a network device or specify an IP address to display.

        3000    wud (getwud/wud:latest)                            wud          getwud/wud:latest                       3000    http://localhost:3000
        3001    dockhand (fnsys/dockhand:latest)                   dockhand     fnsys/dockhand:latest                   3000    http://localhost:3001
  • mustafa0x 1 day ago
    Nice util. It should be possible to kill inline.

    I like clack/prompts. See its multiselect API.

    https://github.com/bombshell-dev/clack/tree/main/packages/pr...

  • chwzr 1 day ago
    i have this in my .zshrc which provides same functionality:

      lk() {
        if [ $# -eq 0 ]; then
            local output=$(sudo lsof -iTCP -sTCP:LISTEN -n -P)
        elif [ $# -eq 1 ]; then
            local output=$(sudo lsof -iTCP -sTCP:LISTEN -n -P | grep -i --color=always $1)
        else
            echo "find and kill processes listening on ports. Usage: lk [pattern]"
            return 1
        fi
    
        if [ -z "$output" ]; then
            echo "No listening processes found."
            return 0
        fi
    
        # Show header + results
        echo "$(sudo lsof -iTCP -sTCP:LISTEN -n -P | head -1)"
        echo "$output"
        echo ""
    
        # Extract unique PIDs (skip the header row if no grep was applied)
        local pids=($(echo "$output" | awk '{print $2}' | grep -E '^[0-9]+$' | sort -u))
    
        if [ ${#pids[@]} -eq 0 ]; then
            echo "No PIDs found."
            return 0
        fi
    
        echo "PIDs to kill: ${pids[*]}"
        echo -n "Kill these ${#pids[@]} process(es)? [y/N] "
        read -r confirm
    
        if [[ "$confirm" =~ ^[Yy]$ ]]; then
            for pid in "${pids[@]}"; do
                echo "Killing PID $pid..."
                sudo kill -9 $pid
            done
            echo "Done."
        else
            echo "Aborted."
        fi
      }
    • raskrebs 1 day ago
      I have added quite a lot of functionality beyond listing and killing ports. Please check out the readme, it may convince you to try it out.
      • fionic 1 day ago
        Its funny bc the title suggests a tool for listing and killing
        • tomcatfish 1 day ago
          HN is a place where people can be expected to go beyond the title (though I like the limited script and am glad it was posted). Misleading titles are not uncommonly flagged and changed, even.
        • raskrebs 1 day ago
          True, it was what it started as, but grew as i found my self missing features. Got a few users and now i don't want to update the name. Also easy and quick to write in the terminal
          • bartek_gdn 1 day ago
            Why not grep the output to abother tool?
  • fcoury 1 day ago
    We live in crazy times. I wanted to add a PID to the list for my personal use and since I use Rust way more than Go, I decided to one-shot one app, and Codex indeed one shotted it, wow.

    https://github.com/fcoury/sonars

    • raskrebs 1 day ago
      You can add PID with the c-flag, e.g.

      sonar list -c port,process,pid,type,url,container

      or just show all columns:

      sonar list --all-columns

      But yeah, it's quite cool. I believe the future lies in software distillation, so cool to see it happen on my own project :D

      • fcoury 1 day ago
        I did this more as an experiment but man it sucks, doesn't it?
        • raskrebs 1 day ago
          What sucks? Software distillation? :)
          • fcoury 20 hours ago
            Yes, instead of everybody contributing to one central codebase, we now have the same concept being rewritten a thousand times :-)
            • raskrebs 14 hours ago
              Ahhh yes sure, but not too worried about that i think. Its not so difficult to create, but to innovate and maintain requires participation. It can be the bulk of the job. But most def its sad if people leave their good ideas in a fork of their own. But also great that you can reproduce a good idea thats been conflated with wayy too much crap, because a product lost site of ehat their value proposition was
    • zenoprax 1 day ago
      I'm tempted to one-shot this into a series of FISH abbreviations.

      And I would want someone to use that to one-shot a python implementation. And on and on like a game of telephone until the context degrades so far that it becomes an entirely different program.

  • clutchski 1 day ago
    • Brainspackle 1 day ago
      you picked a unique name. There is already a massively popular product called Sonar
    • raskrebs 1 day ago
      Any features you are missing or ideas for the use case?
  • pdimitar 1 day ago
    I am absolutely installing this and starting to use it daily!

    For the even less patient there's also this (not mine): https://github.com/jkfran/killport

    • raskrebs 1 day ago
      Glad to hear! Please join the party and post any issues you have.
  • mfkrause 1 day ago
    I always find myself going through my zsh history for `lsof`. Will definitely check this out, seems interesting (even though I'm generally reluctant of installing third-party tools for such jobs).
    • raskrebs 1 day ago
      I get that, i also often install some and forgot about them. But i felt that there was a big gap in managing multiple services running on localhost. It's pretty lightweight if that helps
  • RonanSoleste 1 day ago
    I just have an alias in my .bashrc :)
    • raskrebs 1 day ago
      Try and have a look at the readme, this adds a bit more functionality. Maybe some if it you'll find useful :)

      Started with the same, but found my self wanting a bit more, so just built it

      • RonanSoleste 1 day ago
        Can it do things that existing tools build into my distro cannot? It looks to me like a convenience tool. Which is fine, but i see no need for it myself
        • raskrebs 1 day ago
          Probably not, that's what makes it so lean (it doesn't really pull in anything).

          Although i have a feature branch with a tray app (for macos) that let's you monitor and track any process (will send notifications when one goes up or down). But it's just gimmicks i felt i needed to make life a bit easier when working with compose and across worktrees

          • Barbing 1 day ago
            That monitor sounds interesting; I manually set something up in Keyboard Maestro to notify me that something quit and potentially re-launch it.
  • quotemstr 1 day ago
    Christ Almighty I hate our industry practice of binding to some inscrutable port number on localhost. Unix domain sockets aren't that hard! They're secure against all sorts of attacks and more convenient to boot. Instead of connecting to a number, you connect to a file. An ordinary file, with an ordinary name you can mv, chmod, and rm. Boring on a good way.

    So why doesn't everyone run local services over Unix sockets?

    The only problems: 1) web browsers don't support AF_UNIX URI scheme, and 2) ancient versions of Java don't have built-in APIs for AF_UNIX sockets.

    That's it. For these trivial reasons, we've beat our head against arbitrary opaque numbers for decades.

    And so, for want of a nail, the Unix was lost.

    • 0x457 1 day ago
      Some random daemon binding to 3000 because it's the express default drives me nuts. I either do a Unix socket, a pick any random port if it has to bind on a port.
    • formerly_proven 1 day ago
      > So why doesn't everyone run local services over Unix sockets?

      > The only problems:

      3) 40 years of Windows not supporting UDS.

      • quotemstr 1 day ago
        Yeah, that too. Windows supports them nowadays too, just to be clear. I think we're still bottlenecked, right now, on #1 and #2 in the form of Java 8 refusing to die.
        • formerly_proven 1 day ago
          Yeah, doing the math it's actually only 33 years of not supporting AF_UNIX, but that's not really right either, since those versions of Windows didn't support any sockets. I guess the technically correct answer then is that Windows didn't support UDS for 26 years.

          Which is still enough for most portable software to go "eh, localhost is fine*"

          * resolving localhost is actually a pretty bad idea (yet very common) and it's way more robust to listen directly on a numeric address.

  • koinedad 22 hours ago
    I run lsof a good amount. This is a great idea.
  • moezd 1 day ago
    Sonar as in SonarQube? That's an interesting choice for a name :)
    • raskrebs 1 day ago
      I think a cli tool that detects objects beneath the surface is a pretty intuitive name, but i was also reluctant in the beginning. But they are pretty keen on always using the qube part, i believe theirs is sonar-qube.
    • beart 1 day ago
      How about Sonar as in SOund Navigation And Ranging?
  • cyanydeez 11 hours ago
    Should git relos with CLI tools have opencode skills directory?
  • kohexo 1 day ago
    Honestly, pretty cool. I was wondering if something like this existed. Right now I have scripts to kill the ports I use consistently to avoid issues when developing. Kudos!
    • raskrebs 1 day ago
      I looked around for a while, couldn't find anything. Someone posted about killport but never stumbled upon it. Has some features that are the same, but not all. From the reaction online people don't seem to know of other solutions like this, or have something they have made them selves in the .zshrc :)
      • kohexo 18 hours ago
        Yeah, I had to create the script myself and used an extension so I could click a button in VS Action bar. Once I outgrow my current setup, this is probably what I'm migrating to.
  • Bradd3rs 1 day ago
    love this, i get tired of spamming lsof -i tcp:xxxx
    • Doublon 1 day ago
      The README made me realize I just needed a simple `alias local-tcp-listeners='lsof -iTCP -sTCP:LISTEN'` in my `~/.bash_aliases` :)
      • deadbabe 1 day ago
        Same, not sure why a whole cli app is needed.
        • raskrebs 1 day ago
          Developers are nitpicky, atleast i am and i know a lot of others that are as well. So don't underestimate the value of a nice tool with good developer experience, one that's intuitive, clean and easy to use means a lot when juggling so many things during a workday. So having a clean and light implementation to make job even easier is in my opinion worth it (and thus needed) :)
        • paddim8 1 day ago
          Because it gives more context. Quite obvious if you look at the readme...
      • raskrebs 1 day ago
        True, but as i write their are workarounds, the problem is that they are unintuitive, difficult to remember and don't provide all that much usability beyond listing. So these lack useful features like getting process stats, killing ports easily without having to remember the the pid after lsof and so on. I often have to kill multiple process at once after a failed cleanup. If you are into agentic coding, then having your agent create a profile for all the processes it stats, which it can easily kill of when finished is a lot easier for me atleast.

        Some features on the way are: next available port; wait (wait for a host to return a successful health check before proceeding - good for migrations etc.). And lots more. It's not just about listing running ports, but a tool for managing them.

        But to each their own, that's what's lovely about the many options available. But if you have anything in relation to this you think is neat, feel free to open an issue. It may be able to convince you that a simple alias won't suffice.

    • raskrebs 1 day ago
      Glad to hear! Have quite a few ideas in mind so keep an eye out for some updates (one of the ideas is an easy update command). There's a couple of open enhancement ideas as well. Feel free to add any or contribute.
  • frankdenbow 1 day ago
    love this, happens too often
    • raskrebs 1 day ago
      Exactly, really hates it as well. Please post any issues you may have
  • jkestner 1 day ago
    I read the readme. :) Very nice. Thoughtful features. Get this on Homebrew!
    • raskrebs 1 day ago
      I will, but the process is a bit tedious. But will look into it over the weekend
      • 0x457 1 day ago
        I recommend just doing a tap in the same repo rather than adding this to homebrew first.
  • Servant-of-Inos 1 day ago
    [dead]
  • rrojas-nexus 1 day ago
    [dead]
  • abitabovebytes 1 day ago
    [dead]
  • aplomb1026 1 day ago
    [dead]
  • takahitoyoneda 1 day ago
    [dead]
  • olivercoleai 1 day ago
    [dead]