Skip to content
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

[stdlib-candidate] set-env #787

Merged
merged 2 commits into from
Mar 12, 2024
Merged

[stdlib-candidate] set-env #787

merged 2 commits into from
Mar 12, 2024

Conversation

texastoland
Copy link
Contributor

@texastoland texastoland commented Mar 12, 2024

Rewrite of nushell/nushell#12156 for jdx/mise#1763.

Why?

Nushell philosophically omits a set list mutation. But $env is inherently mutable leading to issues described in nushell/nushell#12148. set-env provides such an operation exclusively for $env.

What changed?

  1. Explicit flag instead of implicit list concatenation
  2. Expands updates to any $env field not only $env.config

How is it used?

❯ set-env -h
Gracefully set an environment variable or merge a nested option.

Examples:
  Set $env.NUPM_HOME
  > set-env NUPM_HOME $'($nu.home-path)/.local/share/nupm'

  Add to $env.NU_LIB_DIRS
  > set-env --append NU_LIB_DIRS $'($env.NUPM_HOME)/modules'

  Set a nested config option
  > set-env config.filesize.metric true

  Add a config hook
  > set-env -a config.hooks.pre_prompt 'ellie | print'

Usage:
  > main {flags} <field> <value>

Flags:
  -a, --append - Append to the previous value or wrap in a new list
  -h, --help - Display the help message for this command

Parameters:
  field <cell-path>: The environment variable name or nested option cell path
  value <any>: The value to set or append

Input/output types:
  ╭───┬─────────┬─────────╮# │  input  │ output  │
  ├───┼─────────┼─────────┤
  │ 0 │ nothing │ nothing │
  ╰───┴─────────┴─────────╯

How does it work?

export def --env main [
  field: cell-path
  value: any
  --append (-a)
]: nothing -> nothing {

  # just an alias
  def 'get or' [default field] {
    get --ignore-errors $field | default $default
  }

  let value = if $append {
    # append to the previous value or empty list
    $env | get or [] $field | append $value
  } else {
    $value
  }

  # work around nushell/nushell#12168
  let field = $field | to text | split row .
  let value = match $field {

    [_] => $value
    # if cell path is nested
    [$root, ..$field] => {
      let field = $field | into cell-path

      # reassigning $env would be an error
      # merging reserved names like PWD would be an error
      # so merge from 1 level deep instead
      $env | get or {} $root | upsert $field $value
    }
  }

  # avoid issues noted above
  load-env { ($field | first): $value }
}

Where are the tests?

Pending next PR for nupm integration.

@fdncred
Copy link
Contributor

fdncred commented Mar 12, 2024

Nice work on this! My only minor issue is with the parameter named path which is not a path but a cellpath and could lead to confusion. I'd probably land this with or without that change though because it seems useful.

@texastoland
Copy link
Contributor Author

Thanks! Renamed field like insert/update/upsert but also happy to go with cell_path like get.

@fdncred
Copy link
Contributor

fdncred commented Mar 12, 2024

Thanks.

@fdncred fdncred merged commit 74ba060 into nushell:main Mar 12, 2024
@texastoland texastoland changed the title [stdlib-candidate] set-env [stdlib-candidate] set-env Mar 20, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants