Search code examples
clojureread-eval-print-loopleiningendeps-edn

Clojure: how to tell if code is running in the REPL or a JAR?


I am writing a CLI framework in Clojure called OneCLI. The main center piece of this framework is a function called go! which parses the command line, environment variables, and config files "for you" and runs one of several different user provided functions based on what was provided in those inputs.

Typically, go! is called from the -main function of the user's calling Clojure program. I use my own library, for example, in another "uberjar" style app called zic. The function go! calls System/exit as part of its run, passing it an exit code that comes from the result of the user provided function. This works great "in production", but it also means that I can't run the zic.cli/-main function from the REPL, as whenever I do it calls System/exit and the REPL exits.

Before you ask, running it from the REPL while developing on a raspberry pi avoids the expensive 45 seconds it takes to run lein uberjar/1 minute 30 seconds to run clj -X:depstar uberjar :jar ....

My question is: Is there some var or value I can check as part of Clojure's standard library that tells my OneCLI code whether it's running from the REPL or if it's running from a JAR?

Such a variable would enable me in OneCLI to detect that we're running from a REPL so that it can avoid calling System/exit.


Solution

  • Instead of trying to have one function that magically detects what environment you're running from, it's quite simple to just have two functions that behave differently.

    • Extract out the shared behavior to a function that is not part of -main. Call it run or whatever.
    • Have -main call that function, and then call System/exit
    • When you wish to use the program from a repl, call run instead of -main. It will finish normally, and not call System/exit.