Context: Docker bypasses all UFW firewall rules

10 months ago by qaz to c/programmer_humor

Docker docs:

Docker routes container traffic in the nat table, which means that packets are diverted before it reaches the INPUT and OUTPUT chains that ufw uses. Packets are routed before the firewall rules can be applied, effectively ignoring your firewall configuration.

veganpizza69 123 points 10 months ago

path: 0 19114385, hotness: undefined, score: 123, children: 0
Static_Rocket 109 points 10 months ago

This was a large part of the reason I switched to rootless podman for everything

path: 0 19105390, hotness: undefined, score: 109, children: 16
False 21 points 10 months ago

Explicitly binding certain ports to the container has a similar effect, no?

path: 0 19105390 19105549, hotness: undefined, score: 21, children: 6
doughless 10 points 10 months ago

I still need to allow the ports in my firewall when using podman, even when I bind to 0.0.0.0.

path: 0 19105390 19105549 19110910, hotness: undefined, score: 10, children: 4
qaz 3 points 10 months ago

Also when using a rootfull Podman socket?

path: 0 19105390 19105549 19110910 19113922, hotness: undefined, score: 3, children: 3
doughless 5 points 10 months ago

I haven't tried rootful since I haven't had issues with rootless. I'll have to check on that and get back to you.

path: 0 19105390 19105549 19110910 19113922 19114038, hotness: undefined, score: 5, children: 0
doughless 3 points 10 months ago

When running as root, I did not need to add the firewall rule.

path: 0 19105390 19105549 19110910 19113922 19119457, hotness: undefined, score: 3, children: 1
Static_Rocket 4 points 10 months ago

It's better than nothing but I hate the additional logs that came from it constantly fighting firewalld.

path: 0 19105390 19105549 19109007, hotness: undefined, score: 4, children: 0
Kr4u7 10 points 10 months ago

My problem with podman is the incompatibility with portainer :(

Any recommendations?

path: 0 19105390 19106255, hotness: undefined, score: 10, children: 7
giacomo 15 points 10 months ago

cockpit has a podman/container extension you might like.

path: 0 19105390 19106255 19107210, hotness: undefined, score: 15, children: 2
qaz 8 points 10 months ago

It's okay for simple things, but too simple for anything beyond that, IMO. One important issue is that unlike with Portainer you can't edit the container in any way without deleting it and configuring it again, which is quite annoying if you just want to change 1 environment variable (GH Issue). Perhaps they will add a quadlet config tool to cockpit sometime in the future.

path: 0 19105390 19106255 19107210 19108541, hotness: undefined, score: 8, children: 1
giacomo 6 points 10 months ago

i mean, you can just redeploy the container with the updated variable. thats kinda how they work.

path: 0 19105390 19106255 19107210 19108541 19112071, hotness: undefined, score: 6, children: 0
slate 13 points 10 months ago

CLI and Quadlet? /s but seriously, that's what I use lol

path: 0 19105390 19106255 19106919, hotness: undefined, score: 13, children: 1
chunkystyles 5 points 10 months ago

Quadlets are so nice.

path: 0 19105390 19106255 19106919 19113191, hotness: undefined, score: 5, children: 0
Guilvareux 3 points 10 months ago

I assume portainer communicates via the docker socket? If so, couldn’t you just point portainer to the podman socket?

path: 0 19105390 19106255 19107394, hotness: undefined, score: 3, children: 1
qaz 3 points 10 months ago path: 0 19105390 19106255 19107394 19113936, hotness: undefined, score: 3, children: 0
pizza_the_hutt 7 points 10 months ago

This is the way.

path: 0 19105390 19105564, hotness: undefined, score: 7, children: 0
dangling_cat 38 points 10 months ago

I DIDNT KNOW THAT! WOW, this puts “not to use network_mode: host” another level.

path: 0 19105906, hotness: undefined, score: 38, children: 3
exu 17 points 10 months ago

network: host gives the container basically full access to any port it wants. But even with other network modes you need to be careful, as any -p <external port>:<container port> creates the appropriate firewall rule automatically.

path: 0 19105906 19106189, hotness: undefined, score: 17, children: 1
AmbiguousProps 3 points 10 months ago

I just use caddy and don't use any port rules on my containers. But maybe that's also problematic.

path: 0 19105906 19106189 19110689, hotness: undefined, score: 3, children: 0
greyfox 3 points 10 months ago

Actually I believe host networking would be the one case where this isn't an issue. Docker isn't adding iptables rules to do NAT masquerading because there is no IP forwarding being done.

When you tell docker to expose a port you can tell it to bind to loopback and this isn't an issue.

path: 0 19105906 19114617, hotness: undefined, score: 3, children: 0
dohpaz42 34 points 10 months ago

It’s my understanding that docker uses a lot of fuckery and hackery to do what they do. And IME they don’t seem to care if it breaks things.

path: 0 19105627, hotness: undefined, score: 34, children: 3
marcos 33 points 10 months ago

To be fair, the largest problem here is that it presents itself as the kind of isolation that would respect firewall rules, not that they don't respect them.

People wouldn't make the same mistake in NixOS, despite it doing exactly the same.

path: 0 19105627 19105783, hotness: undefined, score: 33, children: 0
LodeMike 4 points 10 months ago

This is why I hate Docker.

path: 0 19105627 19105778, hotness: undefined, score: 4, children: 0
Guilvareux 3 points 10 months ago

I don’t know how much hackery and fuckery there is with docker specifically. The majority of what docker does was already present in the Linux kernel namespaces, cgroups etc. Docker just made it easier to build and ship the isolated environments between systems.

path: 0 19105627 19107427, hotness: undefined, score: 3, children: 0
grrgyle 28 points 10 months ago

If I had a nickel for every database I've lost because I let docker broadcast its port on 0.0.0.0 I'd have about 35¢

path: 0 19107112, hotness: undefined, score: 28, children: 9
MangoPenguin 9 points 10 months ago

How though? A database in Docker generally doesn't need any exposed ports, which means no ports open in UFW either.

path: 0 19107112 19109275, hotness: undefined, score: 9, children: 8
grrgyle 3 points 10 months ago

I exposed them because I used the container for local development too. I just kept reseeding every time it got hacked before I figured I should actually look into security.

path: 0 19107112 19109275 19109330, hotness: undefined, score: 3, children: 7
MangoPenguin 18 points 10 months ago

For local access you can use 127.0.0.1:80:80 and it won't put a hole in your firewall.

Or if your database is access by another docker container, just put them on the same docker network and access via container name, and you don't need any port mapping at all.

path: 0 19107112 19109275 19109330 19109412, hotness: undefined, score: 18, children: 1
grrgyle 1 point 10 months ago

Yeah, I know that now lol, but good idea to spell it out. So what Docker does, which is so confusing when you first discover the behaviour, is it will bind your ports automatically to 0.0.0.0 if all you specify is 27017:27017 as you port (without an IP address prefixing). AKA what the meme is about.

path: 0 19107112 19109275 19109330 19109412 19138155, hotness: undefined, score: 1, children: 0
firelizzard 2 points 10 months ago

Where are you working that your local machine is regularly exposed to malicious traffic?

path: 0 19107112 19109275 19109330 19136567, hotness: undefined, score: 2, children: 4
grrgyle 1 point 10 months ago

My use case was run a mongodb container on my local, while I run my FE+BE with fast live-reloading outside of a container. Then package it all up in services for docker compose on the remote.

path: 0 19107112 19109275 19109330 19136567 19137895, hotness: undefined, score: 1, children: 3
salacious_coaster 22 points 10 months ago

My impression from a recent crash course on Docker is that it got popular because it allows script kiddies to spin up services very fast without knowing how they work.

OWASP was like "you can follow these thirty steps to make Docker secure, or just run Podman instead." https://cheatsheetseries.owasp.org/...

path: 0 19108173, hotness: undefined, score: 22, children: 5
domi 49 points 10 months ago

My impression from a recent crash course on Docker is that it got popular because it allows script kiddies to spin up services very fast without knowing how they work.

That's only a side effect. It mainly got popular because it is very easy for developers to ship a single image that just works instead of packaging for various different operating systems with users reporting issues that cannot be reproduced.

path: 0 19108173 19109387, hotness: undefined, score: 49, children: 0
null_dot 32 points 10 months ago

I dont really understand the problem with that?

Everyone is a script kiddy outside of their specific domain.

I may know loads about python but nothing about database management or proxies or Linux. If docker can abstract a lot of the complexities away and present a unified way you configure and manage them, where's the bad?

path: 0 19108173 19110697, hotness: undefined, score: 32, children: 0
LordKitsuna 15 points 10 months ago

That is definitely one of the crowds but there are also people like me that just are sick and tired of dealing with python, node, ruby depends. The install process for services has only continued to become increasingly more convoluted over the years. And then you show me an option where I can literally just slap down a compose.yml and hit "docker compose up - d" and be done? Fuck yeah I'm using that

path: 0 19108173 19111593, hotness: undefined, score: 15, children: 0
MangoPenguin 11 points 10 months ago

No it's popular because it allows people/companies to run things without needing to deal with updates and dependencies manually

path: 0 19108173 19109251, hotness: undefined, score: 11, children: 0
Appoxo 1 point 10 months ago

Another take: Why should I care about dependency hell if I can just spin up the same service on the same machine without needing an additional VM and with minimal configuration changes.

path: 0 19108173 19119998, hotness: undefined, score: 1, children: 0
steventhedev 18 points 10 months ago

You're forgetting the part where they had an option to disable this fuckery, and then proceeded to move it twice - exposing containers to everyone by default.

I had to clean up compromised services twice because of it.

path: 0 19106178, hotness: undefined, score: 18, children: 0
MangoPenguin 17 points 10 months ago

This only happens if you essentially tell docker "I want this app to listen on 0.0.0.0:80"

If you don't do that, then it doesn't punch a hole through UFW either.

path: 0 19109292, hotness: undefined, score: 17, children: 0
MnemonicBump 16 points 10 months ago

This is why I install on bare metal, baby!

path: 0 19106072, hotness: undefined, score: 16, children: 0
ohshit604 15 points 10 months ago

This post inspired me to try podman, after it pulled all the images it needed my Proxmox VM died, VM won’t boot cause disk is now full. It’s currently 10pm, tonight’s going to suck.

path: 0 19114060, hotness: undefined, score: 15, children: 6
sip 3 points 10 months ago

eh, booting into single user mod should work fine, uninstall podman and init 5

path: 0 19114060 19114097, hotness: undefined, score: 3, children: 5
ohshit604 2 points 10 months ago

Okay so I’ve done some digging and got my VM to boot up! This is not Podman’s fault, I got lazy setting up Proxmox and never really learned LVM volume storage, while internally on the VM it shows 90Gb used of 325Gb Proxmox is claiming 377Gb is used on the LVM-Thin partition.

I’m backing up my files as we speak, thinking of purging it all and starting over.

Edit: before I do the sacrificial purge This seems promising.

path: 0 19114060 19114097 19114353, hotness: undefined, score: 2, children: 4
SpaceCadet 5 points 10 months ago

thinking of purging it all and starting over.

Don't do that. You'll learn nothing.

path: 0 19114060 19114097 19114353 19114451, hotness: undefined, score: 5, children: 3
ohshit604 4 points 10 months ago

So I happened to follow the advice from that Proxmox post, enabled the “Discard” option for the disk and ran sudo fstrim / within the VM, now the Proxmox LVM-Thin partition is sitting at a comfortable 135Gb out of 377Gb.

Think I’m going to use this fstrim command on my main desktop to free up space.

path: 0 19114060 19114097 19114353 19114451 19114581, hotness: undefined, score: 4, children: 2
MasterNerd 12 points 10 months ago

I mean if you're hosting anything publicly, you really should have a dedicated firewall

path: 0 19107951, hotness: undefined, score: 12, children: 6
qaz 1 point 10 months ago

Do you mean a hardware firewall?

path: 0 19107951 19113891, hotness: undefined, score: 1, children: 4
MasterNerd 2 points 10 months ago

Basically yeah, though I didn't specify hardware because of how often virtualization is done now

path: 0 19107951 19113891 19117848, hotness: undefined, score: 2, children: 3
qaz 1 point 10 months ago

The VPS I'm using unfortunately doesn't offer an external firewall

path: 0 19107951 19113891 19117848 19118714, hotness: undefined, score: 1, children: 2
MasterNerd 2 points 10 months ago

Well, if you have the option you could set up a virtual network through the VPS and have a box with pfsense or something to route all traffic through. Take this with a grain of salt - I've seen this done but never done it fully myself.

path: 0 19107951 19113891 19117848 19118714 19122624, hotness: undefined, score: 2, children: 1
ohshit604 1 point 10 months ago

have a dedicated firewall

I mean, don’t router firewalls count in this regard? Isn’t that kinda part of their job?

path: 0 19107951 19127467, hotness: undefined, score: 1, children: 0
Harbinger01173430 12 points 10 months ago

Nat is not security.

Keep that in mind.

It's just a crutch ipv4 has to use because it's not as powerful as the almighty ipv6

path: 0 19120544, hotness: undefined, score: 12, children: 1
qaz 3 points 10 months ago path: 0 19120544 19120959, hotness: undefined, score: 3, children: 0
jwt 11 points 10 months ago

Somehow I think that's on ufw not docker. A firewall shouldn't depend on applications playing by their rules.

path: 0 19108325, hotness: undefined, score: 11, children: 9
qaz 18 points 10 months ago

ufw just manages iptables rules, if docker overrides those it's on them IMO

path: 0 19108325 19108399, hotness: undefined, score: 18, children: 7
jwt 9 points 10 months ago

Feels weird that an application is allowed to override iptables though. I get that when it's installed with root everything's off the table, but still....

path: 0 19108325 19108399 19108525, hotness: undefined, score: 9, children: 4
MangoPenguin 5 points 10 months ago

Linux lets you do whatever you want and that's a side effect of it, there's nothing preventing an app from messing with things it shouldn't.

path: 0 19108325 19108399 19108525 19109266, hotness: undefined, score: 5, children: 2
WhyJiffie 3 points 10 months ago

there's nothing preventing an app from messing with things it shouldn't.

that's not exactly a linux specialty

path: 0 19108325 19108399 19108525 19109266 19112725, hotness: undefined, score: 3, children: 0
ryannathans 3 points 10 months ago

If you give it root

path: 0 19108325 19108399 19108525 19109266 19110621, hotness: undefined, score: 3, children: 0
null 3 points 10 months ago

It is decidedly weird, and it's something docker handles very poorly.

path: 0 19108325 19108399 19108525 19108597, hotness: undefined, score: 3, children: 0
null_dot 6 points 10 months ago

Not really.

Both docker and ufw edit iptables rules.

If you instruct docker to expose a port, it will do so.

If you instruct ufw to block a port, it will only do so if you haven't explicitly exposed that port in docker.

Its a common gotcha but it's not really a shortcoming of docker.

path: 0 19108325 19108399 19110647, hotness: undefined, score: 6, children: 0
pressanykeynow 1 point 10 months ago

iptables is deprecated for like a decade now, the fact that both still use it might be the source of the problem here.

path: 0 19108325 19108399 19112894, hotness: undefined, score: 1, children: 0
IsoKiero 7 points 10 months ago

Docker spesifically creates rules for itself which are by default open to everyone. UFW (and underlying eftables/iptables) just does as it's told by the system root (via docker). I can't really blame the system when it does what it's told to do and it's been administrators job to manage that in a reasonable way since forever.

And (not related to linux or docker in any way) there's still big commercial software which highly paid consultants install and the very first thing they do is to turn the firewall off....

path: 0 19108325 19108526, hotness: undefined, score: 7, children: 0
stupidcasey 10 points 10 months ago

On windows (coughing)

path: 0 19110805, hotness: undefined, score: 10, children: 0
jbk 8 points 10 months ago

rootless podman and sockets ❤️

path: 0 19114474, hotness: undefined, score: 8, children: 0
skuzz 8 points 10 months ago
path: 0 19114853, hotness: undefined, score: 8, children: 3
Diplomjodler3 17 points 10 months ago

Or maybe it should be easy to configure correctly?

path: 0 19114853 19116682, hotness: undefined, score: 17, children: 2
reksas 3 points 10 months ago

instructions unclear, now its hard to use and to configure

path: 0 19114853 19116682 19122681, hotness: undefined, score: 3, children: 0
skuzz 1 point 10 months ago
path: 0 19114853 19116682 19198771, hotness: undefined, score: 1, children: 0
iamroot 7 points 10 months ago

We use Firewalld integration with Docker instead due to issues with UFW. Didn't face any major issues with it.

path: 0 19114423, hotness: undefined, score: 7, children: 1
qaz 4 points 10 months ago

I also ended up using firewalld and it mostly worked, although I first had to change some zone configs.

path: 0 19114423 19114661, hotness: undefined, score: 4, children: 0
cadekat 7 points 10 months ago

I've been playing with systemd-nspawn for my containers recently, and I've been enjoying it!

path: 0 19107285, hotness: undefined, score: 7, children: 3
kurikai 5 points 10 months ago

Try podman and quadlets

path: 0 19107285 19109183, hotness: undefined, score: 5, children: 2
AmbiguousProps 4 points 10 months ago

Quadlets are so good.

path: 0 19107285 19109183 19110704, hotness: undefined, score: 4, children: 0
cadekat 1 point 10 months ago

What advantage does it have over nspawn?

path: 0 19107285 19109183 19120646, hotness: undefined, score: 1, children: 0
peoplebeproblems 5 points 10 months ago

Ok

So, confession time.

I don't understand docker at all. Everyone at work says "but it makes things so easy." But it doesnt make things easy. It puts everything in a box, executes things in a box, and you have to pull other images to use in your images, and it's all spaghetti in the end anyway.

If I can build an Angular app the same on my Linux machine and my windows PC, and everything works identically on either, and The only thing I really have to make sure of is that the deployment environment has node and the angular CLI installed, how is that not simpler than everything you need to do to set up a goddamn container?

path: 0 19108688, hotness: undefined, score: 5, children: 14
qaz 15 points 10 months ago

This is less of an issue with JS, but say you're developing this C++ application. It relies on several dynamically linked libraries. So to run it, you need to install all of these libraries and make sure the versions are compatible and don't cause weird issues that didn't happen with the versions on the dev's machine. These libraries aren't available in your distro's package manager (only as RPM) so you will have to clone them from git and install all of them manually. This quickly turns into hassle, and it's much easier to just prepare one image and ship it, knowing the entire enviroment is the same as when it was tested.

However, the primary reason I use it is because I want to isolate software from the host system. It prevents clutter and allows me to just put all the data in designated structured folders. It also isolates the services when they get infected with malware.

path: 0 19108688 19108883, hotness: undefined, score: 15, children: 4
peoplebeproblems -1 points 10 months ago

Ok, see the sandboxing makes sense and for a language like C++ makes sense. But every other language I used it with is already portable to every OS I have access to, so it feels like that defeats the benefit of using a language that's portable.

path: 0 19108688 19108883 19109134, hotness: undefined, score: -1, children: 3
hitmyspot 6 points 10 months ago

I think it's less about making it portable to different OS, and more different versions of the same os on different machines with different configuration and libraries and software versions.

path: 0 19108688 19108883 19109134 19111613, hotness: undefined, score: 6, children: 0
WhyJiffie 3 points 10 months ago

it does not solve portability across OS families. you can't run a windows based docker image on linux, and running a linux image on windows is solved by starting a linux VM.

path: 0 19108688 19108883 19109134 19112792, hotness: undefined, score: 3, children: 1
peoplebeproblems 2 points 10 months ago

Oh, fair. That's a good point.

path: 0 19108688 19108883 19109134 19112792 19112970, hotness: undefined, score: 2, children: 0
sidelove 13 points 10 months ago

have to make sure of is that the deployment environment has node and the angular CLI installed

I have spent so many fucking hours trying to coordinate the correct Node version to a given OS version, fucked around with all sorts of Node management tools, ran into so many glibc compat problems, and regularly found myself blowing away the packages cache before Yarn fixed their shit and even then there's still a serious problem a few times a year.

No. Fuck no, you can pry Docker out of my cold dead hands, I'm not wasting literal man-weeks of time every year on that shit again.

(Sorry, that was an aggressive response and none of it was actually aimed at you, I just fucking hate managing Node.js manually at scale.)

path: 0 19108688 19108851, hotness: undefined, score: 13, children: 2
TrickDacy 3 points 10 months ago

I agree, for any context that it makes sense, docker is so worth it

path: 0 19108688 19108851 19109142, hotness: undefined, score: 3, children: 0
peoplebeproblems 0 points 10 months ago

Well, I guess that's a good reason. Node version management seems to handle most of that for me though. I haven't worked on an OS without support for it.

path: 0 19108688 19108851 19109097, hotness: undefined, score: 0, children: 0
null_dot 7 points 10 months ago

Sure but thats an angular app, and you already know how to manage its environment.

People self host all sorts of things, with dozens of services in their home server.

They dont need to know how to manage the environment for these services because docker "makes everything so easy".

path: 0 19108688 19110730, hotness: undefined, score: 7, children: 0
missphant 5 points 10 months ago

I put off docker for a long time for similar reasons but what won me over is docker volumes and how easy they make it to migrate services to another machine without having to deal with all the different config/data paths.

path: 0 19108688 19109419, hotness: undefined, score: 5, children: 0
Lightfire228 4 points 10 months ago

Think of it more like pre-canned build scripts. I can just write a script (DockerFile), which tells docker how to prepare the environment for my app. Usually, this is just pulling the pre-canned image for the app, maybe with some extra dependencies pulled in.

This builds an image (a non-running snapshot of your environment), which can be used to run a container (the actual running app)

Then, i can write a config file (docker-compose.yaml) which tells docker how to configure everything about how the container talks to the host.

  • shared folders (volumes)
  • other containers it needs to talk to
  • network isolation and exposed ports

The benefit of this, is that I don't have to configure the host in any way to build / host the app (other than installing docker). Just push the project files and docker files, and docker takes care of everything else

This makes for a more reliable and dependable deploy

You can even develop the app locally without having any of the devtools installed on the host

As well, this makes your app platform agnostic. As long as it has docker, you don't need to touch your build scripts to deploy to a new host, regardless of OS


A second benefit is process isolation. Should your app rely on an insecure library, or should your app get compromised, you have a buffer between the compromised process and the host (like a light weight VM)

path: 0 19108688 19109377, hotness: undefined, score: 4, children: 0
MangoPenguin 4 points 10 months ago

The only thing I really have to make sure of is that the deployment environment has node and the angular CLI installed

That's why Docker is popular. Making sure every single system running your app has the correct versions of node and angular installed is a royal pain in the butt.

path: 0 19108688 19109240, hotness: undefined, score: 4, children: 0
TrickDacy 3 points 10 months ago

Even when it seems like an app runs identically on every platform, you can easily run into issues down the road. If you have a well configured docker image, that issue is just solved ahead of time. Hell, I find it worth messing with just moving a node.js app between Linux boxes, which would experience the least issues I can think of.

path: 0 19108688 19109107, hotness: undefined, score: 3, children: 0
llii 1 point 10 months ago

You’re right. As an old-timey linux user I find it more confusing than running the services directly, too. It’s another abstraction layer that you need to manage and which has its own pitfalls.

path: 0 19108688 19108858, hotness: undefined, score: 1, children: 0
m33 5 points 10 months ago

Docker does not play fair, does not play nice. It’s a dozer that plow through everything for devops that yolo and rush to production.

path: 0 19106419, hotness: undefined, score: 5, children: 0
purplemonkeymad 3 points 10 months ago

Well yea ofc it works like that, the services are not on the same network, so the packets need to be sent onto another adapter. That means either nat or forwarding tables.

Now if that was a good design of docker is another question.

path: 0 19109114, hotness: undefined, score: 3, children: 0
communism 2 points 10 months ago path: 0 19111237, hotness: undefined, score: 2, children: 3
qaz 1 point 10 months ago

That caused issues with Docker containers being unable to communicate with eachother for me.

path: 0 19111237 19113852, hotness: undefined, score: 1, children: 2
tux7350 2 points 10 months ago

Did you allow the containers to talk to eachother with ufw after setting it up?

path: 0 19111237 19113852 19114432, hotness: undefined, score: 2, children: 1
qaz 2 points 10 months ago

Yes, you can fix it by doing this

echo "Adding rules for Docker subnets to allow communication between containers..."
for subnet in $(docker network inspect bridge -f '{{range .IPAM.Config}}{{.Subnet}} {{end}}'); 
    sudo ufw allow from $subnet
    echo "Added rule for $subnet"
done
path: 0 19111237 19113852 19114432 19114644, hotness: undefined, score: 2, children: 0
lemming741 2 points 10 months ago

Wait, that's illegal

path: 0 19107831, hotness: undefined, score: 2, children: 0
programmer_humor
programmer_humor

@programming.dev

login for more options
31959
2364
7242

Welcome to Programmer Humor!

This is a place where you can post jokes, memes, humor, etc. related to programming!

For sharing awful code theres also Programming Horror.

Rules

  • Keep content in english
  • No advertisements
  • Posts must be related to programming or programmer topics

go to feed...