Snapcast 多房间音频控制一体化

目录

动机

想要手头的两台 MBP 能同时无延迟播放音频,拒绝使用直播流的形式。

Tip

update at 2025-04-13

这里补充一下在 macOS 中安装 snapserver 并且用这台 mac 的输出作为 snapserver 的输入的方法:

  1. 安装 snapcast: brew install snapcast
  2. 安装 BlackHole: brew install blackhole-2ch 安装完成后建议重启就能看见这 BlackHole 设备
  3. 安装 sox: brew install sox
  4. 运行 snapcast: snapcast 默认就会创建一个管道 /tmp/snapfifo ,如果没有的话需要检查并指定配置文件(默认配置文件 在这里 )或者手动创建管道:mkfifo /tmp/snapfifo
  5. 用 sox 捕获 BlackHole 的音频流并写入命名管道: sox -d -r 48000 -e signed-integer -b 16 -c 2 -t raw /tmp/snapfifo
  6. 将 macOS 的默认的设备输入&输出设备换成 BlackHole 或者创建一个多输出设备(Apple自身+虚拟的BlackHole)
  7. 可以在这台 macOS 上放歌听听,又或者用其他 iOS 设备隔空投放到这台 macOS 听听,多加一些 snapclient,比如安卓客户端呀。
  8. 美中不足的是有杂音。

主旨

Vmware Ubuntu 中安装 snapserver

接受 Airplay 的信号

sudo apt install snapserver

sudo apt install -y shairport-sync
sudo systemctl disable --now shairport-sync
  1. 该虚拟机网络采用桥接方式

  2. 注意该服务会自动启动,你可能需要设置下配置文件位置 sudo vi /lib/systemd/system/snapserver.service

    Untitled

    Untitled

    Untitled

  3. 配置文件 /etc/snapserver.conf 中添加:source = airplay:///shairport-sync?name=Airplay

    [stream]
    source = airplay:///shairport-sync?name=Airplay
    

    Untitled

  4. 重启 sudo service snapserver restart

  5. snapweb 是需要手动添加的,建议从 macos 中复制到该虚拟机,在 mac 上的位置为:

    /usr/local/Cellar/snapcast/0.27.0/share/snapserver

    Untitled

  6. 使用 Tabby 终端连接上虚拟机的 SFTP 上传,上传到配置文件中 doc_root 配置的位置:

    Untitled

    Untitled

macOS 上启动 snapclient

brew install snapcast

snapclient

会自动找到局域网中的 server 并连接

总结

任意终端打开控制网页:

Untitled

Untitled

发现两台 MBP 都连接上了,使用 iPhone 隔空投送到 Snapcast:

Untitled

两台 MBP 无缝播放音乐咯!

缺点是无法用两台 MBP 中的一个作为输出,还需要 iPhone 介入,应该有解决方案,之后再看了

  • 在其中一台 MBP 中部署了 navidrome,然后通过 iPhone 放歌。配合 iPhone 上的 substreamer 免费 app ,好用到飞起。但要注意 local music 的质量,需要 ~~mp3tag~~Tagr (自动提取真的好用极了)这样的软件编辑好音频元数据。

Untitled

专辑封面哪里找?Last.fm | Play music, find songs, and discover artists

Untitled

音乐下载使用 msc:https://github.com/zonemeen/musicn https://github.com/zonemeen/musicn

参考文章

GitHub Snapcast

标签 :
comments powered by Disqus

相关文章

beancount-gs 一款 self-hosted 复式记账程序,简化你的记账方式!

本程序将部署在 macOS 上,不使用 Docker(mac 上的 Docker 太卡了) 动机 我利用 beancount 来记账已经有一段时间了,但有些痛点问题困扰着我: 没有 web 界面,fava 真的只能用来展示和分析且不好理解,需要有一个方便记账的界面 text 记账的方式,在 vscode 插件能力有限的情况下,很容易忘记 assets 和 expenses 到底叫啥名,如果有 web 界面那么一定会好很多,一个下拉列表就可以解决 有没有解决方案,不需要自己造轮子的那种?

阅读更多
如何在 React 中进行状态管理?使用 Zustand!

计数器 import { create } from 'zustand' const useStore = create(set => ({ count: 1, inc: () => set(state => ({ count: state.count + 1 })), })) function Controls() { const inc = useStore(state => state.inc) return <button onClick={inc}>one up</button> } function Counter() { const count = useStore(state => state.count) return <h1>{count}</h1> } 用法 创建状态 state 操作 action Basic typescript usage doesn’t require anything special except for writing create<State>()(...) instead of create(...)… import { create } from 'zustand' interface BearState { bears: number increase: (by: number) => void } const useBearStore = create<BearState>()((set) => ({ bears: 0, increase: (by) => set((state) => ({ bears: state.bears + by })), })) const useFishStore = create((set) => ({ salmon: 1, tuna: 2, deleteEverything: () => set({}, true), // clears the entire store, actions included deleteTuna: () => set((state) => omit(state, ['tuna']), true), })) const useSoundStore = create((set, get) => ({ sound: "grunt", action: () => { const sound = get().sound // you still have access to state outside of it through get // ... } }) export default useBearStore 在 react 之外使用呢?

阅读更多
图数据库 Cypher 查询语言的子查询 CALL (subquery)

子查询允许将查询组合起来,这在使用UNION或聚合时特别有用。 子查询与封闭查询交互的方式有一些限制: 子查询只能引用外部查询中显式导入的变量。 子查询不能返回与外围查询中变量名称相同的变量。 从子查询返回的所有变量随后都可在外部查询中使用。 WITH导入变量进子句必须:

阅读更多