123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252 |
- diff --git a/completers/nix-build_completer/cmd/root.go b/completers/nix-build_completer/cmd/root.go
- new file mode 100644
- index 00000000..544f0c3c
- --- /dev/null
- +++ b/completers/nix-build_completer/cmd/root.go
- @@ -0,0 +1,43 @@
- +package cmd
- +
- +import (
- + "github.com/rsteube/carapace"
- + "github.com/rsteube/carapace-bin/pkg/actions/tools/nix"
- + "github.com/spf13/cobra"
- +)
- +
- +var rootCmd = &cobra.Command{
- + Use: "nix-build",
- + Short: "build a Nix expression",
- + Long: "https://nixos.org/manual/nix/stable/command-ref/nix-build.html",
- + Run: func(cmd *cobra.Command, args []string) {},
- +}
- +
- +func Execute() error {
- + return rootCmd.Execute()
- +}
- +func init() {
- + carapace.Gen(rootCmd).Standalone()
- +
- + rootCmd.Flags().StringP("attr", "A", "", "Attribute to build")
- + rootCmd.Flags().Bool("dry-run", false, "Show what store paths would be built or downloaded")
- + rootCmd.Flags().StringP("include", "I", "", "Include paths")
- + rootCmd.Flags().Bool("no-out-link", false, "Do not create a symlink to the output path")
- + rootCmd.Flags().StringP("out-link", "o", "", "Change the name of the symlink to the output path")
- + // TODO support --arg and --argstr
- +
- + carapace.Gen(rootCmd).FlagCompletion(carapace.ActionMap{
- + "attr": carapace.ActionCallback(func(c carapace.Context) carapace.Action {
- + opts := nix.AttributeOpts{Source: "default.nix", Include: rootCmd.Flag("include").Value.String()}
- + if len(c.Args) > 0 {
- + opts.Source = c.Args[0]
- + }
- + return nix.ActionAttributes(opts)
- + }),
- + })
- +
- + carapace.Gen(rootCmd).PositionalAnyCompletion(
- + nix.ActionPaths(),
- + )
- +
- +}
- diff --git a/completers/nix-build_completer/main.go b/completers/nix-build_completer/main.go
- new file mode 100644
- index 00000000..a733fad2
- --- /dev/null
- +++ b/completers/nix-build_completer/main.go
- @@ -0,0 +1,7 @@
- +package main
- +
- +import "github.com/rsteube/carapace-bin/completers/nix-build_completer/cmd"
- +
- +func main() {
- + cmd.Execute()
- +}
- diff --git a/completers/nix-channel_completer/cmd/root.go b/completers/nix-channel_completer/cmd/root.go
- index 4da5434f..1fd1ba5d 100644
- --- a/completers/nix-channel_completer/cmd/root.go
- +++ b/completers/nix-channel_completer/cmd/root.go
- @@ -34,7 +34,7 @@ func init() {
- } else if rootCmd.Flag("add").Changed {
- switch len(c.Args) {
- case 0:
- - return nix.ActionRemoteChannels()
- + return nix.ActionRemoteChannels().Invoke(c).Prefix("https://nixos.org/channels/").ToA()
- case 1:
- return nix.ActionLocalChannels()
- }
- diff --git a/pkg/actions/tools/nix/attribute.go b/pkg/actions/tools/nix/attribute.go
- new file mode 100644
- index 00000000..188d0820
- --- /dev/null
- +++ b/pkg/actions/tools/nix/attribute.go
- @@ -0,0 +1,89 @@
- +package nix
- +
- +import (
- + _ "embed"
- + "encoding/json"
- + "fmt"
- + "path"
- + "strings"
- +
- + "github.com/rsteube/carapace"
- +)
- +
- +//go:embed attributeComplete.nix
- +var attributeCompleteScript string
- +
- +type AttributeOpts struct {
- + Source string
- + Include string
- +}
- +
- +// ActionAttributes completes attributes
- +//
- +// firefox
- +// git
- +func ActionAttributes(opts AttributeOpts) carapace.Action {
- + return carapace.ActionCallback(func(c carapace.Context) carapace.Action {
- + if opts.Source == "" {
- + opts.Source = "default.nix"
- + }
- +
- + attrPath := c.CallbackValue
- +
- + if strings.HasPrefix(opts.Source, "<") {
- + // expression is a local channel, use as-is
- + } else if strings.HasPrefix(opts.Source, "http:") || strings.HasPrefix(opts.Source, "https:") {
- + // expression is a url, wrap in fetchTarball
- + opts.Source = fmt.Sprintf("(fetchTarball %s)", opts.Source)
- + } else if strings.HasPrefix(opts.Source, "channel:") {
- + // expression is a channel alias, convert to url
- + channelName := opts.Source[8:]
- + opts.Source = fmt.Sprintf("(fetchTarball https://nixos.org/channels/%s/nixexprs.tar.xz)", channelName)
- + } else {
- + // expression is a local file
- + filePath := path.Clean(opts.Source)
- + if path.Base(filePath) == filePath {
- + // paths must have at least one `/`
- + filePath = fmt.Sprintf("./%s", filePath)
- + }
- + opts.Source = filePath
- + }
- + expression := fmt.Sprintf("import %s", opts.Source)
- +
- + // nix itself handles quotes weirdly so we just avoid them altogether
- + if strings.Contains(attrPath, "\"") {
- + return carapace.ActionMessage("attrPath may not contain double-quotes")
- + }
- +
- + // strip right-most attr
- + lastDotIndex := strings.LastIndex(attrPath, ".")
- + if lastDotIndex >= 0 {
- + attrPath = attrPath[0:lastDotIndex]
- + } else {
- + attrPath = ""
- + }
- +
- + args := []string{
- + "--eval", "--json",
- + "--expr", attributeCompleteScript,
- + "--arg", "__carapaceInput__", expression,
- + "--argstr", "__carapaceAttrPath__", attrPath,
- + }
- + if opts.Include != "" {
- + args = append(args, "-I", opts.Include)
- + }
- + // TODO handle passing through --arg and --argstr from original command
- +
- + return carapace.ActionExecCommand("nix-instantiate", args...)(func(output []byte) carapace.Action {
- + var data []string
- + json.Unmarshal(output, &data)
- +
- + prefix := ""
- + if attrPath != "" {
- + prefix = fmt.Sprintf("%s.", attrPath)
- + }
- +
- + return carapace.ActionValues(data...).Prefix(prefix).NoSpace()
- + })
- + })
- +}
- diff --git a/pkg/actions/tools/nix/attributeComplete.nix b/pkg/actions/tools/nix/attributeComplete.nix
- new file mode 100644
- index 00000000..4f98d858
- --- /dev/null
- +++ b/pkg/actions/tools/nix/attributeComplete.nix
- @@ -0,0 +1,25 @@
- +all@{ __carapaceInput__, __carapaceAttrPath__, ... }:
- +let
- + inputArgs = builtins.removeAttrs all ["__carapaceInput__" "__carapaceAttrPath__"];
- +
- + input =
- + if builtins.isFunction __carapaceInput__ then
- + __carapaceInput__ inputArgs
- + else
- + __carapaceInput__;
- +
- + autocall = fnOrAttrset:
- + if builtins.isFunction fnOrAttrset then
- + fnOrAttrset {}
- + else
- + fnOrAttrset;
- +
- + paths = builtins.filter (path: (builtins.isString path) && path != "") (builtins.split "\\." __carapaceAttrPath__);
- +
- + reducer = attrset: name: autocall (builtins.getAttr name attrset);
- + result = builtins.foldl' reducer input paths;
- +in
- + if builtins.isAttrs result then
- + builtins.attrNames result
- + else
- + []
- diff --git a/pkg/actions/tools/nix/channel.go b/pkg/actions/tools/nix/channel.go
- index 930da5a9..62c5ffd4 100644
- --- a/pkg/actions/tools/nix/channel.go
- +++ b/pkg/actions/tools/nix/channel.go
- @@ -63,8 +63,6 @@ func ActionRemoteChannels() carapace.Action {
- for _, result := range response.Data.Result {
- vals = append(vals, result.Metric.Channel, result.Metric.Status, styleForStatus(result.Metric.Status))
- }
- - return carapace.ActionCallback(func(c carapace.Context) carapace.Action {
- - return carapace.ActionStyledValuesDescribed(vals...).Invoke(c).Prefix("https://nixos.org/channels/").ToA()
- - })
- + return carapace.ActionStyledValuesDescribed(vals...)
- })
- }
- diff --git a/pkg/actions/tools/nix/path.go b/pkg/actions/tools/nix/path.go
- new file mode 100644
- index 00000000..1bacd46c
- --- /dev/null
- +++ b/pkg/actions/tools/nix/path.go
- @@ -0,0 +1,31 @@
- +package nix
- +
- +import (
- + "strings"
- +
- + "github.com/rsteube/carapace"
- +)
- +
- +// ActionPaths completes paths
- +//
- +// A path can be one of:
- +// - a local file path (default.nix)
- +// - an http/https URL (https://releases.nixos.org/../nixexprs.tar.xz
- +// - a channel: specifier (channel:nixos-22.05)
- +// - a local channel (<nixpkgs>)
- +func ActionPaths() carapace.Action {
- + return carapace.ActionCallback(func(c carapace.Context) carapace.Action {
- + batch := carapace.Batch(
- + ActionLocalChannels().Prefix("<").Suffix(">"),
- + carapace.ActionFiles(),
- + )
- +
- + if !strings.HasPrefix(c.CallbackValue, "channel:") {
- + batch = append(batch, carapace.ActionValues("channel:").NoSpace())
- + } else {
- + batch = append(batch, ActionRemoteChannels().Prefix("channel:"))
- + }
- +
- + return batch.ToA()
- + })
- +}
|