-
Notifications
You must be signed in to change notification settings - Fork 302
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
all: to what extent should CUE support symlinks? #3300
Comments
In my case—the basis for the aforementioned #3299—I'm not creating these symbolic links deliberately in my primary source repository. Rather, I'm using my seh/rules_cue Bazel ruleset to evaluate the CUE files within Bazel's actions, and Bazel's normal sandbox preparation behavior involves creating a directory tree full of symbolic links to regular files sitting elsewhere, as an optimization against copying all of those files. |
@seh - thanks for adding this context, extremely useful. |
As a data point, I checked the
|
Regarding support for Bazel's sandboxes, it wouldn't be much trouble to require that callers of the cue tool pass a command-line flag to enable tolerance for symbolic links, if you didn't feel that it was safe or fair to use that as the default behavior. If there was such a command-line flag, I'd have my Bazel ruleset pass it to the cue tool when invoking it inside of Bazel actions. |
Perhaps I jumped the gun by approving the patch for #3298, which makes the loader follow symlinks locally. The way that issue was raised before this one, in the format of a bug report, didn't make it clear that it was a slightly open question. Given that Go does this locally, doing the same seems fine to me. But given that symlinks cannot be published or consumed via a GOPROXY (nor a CUE registry), I think it would also be fine to leave them as unsupported in a consistent way. |
Making a decision here should also close #1672, hopefully. Note that some people in that thread were wondering about symlink support for loading local packages, but it's not clear to me that any of them actually relied on it in the past. |
With the caveat that we would then need to understand/answer how things could/should work with Bazel. @seh how familiar are you with what Go does with respect to Bazel? Anything we should be learning from there? |
I've been relying on it since first using CUE with Bazel three and a half years ago. |
As far as I can tell from inspecting the directory trees created by the rules_go ruleset when I build Go programs with Bazel, every file presented to the Go toolchain is a symbolic link. If we happen to have Bazel generate any files used as input to compiling or linking action, that artifact might wind up as a regular file sitting within that same sandbox's directory tree, but not necessarily. |
My sense is that we keep things working as-is, by first tightening up what we mean by "as-is", and adding any missing tests. So here is an attempt to define what "as-is" is/should be:
That second part of the sentence is worth us making sure with a test. Because it's possible to construct a "rogue" zip file that does include symlinks using the regular Unix Note that the rule would also apply in the case of:
As such, an embed "read" will only ever happen within the CUE module cache for a zip-based dependency. That way we can say that if the CUE code did not come from a zip archive (working with a main module on disk, a directory-based
|
When we have those tests, we should also verify how older CUE versions behaved, for example v0.5.0. |
Following my comment in #3299 (comment), I have updated my comment above to state clearly that:
This brings CUE inline with Go's behaviour with respect to @seh - please could I ask you to take a look at my conclusion above and raise any concerns you might have from a Bazel perspective? |
The treatment of files in modules proposed above sounds like it would work fine with Bazel's sandbox technique, at least for files in modules that come from the same Bazel workspace. That would cover my use of modules thus far. The restriction on I can imagine a softer but more complicated rule, something like: If we read the CUE file containing the |
This adds explicit tests for the behavior of CUE when it encounters symbolic links. Namely: - symbolic links are followed for files that are directly part of the build. - symbolic links are disallowed for embedded files. - symbolic links are silently dropped when publishing modules. Modulo the fact that modules did not exist previously, this matches previous behavior (checked against v0.4.3): the non-module parts of `symlink.txtar` test pass against v0.4.3. For #3300. Signed-off-by: Roger Peppe <[email protected]> Change-Id: Iaf6c4668101f3e91d78a8cab5734efea0a2a9624 Reviewed-on: https://review.gerrithub.io/c/cue-lang/cue/+/1198252 TryBot-Result: CUEcueckoo <[email protected]> Reviewed-by: Daniel Martí <[email protected]> Unity-Result: CUE porcuepine <[email protected]>
Given that Go seems to have exactly the same rule, do you know how Go manages to avoid that difficulty, by any chance? |
In the rules_go Bazel ruleset, both the rules_go prepares a so-called embedcfg file that it then supplies to the Go compiler via the There's a lot of bundling up the files to be embedded and figuring out their relative paths. It's hard to follow what's going on, but I think it's mainly about working around the restriction on following symbolic links. |
With the tests from https://review.gerrithub.io/c/cue-lang/cue/+/1198252 merged, I'm pushing the rest of the work here to alpha.3, as we are releasing alpha.2 today. |
That's all super helpful, thanks! I'm totally unfamiliar with Bazel so it's hard for me to know how much of this applies only to Go and how much might apply to CUE too, but it seems like we will probably need a similar mechanism to Go's |
Bazel will create symbolic links to all the files nominated as "inputs" for an action that it's going to run, regardless of where the source files sit relative to each other either in the primary repository or as generated outputs from other actions. So long as we mention that the embedded files need to be inputs to the action that runs cue def or cue export, we'll get symbolic links to them in the sandbox that Bazel creates for the action. The reason that rules_go wants the embedded source files specified in a separate attribute on the |
Introduce version 0.10.0, establishing it as the new default, removing the prior versions 0.9.0, 0.9.1, and 0.9.2, as none of them could accommodate symbolic links within imported modules that most of Bazel's sandboxing strategies require. Pertinent URLs to related discussion: - https://cuelang.slack.com/archives/C063DSMKS56/p1720620420965109 - cue-lang/cue#3300
Introduce version 0.10.0, establishing it as the new default, removing the prior versions 0.9.0, 0.9.1, and 0.9.2, as none of them could accommodate symbolic links within imported modules that most of Bazel's sandboxing strategies require. Pertinent URLs to related discussion: - https://cuelang.slack.com/archives/C063DSMKS56/p1720620420965109 - cue-lang/cue#3300
I've retitled this issue to the much broader "to what extent should CUE support symlinks?" With a view to this issue being closed with a holistic explanation of where CUE does/doesn't support symlinks, along with brief rationale for each relevant point. @rogpeppe is going to follow up with that summary. We will also create further issues as needs be to capture outstanding bugs/work. e.g. if our aim is to support symlinks in such a way that things work for Bazel, we should have tests that lock that in. |
@seh Do I have it right that you're happy with CUE's current behaviour with respect to symbolic links? |
From what I've read, it sounds workable for use with Bazel. I'd like to try an example involving a mixture of static source and generated files that I can have Bazel run through cue export. |
For the record, the executive summary of current (and proposed future) behaviour with respect to symlinks is as follows: We allow and respect symbolic links except that we reject/ignore them whenever they become the contents of a module in a registry. As such, symbolic links will not be unarchived from the contents of a module archive and will not appear in the module cache. |
Great. Thanks, @rogpeppe. This I think concludes the discussion. We follow up on a possible FAQ for detailed points implied by this conclusion in cue-lang/docs-and-content#187. Publishing such an FAQ to cuelang.org will allow for the more easy surfacing of answers/results. FYI @jpluscplusm |
Does this mean that symbolic links are only supported in local modules, but not registry modules? This seems reasonable, I'm just a bit confused by the phrasing. 🙏 |
@infogulch correct. https://cuelang.org/docs/reference/modules/ says:
so any symbolic links present are ignored when using |
We recently published https://cuelang.org/docs/concept/faq/symbolic-link-support/ - does its wording feel less confusing and more useful to you, @infogulch? |
Thanks for clarifying. Yes the docs are better. Maybe it would be helpful to have a name to positively identify and distinguish the two different kinds/sources of modules -- like "registry module" vs "local module" -- then defining the behavior by reference to these names. I suspect an explanation of this form would be faster to grok and make it easier to derive the answers to such FAQ for oneself. Is this the only difference between "local" and "remote" modules? If there are more features that distinguish them then maybe it would be worth naming. |
"Inspired" by #3299, I don't think we have been explicit about the extent to which modules support (or otherwise) symbolic links.
Questions not limited to:
This issue might generate further issues for bugs/implementation.
End goal is that we should be clear in our documentation (exact location TBC) on the extent to which module loading does/doesn't support symbolic links.
See also #3298
The text was updated successfully, but these errors were encountered: