-
Notifications
You must be signed in to change notification settings - Fork 370
Spec for init scripts and sandbox updates
opam has a series of embedded shell scripts and shell commands which are used in three contexts as part of:
-
opam init
to set-up automatic switch detection based on CWD and command completion -
opam env
to set-up the shell environment -
opam install
to set-up the sandbox
Updating these requires updating every supported version of opam. The present mechanism is a little cumbersome (requiring the user to opam init --reinit
) and can also result in unusable opam releases (e.g. from fish syntax removal).
This spec proposes moving the scripts and shell data to a separate repository which can be updated and released separately, allowing alterations to the sandbox and fixes to shell commands without having to re-release opam.
src/state/opamEnv.ml
and src/client/opamInitDefaults.ml
are where these scripts are presently used from.
A new root branch scripts
is added to ocaml/opam. Commits to this branch should be signed with the same GPG key as opam binaries. The branch should have linear history (i.e. no merge commits).
opam init
gains a new --init-scripts <giturl>[#<branch>]
parameter whose default value is https://github.com/ocaml/opam.git#scripts
. If no is given then main
is assumed (NOT the default branch of the repo). Two fields are added to .opam/config
:
scripts-repository: "https://github.com/ocaml/opam.git"
scripts-branch: "scripts"
Two new options are added to both opam init
and opam update
: --fetch-scripts
and --no-fetch-scripts
(the default is --fetch-scripts
) and additionally --update-scripts
and --no-update-scripts
to opam update
(the default is --update-scripts
). The opam binary at release time will contain an embedded git bundle of the scripts
branch.
At opam init
, as long as the --init-scripts
parameter was either omitted or is (https://github.com/|git://github.com/|ssh://[email protected]/|[email protected]:)ocaml/opam[.git]#scripts
, then the internal bundle is cloned to ./opam/opam-init/scripts
. Otherwise, the supplied repository URL is cloned. Note that --no-fetch-scripts
conflicts --init-scripts
if the <giturl>
is not the same as the git-bundle's. The cloned repository is referenced as upstream
in the cloned repo (and will always point to <giturl>
not the bundle) and the branch is called main
. If the bundle is extracted, and --no-fetch-scripts
was not specified, then upstream
is pulled.
At opam update
, unless --no-fetch-scripts
is given, upstream
is fetched. If the .opam/opam-init/scripts
contains untracked or uncommitted changes or the branch main
is not checked out then opam update
emits a warning and does not fetch or update the repo. After fetching, if there are new commits and --no-update-scripts
was not given, opam update
either:
- fast-forwards
main
toupstream/<branch>
, if the HEAD ifmain
is a commit onupstream/<branch>
and informs the user that the scripts were updated. - attempts an automatic rebase of
main
ontoupstream/<branch>
. If this succeeds, and custom commits remain, then the user is informed that the scripts were updated and custom patches have been retained. If this fails, thenmain
is left unaltered and the user is warned that the upstream scripts have been updated but the patches made to them do not rebase cleanly.
Note that both processes may cause the variables
and init
scripts to be regenerated. opam will display the first line of each commit messages of the new commits which have been adopted.
opamrc
's init-scripts
field contains the actual sandbox scripts. The default will be updated to:
init-scripts: [
[
"sandbox.sh"
"""\
#!/usr/bin/env bash
. "$(dirname "$0")/scripts/sandboxes-v1/bwrap.sh"
"""
] {os = "linux"}
[
"sandbox.sh"
"""\
#!/usr/bin/env bash
. "$(dirname "$0")/scripts/sandboxes-v1/sandbox-exec.sh"
"""
] {os = "macos"}
]
and the two scripts presently in src/state/shellscripts/
get moved to sandboxes-v1/
in the new scripts
branch.
Note: versioning is achieved here by changing the name of sandbox scripts and updating the path in opamrc
to point to it.
The shells are a little more involved. The existing env_hook
and complete
scripts get moved from src/state/shellscripts/
to env/
and complete/
in the new scripts
branch.
The root of the scripts branch then contains a single config
file:
opam-version: "2.2"
shells: [
"bash" {opam-version >= "4.0"}
"sh"
"csh"
"zsh"
"fish" {os = "macos"}
]
shell "bash" {
command: "bash"
profile: [ ".bash_profile" ".bash_login" ".profile" ".bash_profile" ]
profile-message: "Make sure that ~/%{profile}% is well sourced in your ~/.bashrc" {profile != ".bashrc}
eval: "eval $(%{cmd}%)"
variables: "variables.sh"
init: "init.sh"
complete: [
"complete/2.0-complete.sh" {opam-version = "2.0"}
"complete/2.1-complete.sh" {opam-version = "2.1"}
]
env-hook: "env/hook.sh"
comment: "# "
export: "%{name}%=%{value}; export %{name}%;"
source: "test -r %{name}% && . %{name}% > /dev/null 2> /dev/null || true"
tty: [ "if [ -t 0 ]; then" "else" "fi" ]
env_updates: [
"%{single-quote-value}%" # =
"%{single-quote-value}%:\"$%{name}%\"" # += / := / =+=
"\"$%{name}%\":%{single-quote-value}%" # =: =+
]
}
shell "sh" {
command: "sh"
profile: ".profile"
eval: "eval $(%{cmd}%)"
init: "init.sh"
complete: "complete/complete.sh"
env-hook: "env/hook.sh"
comment: "# "
export: "%{name}%=%{value}; export %{name}%;"
source: "test -r %{name}% && . %{name}% > /dev/null 2> /dev/null || true"
tty: [ "if [ -t 0 ]; then" "else" "fi" ]
env_updates: [
"%{single-quote-value}%" # =
"%{single-quote-value}%:\"$%{name}%\"" # += / := / =+=
"\"$%{name}%\":%{single-quote-value}%" # =: =+
]
}
shell "csh" {
command: "csh"
aliases: [ "tcsh" "bsd-csh" ]
profile: [ ".cshrc" ".tcshrc" ]
eval: "eval `%{cmd}%`"
init: "init.csh"
env-hook: "env/hook.csh"
comment: "# "
export: """\
if ( ! ${?%{name}%} ) setenv %{name}% ""
setenv %{name}% %{value}%
"""
source: "if ( -f %{name}% ) source %{name}% >& /dev/null"
tty: [ "if ( $?prompt ) then" "else" "endif" ]
env_updates: [
"%{single-quote-value}%" # =
"%{single-quote-value}%:\"$%{name}%\"" # += / := / =+=
"\"$%{name}%\":%{single-quote-value}%" # =: =+
]
}
shell "zsh" {
command: "zsh"
profile: ".zshrc"
eval: "eval $(%{cmd}%)"
variables: "variables.sh"
init: "init.zsh"
complete: "complete/complete.zsh"
env-hook: "env/hook.zsh"
comment: "# "
export: "%{name}%=%{value}; export %{name}%;"
source: "[[ ! -r %{name}% ]] || source %{name}% > /dev/null 2> /dev/null"
tty: [ "if [[ -o interactive ]] then" "else" "fi" ]
env_updates: [
"%{single-quote-value}%" # =
"%{single-quote-value}%:\"$%{name}%\"" # += / := / =+=
"\"$%{name}%\":%{single-quote-value}%" # =: =+
]
}
shell "fish" {
command: "fish"
profile: ".config/fish/config.fish"
eval: "eval (%{cmd}%)"
init: "init.fish"
env-hook: "env/hook.fish"
comment: "# "
export: [
"set -gx %{name}% %{fish-array-value}%;" {name = "PATH"}
"if [ (count $MANPATH) -gt 0 ]; set -gx MANPATH %{fish-array-value}%; end;" {name = "MANPATH"}
"set -gx %{name}% %{value}%"
]
source: "source %{name}% > /dev/null 2> /dev/null; or true"
tty: [ "if isatty" "else" "end" ]
env_updates: [
"%{fish-single-quote-value}%" # =
"%{fish-single-quote-value}%:\"$%{name}%\"" # += / := / =+=
"\"$%{name}%\":%{fish-single-quote-value}%" # =: =+
]
}