carapace-nix-build.patch 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. commit 29cf307f93c19325ff09f0e324624abd4f6bb7ba
  2. Author: Thomas Dy <thatsmydoing@gmail.com>
  3. Date: Sun Jan 15 18:05:45 2023 +0900
  4. nix-build: add initial support
  5. diff --git a/completers/nix-build_completer/cmd/root.go b/completers/nix-build_completer/cmd/root.go
  6. new file mode 100644
  7. index 00000000..65381edd
  8. --- /dev/null
  9. +++ b/completers/nix-build_completer/cmd/root.go
  10. @@ -0,0 +1,39 @@
  11. +package cmd
  12. +
  13. +import (
  14. + "github.com/rsteube/carapace"
  15. + "github.com/rsteube/carapace-bin/pkg/actions/tools/nix"
  16. + "github.com/spf13/cobra"
  17. +)
  18. +
  19. +var rootCmd = &cobra.Command{
  20. + Use: "nix-build",
  21. + Short: "build a Nix expression",
  22. + Long: "https://nixos.org/manual/nix/stable/command-ref/nix-build.html",
  23. + Run: func(cmd *cobra.Command, args []string) {},
  24. +}
  25. +
  26. +func Execute() error {
  27. + return rootCmd.Execute()
  28. +}
  29. +func init() {
  30. + carapace.Gen(rootCmd).Standalone()
  31. +
  32. + rootCmd.Flags().StringP("include", "I", "", "Include paths")
  33. + rootCmd.Flags().Bool("no-out-link", false, "Do not create a symlink to the output path")
  34. + rootCmd.Flags().StringP("out-link", "o", "", "Change the name of the symlink to the output path")
  35. + rootCmd.Flags().Bool("dry-run", false, "Show what store paths would be built or downloaded")
  36. + rootCmd.Flags().StringP("attr", "A", "", "Attribute to build")
  37. + // TODO support --arg and --argstr
  38. +
  39. + carapace.Gen(rootCmd).PositionalAnyCompletion(nix.ActionPaths())
  40. + carapace.Gen(rootCmd).FlagCompletion(carapace.ActionMap{
  41. + "attr": nix.ActionAttr(rootCmd, func(_ *cobra.Command, c carapace.Context) string {
  42. + if len(c.Args) > 0 {
  43. + return c.Args[0]
  44. + } else {
  45. + return "default.nix"
  46. + }
  47. + }),
  48. + })
  49. +}
  50. diff --git a/completers/nix-build_completer/main.go b/completers/nix-build_completer/main.go
  51. new file mode 100644
  52. index 00000000..a733fad2
  53. --- /dev/null
  54. +++ b/completers/nix-build_completer/main.go
  55. @@ -0,0 +1,7 @@
  56. +package main
  57. +
  58. +import "github.com/rsteube/carapace-bin/completers/nix-build_completer/cmd"
  59. +
  60. +func main() {
  61. + cmd.Execute()
  62. +}
  63. diff --git a/pkg/actions/tools/nix/attr.go b/pkg/actions/tools/nix/attr.go
  64. new file mode 100644
  65. index 00000000..921bf2b6
  66. --- /dev/null
  67. +++ b/pkg/actions/tools/nix/attr.go
  68. @@ -0,0 +1,82 @@
  69. +package nix
  70. +
  71. +import (
  72. + _ "embed"
  73. + "encoding/json"
  74. + "fmt"
  75. + "path"
  76. + "strings"
  77. +
  78. + "github.com/rsteube/carapace"
  79. + "github.com/spf13/cobra"
  80. +)
  81. +
  82. +//go:embed attrComplete.nix
  83. +var attrCompleteScript string
  84. +
  85. +func ActionAttr(cmd *cobra.Command, getSource func(*cobra.Command, carapace.Context) string) carapace.Action {
  86. + return carapace.ActionCallback(func(c carapace.Context) carapace.Action {
  87. + attrPath := c.CallbackValue
  88. + nixPath := ""
  89. + if cmd.Flag("include").Changed {
  90. + nixPath = cmd.Flag("include").Value.String()
  91. + }
  92. +
  93. + source := getSource(cmd, c)
  94. +
  95. + if strings.HasPrefix(source, "<") {
  96. + // expression is a local channel, use as-is
  97. + } else if strings.HasPrefix(source, "http:") || strings.HasPrefix(source, "https:") {
  98. + // expression is a url, wrap in fetchTarball
  99. + source = fmt.Sprintf("(fetchTarball %s)", source)
  100. + } else if strings.HasPrefix(source, "channel:") {
  101. + // expression is a channel alias, convert to url
  102. + channelName := source[8:]
  103. + source = fmt.Sprintf("(fetchTarball https://nixos.org/channels/%s/nixexprs.tar.xz)", channelName)
  104. + } else {
  105. + // expression is a local file
  106. + filePath := path.Clean(source)
  107. + if path.Base(filePath) == filePath {
  108. + // paths must have at least one `/`
  109. + filePath = fmt.Sprintf("./%s", filePath)
  110. + }
  111. + source = filePath
  112. + }
  113. + expression := fmt.Sprintf("import %s", source)
  114. +
  115. + // nix itself handles quotes weirdly so we just avoid them altogether
  116. + if strings.Contains(attrPath, "\"") {
  117. + return carapace.ActionMessage("attrPath may not contain double-quotes")
  118. + }
  119. +
  120. + // strip right-most attr
  121. + lastDotIndex := strings.LastIndex(attrPath, ".")
  122. + if lastDotIndex >= 0 {
  123. + attrPath = attrPath[0:lastDotIndex]
  124. + } else {
  125. + attrPath = ""
  126. + }
  127. +
  128. + args := []string{
  129. + "--eval", "--json",
  130. + "--expr", attrCompleteScript,
  131. + "--arg", "__carapaceInput__", expression,
  132. + "--argstr", "__carapaceAttrPath__", attrPath,
  133. + }
  134. + if nixPath != "" {
  135. + args = append(args, "-I", nixPath)
  136. + }
  137. + // TODO handle passing through --arg and --argstr from original command
  138. +
  139. + return carapace.ActionExecCommand("nix-instantiate", args...)(func(output []byte) carapace.Action {
  140. + var data []string
  141. + json.Unmarshal(output, &data)
  142. + if attrPath != "" {
  143. + for i, value := range data {
  144. + data[i] = fmt.Sprintf("%s.%s", attrPath, value)
  145. + }
  146. + }
  147. + return carapace.ActionValues(data...)
  148. + })
  149. + })
  150. +}
  151. diff --git a/pkg/actions/tools/nix/attrComplete.nix b/pkg/actions/tools/nix/attrComplete.nix
  152. new file mode 100644
  153. index 00000000..4f98d858
  154. --- /dev/null
  155. +++ b/pkg/actions/tools/nix/attrComplete.nix
  156. @@ -0,0 +1,25 @@
  157. +all@{ __carapaceInput__, __carapaceAttrPath__, ... }:
  158. +let
  159. + inputArgs = builtins.removeAttrs all ["__carapaceInput__" "__carapaceAttrPath__"];
  160. +
  161. + input =
  162. + if builtins.isFunction __carapaceInput__ then
  163. + __carapaceInput__ inputArgs
  164. + else
  165. + __carapaceInput__;
  166. +
  167. + autocall = fnOrAttrset:
  168. + if builtins.isFunction fnOrAttrset then
  169. + fnOrAttrset {}
  170. + else
  171. + fnOrAttrset;
  172. +
  173. + paths = builtins.filter (path: (builtins.isString path) && path != "") (builtins.split "\\." __carapaceAttrPath__);
  174. +
  175. + reducer = attrset: name: autocall (builtins.getAttr name attrset);
  176. + result = builtins.foldl' reducer input paths;
  177. +in
  178. + if builtins.isAttrs result then
  179. + builtins.attrNames result
  180. + else
  181. + []
  182. diff --git a/pkg/actions/tools/nix/path.go b/pkg/actions/tools/nix/path.go
  183. new file mode 100644
  184. index 00000000..e3dbc8b2
  185. --- /dev/null
  186. +++ b/pkg/actions/tools/nix/path.go
  187. @@ -0,0 +1,29 @@
  188. +package nix
  189. +
  190. +import (
  191. + "strings"
  192. +
  193. + "github.com/rsteube/carapace"
  194. +)
  195. +
  196. +// ActionPaths completes paths
  197. +//
  198. +// A path can be one of:
  199. +// - a local file path (default.nix)
  200. +// - an http/https URL (https://releases.nixos.org/../nixexprs.tar.xz
  201. +// - a channel: specifier (channel:nixos-22.05)
  202. +// - a local channel (<nixpkgs>)
  203. +func ActionPaths() carapace.Action {
  204. + return carapace.ActionCallback(func(c carapace.Context) carapace.Action {
  205. + arg := c.CallbackValue
  206. + if strings.HasPrefix(arg, "<") {
  207. + return ActionLocalChannels().Invoke(c).Prefix("<").Suffix(">").ToA()
  208. + } else if strings.HasPrefix(arg, "http:") || strings.HasPrefix(arg, "https:") {
  209. + return carapace.ActionValues()
  210. + } else if strings.HasPrefix(arg, "channel:") {
  211. + return ActionRemoteChannels().Invoke(c).Prefix("channel:").ToA()
  212. + } else {
  213. + return carapace.ActionFiles()
  214. + }
  215. + })
  216. +}