How are you hiding your API keys?

I'm just curios how you guys are hiding you api key in the scripting module, are you hashing it in a database or something similar? Just looking for suggestions.

Thanks!

I recommend placing secrets in a json file outside the Ignition install folder (so it won't be caught in backups or GIT) and loading the whole file into a script module top level variable (outside any function, so it caches). This conveniently also makes it easy to have testing secrets in DEV.

1 Like

We use OS environment variables, but how you set them is OS dependent.

On Ubuntu, we tweaked the Ignition-Gateway systemd service to load extra variables from a file that we keep outside of the Ignition folder (like pturmel suggested).

I know this is popular in the devops world, but prevents hiding secrets from privilege escalation attacks. Linux's SELinux framework (and others) can prevent access to file-based secrets and regular process memory even by the root user, but cannot prevent access to a process' environment exposed in /proc/.

I strongly recommend using file-based secrets storage even if you aren't yet using SELinux, so you can migrate to that level of security when you need to. This is particularly important when delegating secrets storage to a external vault, as you must somehow secure the local credentials you use to authenticate to the vault itself.

It's one of the things that makes me unhappy with common practices in Docker. :frowning_face:

Some background:

4 Likes

I'm so used to the idea that "if they have root they have everything" that the concept of "hardening against a compromised root" felt alien at first. But I see your point.

In our case we're halfway there since the secret variables all in a .env file that systemd imports to the environment. We could just turn off the systemd import and make it a scripted import of the same file inside Ignition.

3 Likes

Any suggestions on how to hide a file in a docker container? would hashing it in a database be a bad idea?

Use docker's standard secrets management, which places the secrets file in the container's /run/ memdisk hierarchy. Which is naturally outside Ignition's folders.

1 Like

Thanks, I'll check it out. I have a question about importing the file too. If you are in the gateway, what is to prevent you from printing out the variable to get the apikey, is it worth the effort to keep it in an external file to import? Thanks again!

Nothing. There is no way to stop that. It's one of the reasons designer access is effectively full access to a gateway.

Yes, as it protects against hacks that gain access to the OS, but not from within Ignition.

1 Like

Thanks for the clarification!

Somewhat related question, once i have a config setting or token or whatever that I've pulled into Ignition this way in a script, is there any way to get at it using a simple expression if I haven't pushed it into a tag?

Like if I've saved it into a dictionary in a script library named shared.config then I can obviously get to it as shared.config.whatever from another script. Or I stored it in the globals area I could use system.util.getGlobals(). But neither of those are reachable via expressions, right?

Correct.

You would probably have to use pturmel's Integration Toolkit's globalVarMap method to retrieve it via an expression, and you would also need to use the associated system.util.globalVarMap() method instead of system.util.getGlobals()

https://www.automation-pros.com/toolkit/doc/scripting.html#globalvarmap

2 Likes