7c0h

So you want to use my PC

So you want to use my PC? Sure, no problem, it's right there. But real quick, before you start, there's a couple small tips that you should keep in mind.

If you turn it on and nothing happens, that means the KVM switch is listening on the wrong input. Just press the black button behind the screen to switch it back. Can't find it? Maybe it fell down - it has a short cable and sometimes the other cables pull it back. Or maybe it's underneath all of my drawing stuff. Either way, search for it behind the desk and press it. The screen should turn on.

You may also notice that the computer chair has been relegated to the side and that I'm using a regular wooden chair instead. You can use the fancy one if you want - after a couple weeks it may cause some completely unexpected health issues, but on a short enough scale it's fine.

Now, this computer runs Linux. Technically it runs Windows too, but the screen for choosing a different operating system goes away before the monitor has time to turn on, and we lost too much time to spam the arrow keys anyway. But that's probably for the best: given that I haven't booted Windows in a long time, that Windows loves mandatory updates, and that the computer has a mechanical hard drive, there's a good chance that you would have to wait at least an hour before being able to use it. Why haven't I upgraded to an SSD, you ask? I have considered it, but Windows won't let me for complicated reasons. So Linux it is.

You may see a bunch of text scrolling by. That's fine. If the text stops scrolling for a long time you may want to consider pressing the NumLock key several times. I'm not entirely clear on the details, but something about interruptions and IRQs sometimes keeps the PC from starting if it doesn't detect activity. And since the NumLock key doesn't do anything bad, it is the safest key to spam-press.

Once you've made it to the login screen it is going to ask you for a username and password. That one should be easy - I have been using the same combination for over 20 years, so you probably know it already. Need a hint? It's in the same league as my network name plus my favorite number. It's not exactly hunter2, but it's close.

You may have a bit of trouble reading the letters on screen. Don't worry, the screen works fine - I just keep my brightness and contrast real low, and even the smallest hint of natural light turns them invisible. You could adjust the brightness using the monitor's crappy touch controls, or you could do what I do and close the curtains. I'm not going to tell you which one is right, but I am going to vouch for the one that has kept me glasses-free for more than 25 years of extensive computer use.

If you want a graphical interface, type startx and my Mate Desktop Environment will start. Ignore the error about sticky notes - I have been meaning to fix it for some time, but there's always something else and it doesn't bother me anyway. But before you type anything, a word of caution: the interface is in English, but the keyboard is German and the key mapping is Spanish. That means that regular keys are where you expect them to be (except for Z and Y, which are inverted), but if you want to type a single quote you need to add a space afterwards or you'll get an accented letter otherwise. If you have a password with more than just letters and numbers then you may want to type it in a text editor first and copy-paste.

You probably need a web browser, in which case the Firefox icon is right there. But you probably don't want that: the script blockers almost guarantee that whatever website you are trying to open will not work. For those cases I recommend using the super-outdated Chrome browser instead - the icon is right next to the Firefox one. It won't play Netflix because I didn't install the DRM stuff, but everything else should work fine. If you want audio, remember to plug the headphones first. And that reminds me: I have set the browser to auto-delete all history when closing the browser. So don't close the window until you're done, or you'll have to login again. I hope you didn't close the text editor with your password!

Now, I'll be honest with you: at this point most people give up and resort to their phones. And it's okay, I get it. That's what happens when you keep your tools exactly like you want them to be and suddenly you are asked to share. But most people tend to blame Linux for it, which I think it's unfair - if anything, I'd like to see Windows offer even half the flexibility to do things my way.

Now, let's talk about that WiFi password. But first, one question: do you know how to type the "¿" symbol on your keyboard?

Inspired by this article

On bad deals

An article titled Don't talk to Corp Dev has recently showed up in my feed and the HN discussion around it reminded me of this anecdote.

To clarify: this story is based on something that happened in real life, as told to me by someone who witnessed it first-hand. Time has blurred some of the details and I have changed others. So feel free to dismiss the details, but please keep the main point in mind.

Once upon a time there was a small wine producer we'll call Dave. He kept a small shop and was doing well by himself, selling his wine at $15 a bottle. One day a big supermarket chain contacts him -- the chain needs a new wine supplier, and they heard good things about Dave. They have just one condition: they will take care of distribution and advertising, but Dave has to provide them with one month of free wine in return.

Dave is ecstatic: the supermarket chain can spread his brand across the entire country and turn his small operation into a national contender. Sure, giving up one month of profit is not ideal, but the supermarket representative convinces Dave that he'll recover his lost income in no time. Hands are shaken, contracts are signed, and Dave provides them with one free pallet of wine to start the week.

Week two arrives, and the supermarket chain requests two pallets of wine. "Looks like my wine is selling well", Dave thinks, and increases his production targets for the week. But this is not enough: when Dave is asked for five pallets of wine the next week he is forced to hire a new assistant and buy new machinery to keep up. Predictably, week four closes with an even larger order of ten pallets. The loss of 18 pallets of wine is a tough pill to swallow, but at least the month is over and from now on he will get paid.

A new month begins and Dave receives his weekly order: he is asked to deliver a grand total of one pallet of wine. "This must be a mistake", Dave thinks, and calls the supermarket representative. "Last week I gave you ten pallets of wine. So why are you only asking for one today?". "That's easy", says the representative, "up until last week we were selling your wine for a dollar a bottle. This week we will put the price back at $15 and therefore we are not expecting it to sell as well as before".

Facing a large loss of inventory and having invested both in employees and machinery that are no longer needed, Dave goes broke. The supermarket chain finds a new naive wine supplier, and the cycle starts again.

A wise man once taught me that it's not a good idea to enter into negotiations with someone much bigger than you. This is arguably the reason why South American countries formed Mercosur instead of joining theFTAA agreement with North America, but it's also true for simpler activities like office politics.

For the record, I would have totally fallen for this. The first time I heard this story I didn't even consider that someone could do something like this on purpose. Then I started reading news about startups and learned that it happens relatively often.

Consider yourself warned.

Rust Part II: What is it good for?

Last time I talked about Rust I mentioned that I wanted to like the language but I couldn't find a good reason for using it. Luckily for me, the last Advent of Code gave me the perfect reason for doubling down on Rust, and here's my updated report.

In case you never heard of it, Advent of Code is an online competition that takes place every year in December. It is structured as an Advent calendar where you get a puzzle every day, and where the puzzles get harder and harder every day. Plenty of people use this competition as the perfect excuse for learning a new language, which is how I ended up programming lots of Rust in my spare time.

So here they are: in no particular order, these are the things I like, dislike, and feel mildly uncomfortable about Rust.

Things I like

The one thing I like the most about Rust is the power of the match operator combined with enums. Unlike in Python, where the implementation of the match statement is pretty dangerous, Rust makes it easy to program the type of code that's easy to write, read, and maintain:

let mut pos = 0;
let mut depth = 0;
for instruction in orders {
        match instruction {
                Instruction::Forward(meters) => pos += meters,
                Instruction::Up(meters) => depth -= meters,
                Instruction::Down(meters) => depth += meters,
        }
}

Then, there are the compiler errors. While not true for external crates (we'll get to it), compiler errors in Rust are generally helpful, identify the actual source of the problem, and sometimes even give you good suggestions on how to solve the issue. Gone are the days in which a compiler error meant "I know an error happened 50 lines above, but I'll complain about it here instead".

And finally, as someone who has been doing mostly Python for the last years, it feels so good not to have to worry about indentation anymore. This doesn't mean that I'll stop indenting my code - instead, it means that I can finally move a function around without worrying about pasting it one indentation to the left and turning a class into a class and multiple pieces of code that don't compile.

Things I hate

I am puzzled by how aggressively unhelpful arrays are. The puzzle for day 25 could be easily solved (spoilers!) by writing

horizontal_row = horizontal_row>>1 && !(horizontal_row || vertical_row)

but I ended up having to implement it with Vectors of booleans instead. Why? Because I didn't know how many bits horizontal_row would have at compile time, and Rust refuses to create arrays with dynamic size. I am sure there is a way to keep a large binary in memory and manipulate it at the bit level - otherwise, you wouldn't be able to use Rust for serious game development. But whatever the method is, it is well hidden.

And on the topic of that puzzle, I come back to one of my main complaints from last time: popularity is not the correct way to decide which library is the best one for the job. Do you know the difference between the bitvec, bit-vec, and bitvector libraries? Can you add either of them to your code without worrying about the developer going rogue? How about the fact that the first result that comes up for rust bit vector is an accepted StackOverflow answer suggesting bit-vec... which is no longer maintained?

Minor annoyances

I still can't make sense of the module system. I mean, sure, I know how to put functionality in sub-directories, but that doesn't really explain why I would choose between lib.rs, day24.rs, or day24/vm/mod.rs. The book could use some improvements on this topic.

If I'm doing something like u16 = u16 + u8 (or even better, u16 += u8), the compiler should cast the last value automatically. u8 += u16? Sure, I get it, that's an overflow waiting to happen. But there is no need for me to get in there and write u16 = u16 + u8 as u16 when we all know the data fits just fine.

The collect function is very finicky. This is a function that I used quite often in constructions like .map(|x| something(x)).collect.to_vec(), but more often than not it will complain about not knowing the type required for collect even though there is only one type that would make sense.

And since we are talking about the compiler, one of the crates I needed (it was either nalgebra or ndarray, where I suffered the same problems I had with bit vectors) had a nasty side effect: if one of your instructions failed to compile, they all stopped compiling. Good luck finding the one line that needs fixing!

And finally, I ran a couple times into functionality that had been deprecated in favor of functionality that doesn't currently exist. Not cool.

Conclusion

Would I use Rust again? Yes.

Is it my most loved language? No. But under the right circumstances I could see it happening.

What is it good for? Last time I jokingly said "writing Rust compilers", and I wasn't that far: it's the right programming language for apps that need performance and memory safety, and where we are willing to spend some time calculating who is borrowing from whom in order to get code with fewer bugs. So it's pretty much C++, only with borrowing replacing memory allocations.

I like the idea of giving my original project another try, but I can't make any promises. The Advent of Code has already pushed forward the date of my next project by a couple months, and the time it's taking me to migrate my infrastructure to Ansible is making everything worse.

How not to sell tickets

Let's say you are a band and you want to keep scalpers from buying all of your tickets and reselling them at ridiculous prices.

By now we know what works: you sell tickets with assigned names on them, and check at the door that the name on the person's ID (or, alternatively, credit card) matches the name on the ticket. You can also offer a "+1 option" for bringing someone along, but they must be accompanied by the person who bought the tickets in the first place. No ID, no entrance.

By contrast, here's what doesn't work:

Screenshot from the band's website: Please don't buy from scalpers or scammers. We have launched a scalper-free ticket exchange.

How do I know that it doesn't work? For starters, because all of the regular $40 tickets are sold out:

Screenshot from the ticket seller's website: ticket at $49.52 sold out

And because the only available tickets are those sold by scalpers at 10 times their value:

Screenshot from another ticket website: tickets starting at $500

You know what I would do if I were a scalper? I would buy as many tickets as I could, and then I would also sign up for the exchange with as many fake addresses as possible. It doesn't matter that I can't use the exchanged tickets afterwards - as long as the supply is reduced, the value of the tickets I bought first will only go up. Since every overpriced ticket offsets the cost of the other 9 I got early, it all works out for me at the end.

By now it is well known that ticket companies don't get rid of scalpers because they don't want to. The method I mentioned in the first paragraph is proven to work, but why would ticket companies get rid of scalpers when they can reach deals with them and get a second cut on the same ticket?

Which brings me to my final point. I think there are three types of bands:

  • those that care about their fans and successfully prevent scalpers from getting all the tickets,
  • those that care about their fans but don't know what to do (and/or do it wrong), and
  • those that don't care about scalpers because they get paid anyway (or get paid even more)

If you are a member of the first type of band I applaud you for standing up to your fans. And if you know someone who is in the second situation, feel free to forward them this post.

I could share some rough words with those who belong to the third group. But honestly, why bother? It's not like there's a shortage of good musicians. I'll go watch those instead.

April 2022 edit: John Oliver has a funny segment on this topic. You can watch it on YouTube

Ungooglable Azure errors

I recently found myself in a situation I haven't experienced in a long time: I got an error message that no one on the internet has had before. Neither DuckDuckGo, Google, nor Bing could find a single hit about the problem, much less a solution. The root cause (spoilers!) ended up being rather boring: I tried to use some features of the Azure cloud platform that are apparently blocked by the IT department of my company.

I don't know exactly why these errors are impossible to find - either they were written by someone in my company's IT department or my employer is the only one who blocks the OAuth authentication flow. Either way, and as a service to the community, I am copying the error messages here.

The first one means that you are trying to use any type of authentication other than the allowed one:

AADSTS1000470: The protocol OAuth2DeviceAuth is blocked for tenant <tenant-id>. Please contact your administrator for assistance.

The second one means that you tried to create a resource that the internal rules don't allow. I know I hit this one when following an ML tutorial, but I can't remember the specifics right now. Hint: If you also got the string "RBAC restrictions" somewhere in the description, then you are seeing the same one:

At least one resource deployment operation failed. Please list deployment operations for details. Please see https://aka.ms/DeployOperations for usage details.

code: RequestDisallowedByPolicy
message: Resource '<resource-id>' was disallowed by policy. Policy identifiers: <bunch of json with redacted information>

If you run into these error messages, feel free to get in touch with me and I'll share all of my poor Azure wisdom with you.