tagged: Typescript, bugs
Thanks to Zachary Kyra-Derksen for making this photo available freely on unsplash 🎁
If you feel bad about a production bug you've created, don't worry. I have too. Here is a running list of bugs I've created and/or squashed.
[x] Created and/or [ ] Squashed
c. 2016
I was getting really into ES6 soon after it came out, watching tutorials and doing exercises online too (RIP CodeSchool). The Senór dev on the team (for I was a humble non-Senór at the time) went on a two-week vacation, so it meant that it was my turn to push the deploy button. I was excited to press 'deploy' and saw the opportunity to prove myself to the upper managers. Well, soon after the deploy we started getting phone calls from our users, who are mostly older guys and mostly used Internet Explorer. I maintained that it was their fault and that I did nothing wrong with valiant effort, but was ordered to rollback the deploy anyway. The rollback immediately fixed the issue. I did not impress the upper managers.
When Senór Dev came back he found the issue in about two seconds - I used an arrow function. Our app did not support ES6 transpilation yet. Darn.
Lessons Learned :
• 👉 Multiple browsers are a good idea when testing. If we had tested in more than the two most recent versions of Chrome, we would have caught the issue. We pulled BrowserStack into our QA pipeline after that.
• 👉 Do not deploy when Senór is not there
NODE_ENV='production'
killed my dev dependenciess[x] Created and/or [ ] Squashed
c. 2022
I mostly work alone these days. And when you're a team lead of a team of one, I feel that you're authorized to take certain 'shortcuts'. Sometimes I commit directly to the master branch. Well, one of the shortcuts went too far. And it was born while debugging.
I had to connect my local machine to our server's Azure subscription to debug an issue. To do this I copied the server's systemd
config file (for the whole app ran on a VM) into my local to get all the environment variables - connection strings, guids, etc. Well, I inadvertently copied the following line:
NODE_ENV=production
So I was setting NODE_ENV on my local to production
. Ouch!
I'm sure this caused me a whole bunch of issues, like "why is this if statement not evaluating correctly", but the cake goes to npm run build
, which in production deletes all of your devDependencies
(npm docs for reference) . This was rough for my typescript insecurities because all of the @types
packages are, you guessed it, in devDependencies
. So when I'd run a build I'd get eslint errors about a standard configuration not being defined. I didn't know what to do except to blow the whole root of the repo away and pull it again. Two months this went on.
Lessons Learned :
• 👉 BE CAREFUL when copying config options from server to local; ask yourself, "is this necessary?" In my case it was, but I copied them mindlessly.
• 👉 You can export NODE_ENV=production
to your friend at work for hilarious results, especially if you're using Typescript!
[ ] Created and/or [x] Squashed
One of our apps at work is a dashboard that has different roles. We reserve a super admin role for us to go in and do admin-level things. There are two other, (regular) admin and reader roles. Turns out we stored that user role in local storage. Oof.
Well, the hack goes that if you are a regular admin, and know what the role name is for the super admin, you can just change the role name in local storage, refresh the page, and all the super admin bells and whistles are yours!
It was not trivial to store this information in a more secure manner. The dashboard is pretty lightweight and did not use cookies, so I didn't want to introduce them for this. I thought about trying to correct the role name and id if I detected that it changed, but that seemed to require too much checkwork.
The fix I decided on was pretty simple - md5-hash the role name and id, and add that to localstorage. The next time the role is checked, in a method named getUserInfo()
, it compares the existing (potentially hacked) role name and id to the hash. If they are not the same, the user is immediately logged out. The angular router makes logging out programmatically a cinch.
Lessons Learned :
• 👉 don't store sensitive information in localstorage!
mat-select
as type never[]
[ ] Created and/or [x] Squashed
To be honest here, I am not sure what was going on with this one. The root of the problem was an implicit type declaration that looked like an array initialization.
Here was the offending code:
export interface ProviderCredentials {
providerId: string
supported: CredentialTypes[]
}
nullProvider = {
providerId: '',
supported: []
}
selectedProviderCredentials: ProviderCredentials = this.nullProvider;
ProviderCredentials
is defined in another place:
export enum CredentialTypes {
Tuple,
Certificate,
Instance,
UserConfig
}
export interface ProviderCredentials {
providerId: string
supported: CredentialTypes[]
}
The selectedProviderCredentials
was populated from a fetch call at the beginning of the component' lifecycle. Up to that point it was supposed to be empty, hence it was initialized with the nullProvider
. And the supported
array populated the options in a mat-select
.
If you're conscientous you'll notice that nullProvider
is not defined as a ProviderCredentials
, even though in the next line it is set equal to an instance of one. In fact, it is defined as its own type, with supported
not as a CredentialTypes[]
, but as a never[]
.
For some reason, since the mat-select
is populated as a never[]
, its @ViewChild
correspondent is always undefined. Hence, when the data comes in, the mat-select's value is not set.
I asked ChatGPT to explain what never[]
is used for:
Yes, if TypeScript is inferring the type of the supported array in your nullProvider object as never[], this can indeed be a code smell. The never type in TypeScript is used for values that never occur. When TypeScript infers an array as never[], it generally means that it doesn't expect the array to be used with any values at all—effectively, the array should always remain empty.
Turns out all we had to do was initialize the nullProvider
as an implementation of the interface, along with a default value:
nullProvider: ProviderCredentials = {
providerId: '',
supported: [CredentialTypes.Instance]
}
Apart from the accidental never[]
, I can understand the desire to initialize things with empty or null values. But in most cases, the call to hydrate the UI component will return quickly enough that a reasonable default, especially of an enum value which by nature details the acceptable values of a thing, is just fine.
We could have a conversation about whether CredentialType
shouldn't be an enum, but that question would open up cans of refactoring I don't have the time for..
There's some typescript voodoo here that I'm not understanding fully, but this fixed up our issue.
Lessons Learned :
• 👉 Remember the difference between initialization and property declaration!
• 👉 If you're going to use types, use types everywhere. Failure to do so will fork things up.
• 👉 It's important to remember to reasonable defaults for things; dummy values might work better than empty arrays, at least in the non-React case.