Running a shell command when exiting an Elixir script

I have a mix task that does some stuff, and in the end it created a shell command that I want to run. The command can be something that willl require full IO support, so you can something like a psql session, an ssh session, etc.

Is there some way to run that command before/after exiting the Elixir task/beam so it is executed as a normal command in bash?

If not, is there some way that I can run that command inside the beam, but the beam give full IO handling to the process until it finishes?

Trying to doing something like this with System.cmd doesn’t seem to do the trick.

1 Like

Would Porcelain.shell/2 or Porcelain.exec/3 help you?

Thanks for the suggestion, but it doesn’t seem to work from my tests, it will still not give the keyboard IO control to the external program/shell

1 Like

I used to have an issue with running a system command that needed te be independent of the Beam, I wrote a nif to handle this, don’t know if it will help you. It’s on hex.

I must say it’s been years since I have used it. The project it was for has been archived.

3 Likes

Hey @Hermanverschooten , I just tried it but unfortunately it doesn’t give the process access to the terminal input.

Here is what I did as an example:

iex(1)> Mix.install([:ext_run])
:ok
iex(2)> ExtRun.run("fly ssh console --app eduardo-backend-dev"); :erlang.halt()

This will create an ssh session to a server I have in Fly and halt the beam. Since the process is outside the beam, it does keep running, but then it will give me the following error because it can’t access the input/output:

Connecting to fdaa:20:...:f93c:2... complete
Error: ssh shell: input/output error

Why don’t you do the oposite? Call an Elixir Script from a Bash Script, and when it exits you can do anything.

You could even get the exit codes to do some logic.

2 Likes

Because I want to have these as mix tasks, I don’t think I can call bash directly from a mix task AFAIK

I don’t remember the code it is too long ago I wrote it.
But what comes to mind is if you could pass along the current file descriptors for input and output to the new process or could be you need to start a pty.

I made good use of erlexec in claude_code_sdk_elixir:

(repo is on an exploratory journey on the main branch currently, apologies)

1 Like