一些零碎知识

线上服务在特定情况下批量 core dump,最后发现是因为 Systemd 默认会限制 Service 的最大线程数(参见 TasksMax 参数)。


GoLang 默认的内存分配以 32K 为界,小于该值的内存分配从线程本地缓存中获取,大于该值的内存分配会直接走全局共享的中心缓存,有锁竞争的开销。


Linux 上实时展示日志新增行数可以用 pv 命令,如:

tail -f /var/log/nginx/access.log | grep 'your_domain' | pv -l -r >/dev/null

某天发现线上 NGINX 大量出现 too much open files 的错误,但系统配置的 ulimit 已经很大(4M),最后发现是 NGINX 还有一个 worker_rlimit_nofile 的配置项,需要单独设置。


在 MacOS 上,GraphicsMagickWand-config --ldflags 输出的信息补全,使用 Rust 的绑定库 graphicsmagick-sys 时会因为无法找到相关动态链接库而报错,可以在 Cargo Config 中添加以下配置临时绕过(前提是依赖库都使用 HomeBrew 安装)。

[target.aarch64-apple-darwin]
rustflags = ["-L", "/opt/homebrew/lib"]

dmesg -T 的时间可能有一定偏差,因为它对应的是 CPU 时钟,而不是墙上时间。可以使用 journalctl -k 查看,因为它展示的是其读取到内核日志时的系统时间。


目前(2022年6月)WSL2 不能直接安装 ArchLinux,可以用 Distrod 安装,甚至还能使用 systemd,真香。


常见的一致性哈希算法存在一定的不均匀性,可以通过影子节点的方式缓解,但与此同时也会降低性能。谷歌 2014 年发布的跳跃一致性 Hash 算法可以解决不均匀的问题,但原始算法又无法支持非尾部节点增删时的少迁移性。

相关文章:https://writings.sh/post/consistent-hashing-algorithms-part-1-the-problem-and-the-concept


NGINX 默认使用单 listen socket, 多 worker process 的模型,但是部分情况下,EPOLL 会表现出 LIFO 的特性,这可能会导致 worker 进程负载不均衡。

详细信息:https://blog.cloudflare.com/the-sad-state-of-linux-socket-balancing


想对历史 commit 做一些小修改,又不希望多加一个 commit,可以通过以下命令改写:

git add -u
git commit --fixup $TARGET_COMMIT
export EDITOR=true # 可选,避免跳出交互式窗口
git rebase -i --autostash --autosquash $TARGET_COMMIT

也可以配置以下 alias 到 .gitconfig 方便日常使用:

[alias]
    fixup = "!f() { TARGET=$(git rev-parse \"$1\"); shift; git commit --fixup=$TARGET ${@} && GIT_EDITOR=true git rebase -i --autostash --autosquash $TARGET^; }; f"

相关解释:https://ttys3.dev/post/git-fixup-amend-for-any-older-commits-quickly


ImageMagick 部分操作需要产生随机数,默认会在支持 mkstemp 的环境,将 mkstemp 生成的文件名作为随机数熵的一部分,对于服务端常态运行的场景,会产生大量文件系统读写,有一定的性能影响,可以通过条件编译排除这个特性。

相关讨论:https://github.com/ImageMagick/ImageMagick/discussions/2783


相比 Go 等语言,Rust 编译速度很慢,对于本地开发,可以复用编译缓存,一定程度上缓解这个问题; 但是对于 CI/CD ,一般都是在独立环境中运行,往往不会包含编译缓存。

如果你的 CI/CD Job 在 Docker 环境执行,可以通过 cargo-chef 烘焙依赖项缓存到 Docker 镜像层中,减少构建时间,大概流程如下。