tools-deps

Discuss tools.deps.alpha, tools.build, and the clj/clojure command-line scripts! See also #depstar #clj-new
henrik42 2021-04-10T10:20:38.406100Z

Hi! I'm doing my first experiments with clj and noticed that I can put this into my deps.edn:

:aliases {:foo {:main-opts ["-e" (let [x :foo] (println x))]}
and then use -Mfoo to run. Is this ok? Can I use any form in there? I found https://clojurians-log.clojureverse.org/tools-deps/2018-06-11 and I'm wondering, what came out of it.

practicalli-john 2021-04-10T10:43:20.406200Z

clojure -M:foo should run that expression. Unless you really use an expression a great deal or need it for some kind of work-around, I don't see the value myself. It seems simpler to use the -e flag on the command line.

practicalli-john 2021-04-10T10:45:03.406400Z

There are a whole range of example aliases in https://github.com/practicalli/clojure-deps-edn

henrik42 2021-04-10T10:49:18.406700Z

I'd like to put clojure forms into the aliases as examples. Putting them on the command line would make them platform dependent. So I just wanted to check that putting them there is fine.

alexmiller 2021-04-10T12:21:28.407500Z

That embedded Clojure form there is not valid syntax in deps.edn

alexmiller 2021-04-10T12:22:05.407900Z

So putting that there is not fine

alexmiller 2021-04-10T12:23:02.408700Z

:main-opts is a vector of strings

henrik42 2021-04-10T12:51:19.410Z

@alexmiller thanks for the clarification

2021-04-10T19:15:04.412700Z

Hi all, wondering if there's a way to retrieve the resolved deps environment, specifically the fully resolved :paths attribute after all the aliases have been merged and a main function has been launched? Thanks!

seancorfield 2021-04-10T19:17:09.413800Z

@zalky Not really. By the time your code is actually running, a lot of that information has been merged away.

seancorfield 2021-04-10T19:18:20.415200Z

You can use tools.deps.alpha as a library to compute the “project basis” which is a superset of everything in the root deps.edn, user deps.edn, and project deps.edn files but you can’t tell which aliases were used to invoke your code — only what different paths/deps combinations would be computed from any given aliases.

seancorfield 2021-04-10T19:18:47.415800Z

You can get the classpath though — which might be enough for your needs — via a JVM system property.

seancorfield 2021-04-10T19:20:17.416800Z

(System/getProperty "java.class.path")
will produce a very long string of JAR files and directories separated by (System/getProperty "path.separator") (which is different on Windows vs macOS/Linux).

2021-04-10T19:23:48.419600Z

Thanks @seancorfield, this is very helpful. If I filter that list for directories, would that be equivalent to the fully resolved paths? Might there be anything else thrown in? My motivation is that I'm doing some work with clojure.tools.namespace and I need to know all the directories I might have source in.

seancorfield 2021-04-10T19:24:32.420400Z

You’ll get directories for the equivalents of src, resources, and maybe test plus any git deps or local/root deps in play.

👍 1
seancorfield 2021-04-10T19:27:27.421800Z

You could probably filter out any absolute paths and what is left should be just your own project’s stuff (depending on how you have your project set up).

2021-04-10T19:31:06.423300Z

Gotcha, thanks!

alexmiller 2021-04-10T19:47:48.423700Z

what's your actual goal? project paths?

2021-04-11T14:19:12.430900Z

@seancorfield, I am working with a component based system, and the particular reloaded workflow I am using automatically restarts the running application when reloading a namespace would otherwise leave stale objects to gum up the works. c.t.n is the thing that determines when such a namespace has changed. Do you have any alternatives workflows that you recommend for working with a stateful application? I'm not sure I immediately see a way around it when working with component. At least the reloading task is fairly straightforward. The bigger picture is that we're in the process of migrating our build framework from boot to deps, which has been great in many respects. But because deps is much narrower in scope, the work has been figuring out how to stitch our previous build and development workflows back together.

seancorfield 2021-04-11T16:44:02.431300Z

We use Component very heavily at work - we also migrated from Boot to the CLI - we do not use any sort of reloaded workflow. We evaluate every change we make into the running program with no restarts.

2021-04-11T17:11:39.431800Z

@seancorfield, interesting, I think most of the justification for the reloaded workflow comes down to stale records not implementing newer instances of a protocol. Do you just take special care to make sure protocol namespaces are only ever loaded once?

seancorfield 2021-04-11T17:13:14.432Z

We don’t change records/protocols much after the initial work on them. We also tend to use protocol implementation via metadata these days (since Component supports that).

seancorfield 2021-04-11T17:14:18.432200Z

If you find you are changing protocols so much that you are forced to remove + reload namespaces a lot, I would take a harder look at how you’re using protocols (or records) and see if you can optimize how you design/build them.

2021-04-11T17:27:45.432400Z

I agree, it's like you say: they don't change too much after the initial design, and the need for them is not very common to begin with, but it was nice to have even that edge case covered. Maybe it's worth revisiting at some point during our build migration. Thanks again this was very helpful.

seancorfield 2021-04-11T17:51:16.432600Z

I mean, we do have RCFs (“Rich Comment Forms”) with component/start / component/stop forms in them so we can eval code to start and stop systems when we need to but mostly I start the REPL (outside the editor — because it lives longer than my editor, in general) and just leave it running for weeks, and then eval the main ns for an app and then eval a component/start form and just work on a feature until I’m done, evaluating every single change I make (often without even bothering to save files). If I need to switch to another app in our monorepo, I may stop the first app before starting the second app, but I often just have multiple apps running in my REPL (I tend to make sure tests start/stop components as needed and use different ports to the dev port).

dominicm 2021-04-12T18:28:47.467Z

@zalky

dominicm 2021-04-12T18:28:56.467200Z

ugh, sorry. I have code for this I can share

dominicm 2021-04-12T18:30:26.467400Z

https://git.sr.ht/~severeoverfl0w/wedge/tree/master/item/src/io/dominic/wedge/user.clj#L48-86 I intentionally leave in :local/root deps in case you are doing development on them, but you can use exclude-libs-refresh to remove those libraries manually.

2021-04-13T14:11:15.489800Z

Thanks @dominicm for sharing, that's a useful example!

2021-04-10T20:23:23.423800Z

Hi @alexmiller, I have a clojure.tools.namespace based reloaded workflow, which requires as input a set of directories to watch for changes. While I can supply a list of hard-coded directories in :main-opts , my understanding is that project paths is dynamically computed from all the active aliases.

alexmiller 2021-04-10T20:24:58.424Z

it is, but there is not an explicit place to get just the paths

alexmiller 2021-04-10T20:25:24.424200Z

the classpath computed by tools.deps will put all paths at the front of the classpath though

alexmiller 2021-04-10T20:26:23.424400Z

So you could just walk the (System/getProperty "java.class.path") while it's directories. To be more righter, could probably only take the paths in your project (local or git deps could also be dirs and happen to be at the front)

👍 1
2021-04-10T20:30:34.424600Z

Thanks, I'll give the classpath approach a go.

seancorfield 2021-04-10T20:40:36.424900Z

Why are you using c.t.n / reloaded? I've never felt the need for that sort of thing (and my REPLs run for weeks, sometimes months).

seancorfield 2021-04-10T20:41:24.425100Z

I'm always curious when I find folks using this workflow and -- as in your case -- jumping through hoops to make it work...?

alexmiller 2021-04-10T20:53:06.425300Z

(+1)