Project Icon

envrc

Emacs缓冲区级direnv环境管理工具

envrc.el是GNU Emacs的一个插件,它利用direnv工具在缓冲区级别管理环境变量。这种方法确保了不同项目间的环境隔离,使开发者能够在多个项目中使用不同版本的工具,而无需担心环境冲突。相比全局环境变量管理,envrc.el提供了更精细和清晰的项目环境控制方式。

Melpa 状态 Melpa 稳定版状态 构建状态 支持我

envrc.el - Emacs的缓冲区本地direnv集成

这是一个GNU Emacs库,它使用direnv工具来确定每个目录/项目的环境变量,然后在每个缓冲区的基础上设置这些环境变量。这意味着当你在多个具有.envrc文件的项目之间工作时,从这些项目的缓冲区中启动的所有进程都将使用这些文件中指定的环境变量执行。这允许在每个项目中使用不同版本的代码检查工具和其他工具(如果需要)。

这与direnv.el有何不同?

direnv.el根据你正在工作的缓冲区,反复更改全局Emacs环境。

相反,envrc.el只是在每个缓冲区中设置和存储正确的环境,作为一个缓冲区本地变量。

从用户的角度来看,两者都经过充分测试,通常运行良好,但对我来说,envrc.el的方法感觉更加简洁。

安装

可安装的包可通过MELPA获得:执行M-x package-install RET envrc RET

或者,下载最新版本或克隆仓库,然后用M-x package-install-file安装envrc.el

使用

在你的init.el底部添加类似以下的代码片段:

(envrc-global-mode)

(add-hook 'after-init-hook 'envrc-global-mode)

或者,如果你是use-package的粉丝:

(use-package envrc
  :hook (after-init . envrc-global-mode))

为什么必须在启动序列的后期启用全局模式?通常,你希望在每个缓冲区中初始化envrc-mode之前初始化其他次要模式,比如flycheck-mode,这些模式可能会寻找可执行文件。与直觉相反,这意味着应该在其他全局次要模式之后启用envrc-global-mode,因为每个模式都会将自己前置到各种钩子中。

只有当direnv已安装并在默认的Emacs exec-path中可用时,全局模式才会生效。(还有一个本地次要模式envrc-mode,但你不应该尝试有选择性地启用它,例如针对某些模式或项目,因为编译和其他缓冲区可能无法获得正确的环境设置。)

关于与该模式的交互,请参见envrc-mode-map,以及命令envrc-reloadenvrc-allowenvrc-deny。(目前还有envrc-reload-all作为"核弹式"重置!)

特别地,你可以通过在envrc-mode-map中将你喜欢的前缀绑定到envrc-command-map来为上述命令启用键绑定,例如:

(with-eval-after-load 'envrc
  (define-key envrc-mode-map (kbd "C-c e") 'envrc-command-map))

故障排除

如果你发现某个特定的Emacs命令没有获取到当前缓冲区的环境,并且你确定envrc-mode在该缓冲区中处于活动状态,那么你可能发现了在临时缓冲区中运行进程但在执行之前忽略了将你的环境传播到该缓冲区的代码。

有几个常见的Emacs命令存在这个缺陷,它们也通过envrc.el中的建议直接进行了修补 —— shell-command-to-string就是一个突出的例子!

inheritenv包被设计用来处理这种一般情况。

设计说明

默认情况下,Emacs有一组用于所有子进程的单一全局环境变量集,存储在process-environment变量中。direnv.el在用户执行某些操作时(如在不同项目的缓冲区之间切换)使用来自direnv的值切换该全局环境。

实际上,这很简单,而且大多数情况下运行得很好。但是有一些怪异之处,对我来说,为了支持每个目录的环境而改变全局环境感觉是错误的。

现在,在Emacs中,我们也可以在缓冲区中本地设置process-environment。如果可以根据各自的.envrc文件在所有缓冲区中正确维护这个值,那么跨多个项目的缓冲区就可以同时"连接"到其对应项目目录的环境。我编写envrc.el就是为了探索这种方法。

envrc.el使用一个全局次要模式(envrc-global-mode)来钩入Emacs创建的几乎每个缓冲区,包括隐藏和临时缓冲区。当发现一个缓冲区"位于"一个由.envrc管理的项目内时,通过运行direnv来设置缓冲区本地的process-environment,其结果也被无限期缓存,以使整体成本不会太高。每个缓冲区都有一个本地次要模式(envrc-mode),带有一个指示器,显示该缓冲区中是否有效的direnv。(钩入每个缓冲区很重要,而不仅仅是那些具有特定主要模式的缓冲区,因为经常使用单独的临时、编译和REPL缓冲区来执行进程。)

这种方法也有一些权衡:

  • *Help*这样的缓冲区将基于最初创建它们的缓冲区的目录启用envrc-mode,而这些缓冲区通常会长期存在。如果你在处理不同项目时从这些缓冲区启动程序,结果可能不符合你的预期。我可能会排除某些模式以最小化混淆,但用户始终需要意识到环境是特定于缓冲区的这一事实。

  • 每次创建缓冲区时都会有(非常小的)开销,而这种情况经常发生。

  • direnv更新不是自动的。direnv.el在切换到访问不同目录文件的缓冲区时重新执行direnv,而envrc-mode会缓存环境,直到用户使用envrc-reload显式刷新它。

总的来说,这种方法在实践中运作良好,感觉比试图战略性地修改全局环境更加简洁。

也有可能有一种方法可以更积极地调用direnv,允许它看到之前获得的DIRENV_*值,从而使其成为一个空操作。


💝 通过Patreon支持这个项目和我的其他开源工作

💼 LinkedIn个人资料

✍ sanityinc.com

🐦 @sanityinc

项目侧边栏1项目侧边栏2
推荐项目
Project Cover

豆包MarsCode

豆包 MarsCode 是一款革命性的编程助手,通过AI技术提供代码补全、单测生成、代码解释和智能问答等功能,支持100+编程语言,与主流编辑器无缝集成,显著提升开发效率和代码质量。

Project Cover

AI写歌

Suno AI是一个革命性的AI音乐创作平台,能在短短30秒内帮助用户创作出一首完整的歌曲。无论是寻找创作灵感还是需要快速制作音乐,Suno AI都是音乐爱好者和专业人士的理想选择。

Project Cover

有言AI

有言平台提供一站式AIGC视频创作解决方案,通过智能技术简化视频制作流程。无论是企业宣传还是个人分享,有言都能帮助用户快速、轻松地制作出专业级别的视频内容。

Project Cover

Kimi

Kimi AI助手提供多语言对话支持,能够阅读和理解用户上传的文件内容,解析网页信息,并结合搜索结果为用户提供详尽的答案。无论是日常咨询还是专业问题,Kimi都能以友好、专业的方式提供帮助。

Project Cover

阿里绘蛙

绘蛙是阿里巴巴集团推出的革命性AI电商营销平台。利用尖端人工智能技术,为商家提供一键生成商品图和营销文案的服务,显著提升内容创作效率和营销效果。适用于淘宝、天猫等电商平台,让商品第一时间被种草。

Project Cover

吐司

探索Tensor.Art平台的独特AI模型,免费访问各种图像生成与AI训练工具,从Stable Diffusion等基础模型开始,轻松实现创新图像生成。体验前沿的AI技术,推动个人和企业的创新发展。

Project Cover

SubCat字幕猫

SubCat字幕猫APP是一款创新的视频播放器,它将改变您观看视频的方式!SubCat结合了先进的人工智能技术,为您提供即时视频字幕翻译,无论是本地视频还是网络流媒体,让您轻松享受各种语言的内容。

Project Cover

美间AI

美间AI创意设计平台,利用前沿AI技术,为设计师和营销人员提供一站式设计解决方案。从智能海报到3D效果图,再到文案生成,美间让创意设计更简单、更高效。

Project Cover

AIWritePaper论文写作

AIWritePaper论文写作是一站式AI论文写作辅助工具,简化了选题、文献检索至论文撰写的整个过程。通过简单设定,平台可快速生成高质量论文大纲和全文,配合图表、参考文献等一应俱全,同时提供开题报告和答辩PPT等增值服务,保障数据安全,有效提升写作效率和论文质量。

投诉举报邮箱: service@vectorlightyear.com
@2024 懂AI·鲁ICP备2024100362号-6·鲁公网安备37021002001498号