Compare commits
No commits in common. "master" and "before-nix" have entirely different histories.
master
...
before-nix
627 changed files with 5826 additions and 34781 deletions
|
@ -1,19 +0,0 @@
|
||||||
name: Discord
|
|
||||||
|
|
||||||
on:
|
|
||||||
- workflow_dispatch
|
|
||||||
- push
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
discord_commits:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
name: discord commits
|
|
||||||
if: contains(github.event.head_commit.message, '(servers)')
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Discommit
|
|
||||||
uses: https://github.com/matt1432/discommit@v0.0.2
|
|
||||||
with:
|
|
||||||
discord_webhook: ${{ secrets.DISCORD_WEBHOOK }}
|
|
||||||
api_url: 'https://git.nelim.org/api/v1/repos/$OWNER/$REPO/git/commits/$REF'
|
|
||||||
title: 'New commit containing changes to server configs:'
|
|
5
.gitattributes
vendored
5
.gitattributes
vendored
|
@ -1,5 +0,0 @@
|
||||||
flake.lock -diff
|
|
||||||
flake.nix -diff
|
|
||||||
**/non-declarative-conf -diff
|
|
||||||
**/package-lock.json -diff
|
|
||||||
**/HomeAssistantGenerated -diff
|
|
99
.gitignore
vendored
99
.gitignore
vendored
|
@ -1,23 +1,76 @@
|
||||||
# Python
|
okularrc
|
||||||
*.egg-info
|
user-dirs.dirs
|
||||||
|
.gsd-keyboard.settings-ported
|
||||||
# NPM
|
arkrc
|
||||||
*node_modules
|
evolution/
|
||||||
*build/
|
fontconfig/
|
||||||
|
gnome-initial-setup-done
|
||||||
# Direnv
|
kcharselectrc
|
||||||
*.direnv/
|
user-dirs.locale
|
||||||
|
GalaxyBudsClient/
|
||||||
# Generated by nix
|
baloofilerc
|
||||||
result*
|
Zeal/
|
||||||
!results/
|
environment.d/
|
||||||
.nixd.json
|
gtk-3.0/bookmarks
|
||||||
|
Kvantum/**/
|
||||||
## AGS
|
chromium/
|
||||||
**/vars.ts
|
Electron/
|
||||||
**/config.js
|
GIMP/
|
||||||
*icons
|
Nextcloud/
|
||||||
**/types
|
PhotoQt/
|
||||||
|
QtProject.conf
|
||||||
# Other
|
VSCodium/Cache/
|
||||||
*.temp
|
VSCodium/CachedData/
|
||||||
|
VSCodium/CachedExtensionVSIXs/
|
||||||
|
VSCodium/CachedProfilesData/
|
||||||
|
VSCodium/Code Cache/
|
||||||
|
VSCodium/Cookies
|
||||||
|
VSCodium/Cookies-journal
|
||||||
|
VSCodium/Crashpad/
|
||||||
|
VSCodium/DawnCache/
|
||||||
|
VSCodium/Dictionaries/
|
||||||
|
VSCodium/GPUCache/
|
||||||
|
VSCodium/Local Storage/
|
||||||
|
VSCodium/Network Persistent State
|
||||||
|
VSCodium/Preferences
|
||||||
|
VSCodium/Service Worker/
|
||||||
|
VSCodium/Session Storage/
|
||||||
|
VSCodium/TransportSecurity
|
||||||
|
VSCodium/User/History/
|
||||||
|
VSCodium/User/globalStorage/
|
||||||
|
VSCodium/User/workspaceStorage/
|
||||||
|
VSCodium/WebStorage/
|
||||||
|
VSCodium/code.lock
|
||||||
|
VSCodium/databases/
|
||||||
|
VSCodium/languagepacks.json
|
||||||
|
VSCodium/logs/
|
||||||
|
VSCodium/machineid
|
||||||
|
akregatorrc
|
||||||
|
autostart/
|
||||||
|
baloofileinformationrc
|
||||||
|
dconf/
|
||||||
|
discord/
|
||||||
|
gtk-4.0/
|
||||||
|
htop/
|
||||||
|
kconf_updaterc
|
||||||
|
kde.org/
|
||||||
|
kdeglobals
|
||||||
|
kiorc
|
||||||
|
libreoffice/
|
||||||
|
mimeapps.list
|
||||||
|
pavucontrol.ini
|
||||||
|
pulse/
|
||||||
|
replugged/
|
||||||
|
session/
|
||||||
|
spicetify/
|
||||||
|
spotify/
|
||||||
|
systemd/
|
||||||
|
trashrc
|
||||||
|
tuta_integration/
|
||||||
|
tutanota-desktop/
|
||||||
|
unity3d/
|
||||||
|
unityhub/
|
||||||
|
vlc/
|
||||||
|
wireguard/
|
||||||
|
xsettingsd/
|
||||||
|
waybar/scripts/.heart
|
||||||
|
|
2
Kvantum/kvantum.kvconfig
Normal file
2
Kvantum/kvantum.kvconfig
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
[General]
|
||||||
|
theme=LavandaDark
|
16
LICENSE.md
16
LICENSE.md
|
@ -1,16 +0,0 @@
|
||||||
MIT No Attribution
|
|
||||||
|
|
||||||
Copyright 2024 Mathis H.
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of this
|
|
||||||
software and associated documentation files (the "Software"), to deal in the Software
|
|
||||||
without restriction, including without limitation the rights to use, copy, modify,
|
|
||||||
merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
|
|
||||||
permit persons to whom the Software is furnished to do so.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
|
|
||||||
INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
|
|
||||||
PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
||||||
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
||||||
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
||||||
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
123
README.md
123
README.md
|
@ -1,122 +1,3 @@
|
||||||
# My NixOS configs
|
Kvantum, qt5ct and my environment are for theming QT apps the same way I do with GTK apps with lxappearance (i have to run it with the X11 backend)
|
||||||
|
|
||||||
TODO: add directory structure info and enforce it
|
I use some scripts to make my own tablet mode since my laptop's switch is not yet in the kernel
|
||||||
|
|
||||||
- [x] every root folder in the repo represents a flake output except inputs
|
|
||||||
- [x] every root folder only has an optional `default.nix` and subfolders for each
|
|
||||||
of its attrs
|
|
||||||
- [x] if there is non nix code, it will be in a `config` folder
|
|
||||||
- [x] every module should not do anything if imported
|
|
||||||
- [x] all nix files that represent a module should be `default.nix` (a nix file
|
|
||||||
which is imported directly can be called anything alongside `default.nix`)
|
|
||||||
- [ ] redo docs
|
|
||||||
|
|
||||||
## AGS
|
|
||||||
|
|
||||||
You might find it weird that most of my config is written in TypeScript.
|
|
||||||
That's because all my desktops run
|
|
||||||
[AGS](https://github.com/Aylur/ags)
|
|
||||||
for UI. Click on
|
|
||||||
[this](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/modules/ags)
|
|
||||||
to see my configuration.
|
|
||||||
|
|
||||||
I'm also a victim of Stockholm syndrome at this point and make my scripts
|
|
||||||
in TypeScript because it's the scripting language I am most comfortable with.
|
|
||||||
|
|
||||||
## About
|
|
||||||
|
|
||||||
### General
|
|
||||||
|
|
||||||
This repo is the complete configuration of machines I own,
|
|
||||||
running NixOS or Nix. Its structure is based on a flake's
|
|
||||||
[outputs](https://wiki.nixos.org/wiki/Flakes#Output_schema).
|
|
||||||
|
|
||||||
### Flake Location
|
|
||||||
|
|
||||||
This git repo will always be located at `$FLAKE` (`config.environment.variables.FLAKE`)
|
|
||||||
and symlinked to `/etc/nixos` to have everything where NixOS tools
|
|
||||||
expect things to be.
|
|
||||||
|
|
||||||
ie.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
sudo rm -r /etc/nixos
|
|
||||||
|
|
||||||
echo "$FLAKE" # /home/matt/.nix
|
|
||||||
|
|
||||||
sudo ln -sf /home/matt/.nix /etc/nixos
|
|
||||||
```
|
|
||||||
|
|
||||||
### Flake Outputs
|
|
||||||
|
|
||||||
| Output | Description |
|
|
||||||
| ---------------------------------- | ----------- |
|
|
||||||
| `nixosConfigurations` | [devices](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/devices)' + ISO's configurations |
|
|
||||||
| `nixOnDroidConfigurations.default` | [Nix-On-Droid](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/devices/android)'s configuration |
|
|
||||||
| `packages` | Some custom [packages](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/packages) not available in nixpkgs or modified from it |
|
|
||||||
| `scopedPackages` | Some custom [package scopes](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/scopedPackages) not available in nixpkgs or modified from it |
|
|
||||||
| `apps` | Scripts ran from the flake defined [here](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/apps) |
|
|
||||||
| `homeManagerModules` | [Modules](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/homeManagerModules) made for home-manager |
|
|
||||||
| `homeManagerModules` | [Modules](https://git.nelim.org/matt1432/nixos-configs/src/branch/master/modules) made for NixOS systems |
|
|
||||||
| `formatter` | I format nix code with [alejandra](https://github.com/kamadorueda/alejandra) |
|
|
||||||
| `devShells.default` | A dev shell to build an ISO from the live-image nixosConfiguration |
|
|
||||||
| `devShells.ags` | A dev shell to have a NodeJS env when I enter my AGS's config directory |
|
|
||||||
|
|
||||||
### Flake Inputs
|
|
||||||
|
|
||||||
To allow use of the full nix language for my inputs, I use [genflake](https://github.com/jorsn/flakegen).
|
|
||||||
Therefore, the flake I edit is located at `./outputs.nix`.
|
|
||||||
|
|
||||||
I also prefer using a more descriptive format for my inputs like so:
|
|
||||||
|
|
||||||
```nix
|
|
||||||
nixpkgs = {
|
|
||||||
type = "github";
|
|
||||||
owner = "NixOS";
|
|
||||||
repo = "nixpkgs";
|
|
||||||
|
|
||||||
# Branch name
|
|
||||||
ref = "nixos-unstable";
|
|
||||||
|
|
||||||
# Pin this input to a specific commit
|
|
||||||
rev = "842d9d80cfd4560648c785f8a4e6f3b096790e19";
|
|
||||||
};
|
|
||||||
```
|
|
||||||
|
|
||||||
to make it more clear what is what in the flake URI
|
|
||||||
|
|
||||||
### Secrets
|
|
||||||
|
|
||||||
All my secrets are in a private git repo that makes use of
|
|
||||||
[sops-nix](https://github.com/Mic92/sops-nix).
|
|
||||||
I generate `.sops.yaml` from `.sops.nix`:
|
|
||||||
|
|
||||||
```nix
|
|
||||||
let
|
|
||||||
wim = "somekey";
|
|
||||||
binto = "somekey2";
|
|
||||||
in {
|
|
||||||
creation_rules = [
|
|
||||||
{
|
|
||||||
path_regex = "secrets/[^/]+\\.(yaml|json|env|ini)$";
|
|
||||||
key_groups = [
|
|
||||||
{
|
|
||||||
age = [wim binto];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
which is then converted to `.sops.yaml` using
|
|
||||||
[remarshal](https://github.com/remarshal-project/remarshal)
|
|
||||||
and this shell command:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
nix eval --json --file ./.sops.nix | remarshal --if json --of yaml > .sops.yaml
|
|
||||||
```
|
|
||||||
|
|
||||||
TLDR: I
|
|
||||||
**[hate](https://ruudvanasseldonk.com/2023/01/11/the-yaml-document-from-hell)**
|
|
||||||
YAML
|
|
||||||
|
|
3
TODO.md
Normal file
3
TODO.md
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
change tray icons
|
||||||
|
decide on layers for eww and waybar
|
||||||
|
double tap to right click
|
12
VSCodium/User/settings.json
Normal file
12
VSCodium/User/settings.json
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
{
|
||||||
|
"editor.fontSize": 16,
|
||||||
|
"editor.wordWrap": "on",
|
||||||
|
"omnisharp.useGlobalMono": "always",
|
||||||
|
"workbench.startupEditor": "none",
|
||||||
|
"workbench.colorTheme": "Cherry Midnight",
|
||||||
|
"git.openRepositoryInParentFolders": "always",
|
||||||
|
"git.autofetch": true,
|
||||||
|
"clangd.path": "/home/matt/.config/VSCodium/User/globalStorage/llvm-vs-code-extensions.vscode-clangd/install/16.0.2/clangd_16.0.2/bin/clangd",
|
||||||
|
"git.enableSmartCommit": true,
|
||||||
|
"git.confirmSync": false
|
||||||
|
}
|
920
alacritty/alacritty.yml
Normal file
920
alacritty/alacritty.yml
Normal file
|
@ -0,0 +1,920 @@
|
||||||
|
# Configuration for Alacritty, the GPU enhanced terminal emulator.
|
||||||
|
|
||||||
|
# Import additional configuration files
|
||||||
|
#
|
||||||
|
# Imports are loaded in order, skipping all missing files, with the importing
|
||||||
|
# file being loaded last. If a field is already present in a previous import, it
|
||||||
|
# will be replaced.
|
||||||
|
#
|
||||||
|
# All imports must either be absolute paths starting with `/`, or paths relative
|
||||||
|
# to the user's home directory starting with `~/`.
|
||||||
|
#import:
|
||||||
|
# - /path/to/alacritty.yml
|
||||||
|
|
||||||
|
# Any items in the `env` entry below will be added as
|
||||||
|
# environment variables. Some entries may override variables
|
||||||
|
# set by alacritty itself.
|
||||||
|
env:
|
||||||
|
# TERM variable
|
||||||
|
#
|
||||||
|
# This value is used to set the `$TERM` environment variable for
|
||||||
|
# each instance of Alacritty. If it is not present, alacritty will
|
||||||
|
# check the local terminfo database and use `alacritty` if it is
|
||||||
|
# available, otherwise `xterm-256color` is used.
|
||||||
|
#TERM: alacritty
|
||||||
|
POKE: "true"
|
||||||
|
|
||||||
|
window:
|
||||||
|
# Window dimensions (changes require restart)
|
||||||
|
#
|
||||||
|
# Number of lines/columns (not pixels) in the terminal. Both lines and columns
|
||||||
|
# must be non-zero for this to take effect. The number of columns must be at
|
||||||
|
# least `2`, while using a value of `0` for columns and lines will fall back
|
||||||
|
# to the window manager's recommended size
|
||||||
|
#dimensions:
|
||||||
|
# columns: 0
|
||||||
|
# lines: 0
|
||||||
|
|
||||||
|
# Window position (changes require restart)
|
||||||
|
#
|
||||||
|
# Specified in number of pixels.
|
||||||
|
# If the position is not set, the window manager will handle the placement.
|
||||||
|
#position:
|
||||||
|
# x: 0
|
||||||
|
# y: 0
|
||||||
|
|
||||||
|
# Window padding (changes require restart)
|
||||||
|
#
|
||||||
|
# Blank space added around the window in pixels. This padding is scaled
|
||||||
|
# by DPI and the specified value is always added at both opposing sides.
|
||||||
|
padding:
|
||||||
|
x: 0
|
||||||
|
y: 10
|
||||||
|
|
||||||
|
# Spread additional padding evenly around the terminal content.
|
||||||
|
#dynamic_padding: false
|
||||||
|
|
||||||
|
# Window decorations
|
||||||
|
#
|
||||||
|
# Values for `decorations`:
|
||||||
|
# - full: Borders and title bar
|
||||||
|
# - none: Neither borders nor title bar
|
||||||
|
#
|
||||||
|
# Values for `decorations` (macOS only):
|
||||||
|
# - transparent: Title bar, transparent background and title bar buttons
|
||||||
|
# - buttonless: Title bar, transparent background and no title bar buttons
|
||||||
|
#decorations: full
|
||||||
|
|
||||||
|
# Background opacity
|
||||||
|
#
|
||||||
|
# Window opacity as a floating point number from `0.0` to `1.0`.
|
||||||
|
# The value `0.0` is completely transparent and `1.0` is opaque.
|
||||||
|
opacity: 0.8
|
||||||
|
|
||||||
|
# Startup Mode (changes require restart)
|
||||||
|
#
|
||||||
|
# Values for `startup_mode`:
|
||||||
|
# - Windowed
|
||||||
|
# - Maximized
|
||||||
|
# - Fullscreen
|
||||||
|
#
|
||||||
|
# Values for `startup_mode` (macOS only):
|
||||||
|
# - SimpleFullscreen
|
||||||
|
#startup_mode: Windowed
|
||||||
|
|
||||||
|
# Window title
|
||||||
|
#title: Alacritty
|
||||||
|
|
||||||
|
# Allow terminal applications to change Alacritty's window title.
|
||||||
|
#dynamic_title: true
|
||||||
|
|
||||||
|
# Window class (Linux/BSD only):
|
||||||
|
#class:
|
||||||
|
# Application instance name
|
||||||
|
#instance: Alacritty
|
||||||
|
# General application class
|
||||||
|
#general: Alacritty
|
||||||
|
|
||||||
|
# Decorations theme variant
|
||||||
|
#
|
||||||
|
# Override the variant of the System theme/GTK theme/Wayland client side
|
||||||
|
# decorations. Commonly supported values are `Dark`, `Light`, and `None` for
|
||||||
|
# auto pick-up. Set this to `None` to use the default theme variant.
|
||||||
|
#decorations_theme_variant: None
|
||||||
|
|
||||||
|
# Resize increments
|
||||||
|
#
|
||||||
|
# Prefer resizing window by discrete steps equal to cell dimensions.
|
||||||
|
#resize_increments: false
|
||||||
|
|
||||||
|
# Make `Option` key behave as `Alt` (macOS only):
|
||||||
|
# - OnlyLeft
|
||||||
|
# - OnlyRight
|
||||||
|
# - Both
|
||||||
|
# - None (default)
|
||||||
|
#option_as_alt: None
|
||||||
|
|
||||||
|
#scrolling:
|
||||||
|
# Maximum number of lines in the scrollback buffer.
|
||||||
|
# Specifying '0' will disable scrolling.
|
||||||
|
#history: 10000
|
||||||
|
|
||||||
|
# Scrolling distance multiplier.
|
||||||
|
#multiplier: 3
|
||||||
|
|
||||||
|
# Font configuration
|
||||||
|
#font:
|
||||||
|
# Normal (roman) font face
|
||||||
|
#normal:
|
||||||
|
# Font family
|
||||||
|
#
|
||||||
|
# Default:
|
||||||
|
# - (macOS) Menlo
|
||||||
|
# - (Linux/BSD) monospace
|
||||||
|
# - (Windows) Consolas
|
||||||
|
#family: monospace
|
||||||
|
|
||||||
|
# The `style` can be specified to pick a specific face.
|
||||||
|
#style: Regular
|
||||||
|
|
||||||
|
# Bold font face
|
||||||
|
#bold:
|
||||||
|
# Font family
|
||||||
|
#
|
||||||
|
# If the bold family is not specified, it will fall back to the
|
||||||
|
# value specified for the normal font.
|
||||||
|
#family: monospace
|
||||||
|
|
||||||
|
# The `style` can be specified to pick a specific face.
|
||||||
|
#style: Bold
|
||||||
|
|
||||||
|
# Italic font face
|
||||||
|
#italic:
|
||||||
|
# Font family
|
||||||
|
#
|
||||||
|
# If the italic family is not specified, it will fall back to the
|
||||||
|
# value specified for the normal font.
|
||||||
|
#family: monospace
|
||||||
|
|
||||||
|
# The `style` can be specified to pick a specific face.
|
||||||
|
#style: Italic
|
||||||
|
|
||||||
|
# Bold italic font face
|
||||||
|
#bold_italic:
|
||||||
|
# Font family
|
||||||
|
#
|
||||||
|
# If the bold italic family is not specified, it will fall back to the
|
||||||
|
# value specified for the normal font.
|
||||||
|
#family: monospace
|
||||||
|
|
||||||
|
# The `style` can be specified to pick a specific face.
|
||||||
|
#style: Bold Italic
|
||||||
|
|
||||||
|
# Point size
|
||||||
|
#size: 11.0
|
||||||
|
|
||||||
|
# Offset is the extra space around each character. `offset.y` can be thought
|
||||||
|
# of as modifying the line spacing, and `offset.x` as modifying the letter
|
||||||
|
# spacing.
|
||||||
|
#offset:
|
||||||
|
# x: 0
|
||||||
|
# y: 0
|
||||||
|
|
||||||
|
# Glyph offset determines the locations of the glyphs within their cells with
|
||||||
|
# the default being at the bottom. Increasing `x` moves the glyph to the
|
||||||
|
# right, increasing `y` moves the glyph upward.
|
||||||
|
#glyph_offset:
|
||||||
|
# x: 0
|
||||||
|
# y: 0
|
||||||
|
|
||||||
|
# Use built-in font for box drawing characters.
|
||||||
|
#
|
||||||
|
# If `true`, Alacritty will use a custom built-in font for box drawing
|
||||||
|
# characters (Unicode points 2500 - 259f).
|
||||||
|
#
|
||||||
|
#builtin_box_drawing: true
|
||||||
|
|
||||||
|
# If `true`, bold text is drawn using the bright color variants.
|
||||||
|
#draw_bold_text_with_bright_colors: false
|
||||||
|
|
||||||
|
# Colors (Tomorrow Night)
|
||||||
|
colors:
|
||||||
|
# Default colors
|
||||||
|
primary:
|
||||||
|
# background: '#1d1f21'
|
||||||
|
background: '#0c0c0c'
|
||||||
|
# foreground: '#c5c8c6'
|
||||||
|
|
||||||
|
# Bright and dim foreground colors
|
||||||
|
#
|
||||||
|
# The dimmed foreground color is calculated automatically if it is not
|
||||||
|
# present. If the bright foreground color is not set, or
|
||||||
|
# `draw_bold_text_with_bright_colors` is `false`, the normal foreground
|
||||||
|
# color will be used.
|
||||||
|
#dim_foreground: '#828482'
|
||||||
|
#bright_foreground: '#eaeaea'
|
||||||
|
|
||||||
|
# Cursor colors
|
||||||
|
#
|
||||||
|
# Colors which should be used to draw the terminal cursor.
|
||||||
|
#
|
||||||
|
# Allowed values are CellForeground/CellBackground, which reference the
|
||||||
|
# affected cell, or hexadecimal colors like #ff00ff.
|
||||||
|
#cursor:
|
||||||
|
# text: CellBackground
|
||||||
|
# cursor: CellForeground
|
||||||
|
|
||||||
|
# Vi mode cursor colors
|
||||||
|
#
|
||||||
|
# Colors for the cursor when the vi mode is active.
|
||||||
|
#
|
||||||
|
# Allowed values are CellForeground/CellBackground, which reference the
|
||||||
|
# affected cell, or hexadecimal colors like #ff00ff.
|
||||||
|
#vi_mode_cursor:
|
||||||
|
# text: CellBackground
|
||||||
|
# cursor: CellForeground
|
||||||
|
|
||||||
|
# Search colors
|
||||||
|
#
|
||||||
|
# Colors used for the search bar and match highlighting.
|
||||||
|
#search:
|
||||||
|
# Allowed values are CellForeground/CellBackground, which reference the
|
||||||
|
# affected cell, or hexadecimal colors like #ff00ff.
|
||||||
|
#matches:
|
||||||
|
# foreground: '#000000'
|
||||||
|
# background: '#ffffff'
|
||||||
|
#focused_match:
|
||||||
|
# foreground: '#ffffff'
|
||||||
|
# background: '#000000'
|
||||||
|
|
||||||
|
# Keyboard hints
|
||||||
|
#hints:
|
||||||
|
# First character in the hint label
|
||||||
|
#
|
||||||
|
# Allowed values are CellForeground/CellBackground, which reference the
|
||||||
|
# affected cell, or hexadecimal colors like #ff00ff.
|
||||||
|
#start:
|
||||||
|
# foreground: '#1d1f21'
|
||||||
|
# background: '#e9ff5e'
|
||||||
|
|
||||||
|
# All characters after the first one in the hint label
|
||||||
|
#
|
||||||
|
# Allowed values are CellForeground/CellBackground, which reference the
|
||||||
|
# affected cell, or hexadecimal colors like #ff00ff.
|
||||||
|
#end:
|
||||||
|
# foreground: '#e9ff5e'
|
||||||
|
# background: '#1d1f21'
|
||||||
|
|
||||||
|
# Line indicator
|
||||||
|
#
|
||||||
|
# Color used for the indicator displaying the position in history during
|
||||||
|
# search and vi mode.
|
||||||
|
#
|
||||||
|
# By default, these will use the opposing primary color.
|
||||||
|
#line_indicator:
|
||||||
|
# foreground: None
|
||||||
|
# background: None
|
||||||
|
|
||||||
|
# Footer bar
|
||||||
|
#
|
||||||
|
# Color used for the footer bar on the bottom, used by search regex input,
|
||||||
|
# hyperlink URI preview, etc.
|
||||||
|
#
|
||||||
|
#footer_bar:
|
||||||
|
# background: '#c5c8c6'
|
||||||
|
# foreground: '#1d1f21'
|
||||||
|
|
||||||
|
# Selection colors
|
||||||
|
#
|
||||||
|
# Colors which should be used to draw the selection area.
|
||||||
|
#
|
||||||
|
# Allowed values are CellForeground/CellBackground, which reference the
|
||||||
|
# affected cell, or hexadecimal colors like #ff00ff.
|
||||||
|
#selection:
|
||||||
|
# text: CellBackground
|
||||||
|
# background: CellForeground
|
||||||
|
|
||||||
|
# Normal colors
|
||||||
|
#normal:
|
||||||
|
# black: '#1d1f21'
|
||||||
|
# red: '#cc6666'
|
||||||
|
# green: '#b5bd68'
|
||||||
|
# yellow: '#f0c674'
|
||||||
|
# blue: '#81a2be'
|
||||||
|
# magenta: '#b294bb'
|
||||||
|
# cyan: '#8abeb7'
|
||||||
|
# white: '#c5c8c6'
|
||||||
|
|
||||||
|
# Bright colors
|
||||||
|
#bright:
|
||||||
|
# black: '#666666'
|
||||||
|
# red: '#d54e53'
|
||||||
|
# green: '#b9ca4a'
|
||||||
|
# yellow: '#e7c547'
|
||||||
|
# blue: '#7aa6da'
|
||||||
|
# magenta: '#c397d8'
|
||||||
|
# cyan: '#70c0b1'
|
||||||
|
# white: '#eaeaea'
|
||||||
|
|
||||||
|
# Dim colors
|
||||||
|
#
|
||||||
|
# If the dim colors are not set, they will be calculated automatically based
|
||||||
|
# on the `normal` colors.
|
||||||
|
#dim:
|
||||||
|
# black: '#131415'
|
||||||
|
# red: '#864343'
|
||||||
|
# green: '#777c44'
|
||||||
|
# yellow: '#9e824c'
|
||||||
|
# blue: '#556a7d'
|
||||||
|
# magenta: '#75617b'
|
||||||
|
# cyan: '#5b7d78'
|
||||||
|
# white: '#828482'
|
||||||
|
|
||||||
|
# Indexed Colors
|
||||||
|
#
|
||||||
|
# The indexed colors include all colors from 16 to 256.
|
||||||
|
# When these are not set, they're filled with sensible defaults.
|
||||||
|
#
|
||||||
|
# Example:
|
||||||
|
# `- { index: 16, color: '#ff00ff' }`
|
||||||
|
#
|
||||||
|
#indexed_colors: []
|
||||||
|
|
||||||
|
# Transparent cell backgrounds
|
||||||
|
#
|
||||||
|
# Whether or not `window.opacity` applies to all cell backgrounds or only to
|
||||||
|
# the default background. When set to `true` all cells will be transparent
|
||||||
|
# regardless of their background color.
|
||||||
|
#transparent_background_colors: false
|
||||||
|
|
||||||
|
# Bell
|
||||||
|
#
|
||||||
|
# The bell is rung every time the BEL control character is received.
|
||||||
|
#bell:
|
||||||
|
# Visual Bell Animation
|
||||||
|
#
|
||||||
|
# Animation effect for flashing the screen when the visual bell is rung.
|
||||||
|
#
|
||||||
|
# Values for `animation`:
|
||||||
|
# - Ease
|
||||||
|
# - EaseOut
|
||||||
|
# - EaseOutSine
|
||||||
|
# - EaseOutQuad
|
||||||
|
# - EaseOutCubic
|
||||||
|
# - EaseOutQuart
|
||||||
|
# - EaseOutQuint
|
||||||
|
# - EaseOutExpo
|
||||||
|
# - EaseOutCirc
|
||||||
|
# - Linear
|
||||||
|
#animation: EaseOutExpo
|
||||||
|
|
||||||
|
# Duration of the visual bell flash in milliseconds. A `duration` of `0` will
|
||||||
|
# disable the visual bell animation.
|
||||||
|
#duration: 0
|
||||||
|
|
||||||
|
# Visual bell animation color.
|
||||||
|
#color: '#ffffff'
|
||||||
|
|
||||||
|
# Bell Command
|
||||||
|
#
|
||||||
|
# This program is executed whenever the bell is rung.
|
||||||
|
#
|
||||||
|
# When set to `command: None`, no command will be executed.
|
||||||
|
#
|
||||||
|
# Example:
|
||||||
|
# command:
|
||||||
|
# program: notify-send
|
||||||
|
# args: ["Hello, World!"]
|
||||||
|
#
|
||||||
|
#command: None
|
||||||
|
|
||||||
|
#selection:
|
||||||
|
# This string contains all characters that are used as separators for
|
||||||
|
# "semantic words" in Alacritty.
|
||||||
|
#semantic_escape_chars: ",│`|:\"' ()[]{}<>\t"
|
||||||
|
|
||||||
|
# When set to `true`, selected text will be copied to the primary clipboard.
|
||||||
|
#save_to_clipboard: false
|
||||||
|
|
||||||
|
#cursor:
|
||||||
|
# Cursor style
|
||||||
|
#style:
|
||||||
|
# Cursor shape
|
||||||
|
#
|
||||||
|
# Values for `shape`:
|
||||||
|
# - ▇ Block
|
||||||
|
# - _ Underline
|
||||||
|
# - | Beam
|
||||||
|
#shape: Block
|
||||||
|
|
||||||
|
# Cursor blinking state
|
||||||
|
#
|
||||||
|
# Values for `blinking`:
|
||||||
|
# - Never: Prevent the cursor from ever blinking
|
||||||
|
# - Off: Disable blinking by default
|
||||||
|
# - On: Enable blinking by default
|
||||||
|
# - Always: Force the cursor to always blink
|
||||||
|
#blinking: Off
|
||||||
|
|
||||||
|
# Vi mode cursor style
|
||||||
|
#
|
||||||
|
# If the vi mode cursor style is `None` or not specified, it will fall back to
|
||||||
|
# the style of the active value of the normal cursor.
|
||||||
|
#
|
||||||
|
# See `cursor.style` for available options.
|
||||||
|
#vi_mode_style: None
|
||||||
|
|
||||||
|
# Cursor blinking interval in milliseconds.
|
||||||
|
#blink_interval: 750
|
||||||
|
|
||||||
|
# Time after which cursor stops blinking, in seconds.
|
||||||
|
#
|
||||||
|
# Specifying '0' will disable timeout for blinking.
|
||||||
|
#blink_timeout: 5
|
||||||
|
|
||||||
|
# If this is `true`, the cursor will be rendered as a hollow box when the
|
||||||
|
# window is not focused.
|
||||||
|
#unfocused_hollow: true
|
||||||
|
|
||||||
|
# Thickness of the cursor relative to the cell width as floating point number
|
||||||
|
# from `0.0` to `1.0`.
|
||||||
|
#thickness: 0.15
|
||||||
|
|
||||||
|
# Live config reload (changes require restart)
|
||||||
|
#live_config_reload: true
|
||||||
|
|
||||||
|
# Shell
|
||||||
|
#
|
||||||
|
# You can set `shell.program` to the path of your favorite shell, e.g.
|
||||||
|
# `/bin/fish`. Entries in `shell.args` are passed unmodified as arguments to the
|
||||||
|
# shell.
|
||||||
|
#
|
||||||
|
# Default:
|
||||||
|
# - (Linux/BSD/macOS) `$SHELL` or the user's login shell, if `$SHELL` is unset
|
||||||
|
# - (Windows) powershell
|
||||||
|
#shell:
|
||||||
|
# program: /bin/bash
|
||||||
|
# args:
|
||||||
|
# - --login
|
||||||
|
|
||||||
|
# Startup directory
|
||||||
|
#
|
||||||
|
# Directory the shell is started in. If this is unset, or `None`, the working
|
||||||
|
# directory of the parent process will be used.
|
||||||
|
#working_directory: None
|
||||||
|
|
||||||
|
# Offer IPC using `alacritty msg` (unix only)
|
||||||
|
#ipc_socket: true
|
||||||
|
|
||||||
|
#mouse:
|
||||||
|
# Click settings
|
||||||
|
#
|
||||||
|
# The `double_click` and `triple_click` settings control the time
|
||||||
|
# alacritty should wait for accepting multiple clicks as one double
|
||||||
|
# or triple click.
|
||||||
|
#double_click: { threshold: 300 }
|
||||||
|
#triple_click: { threshold: 300 }
|
||||||
|
|
||||||
|
# If this is `true`, the cursor is temporarily hidden when typing.
|
||||||
|
#hide_when_typing: false
|
||||||
|
|
||||||
|
# Hints
|
||||||
|
#
|
||||||
|
# Terminal hints can be used to find text or hyperlink in the visible part of
|
||||||
|
# the terminal and pipe it to other applications.
|
||||||
|
#hints:
|
||||||
|
# Keys used for the hint labels.
|
||||||
|
#alphabet: "jfkdls;ahgurieowpq"
|
||||||
|
|
||||||
|
# List with all available hints
|
||||||
|
#
|
||||||
|
# Each hint must have any of `regex` or `hyperlinks` field and either an
|
||||||
|
# `action` or a `command` field. The fields `mouse`, `binding` and
|
||||||
|
# `post_processing` are optional.
|
||||||
|
#
|
||||||
|
# The `hyperlinks` option will cause OSC 8 escape sequence hyperlinks to be
|
||||||
|
# highlighted.
|
||||||
|
#
|
||||||
|
# The fields `command`, `binding.key`, `binding.mods`, `binding.mode` and
|
||||||
|
# `mouse.mods` accept the same values as they do in the `key_bindings` section.
|
||||||
|
#
|
||||||
|
# The `mouse.enabled` field controls if the hint should be underlined while
|
||||||
|
# the mouse with all `mouse.mods` keys held or the vi mode cursor is above it.
|
||||||
|
#
|
||||||
|
# If the `post_processing` field is set to `true`, heuristics will be used to
|
||||||
|
# shorten the match if there are characters likely not to be part of the hint
|
||||||
|
# (e.g. a trailing `.`). This is most useful for URIs and applies only to
|
||||||
|
# `regex` matches.
|
||||||
|
#
|
||||||
|
# Values for `action`:
|
||||||
|
# - Copy
|
||||||
|
# Copy the hint's text to the clipboard.
|
||||||
|
# - Paste
|
||||||
|
# Paste the hint's text to the terminal or search.
|
||||||
|
# - Select
|
||||||
|
# Select the hint's text.
|
||||||
|
# - MoveViModeCursor
|
||||||
|
# Move the vi mode cursor to the beginning of the hint.
|
||||||
|
#enabled:
|
||||||
|
# - regex: "(ipfs:|ipns:|magnet:|mailto:|gemini:|gopher:|https:|http:|news:|file:|git:|ssh:|ftp:)\
|
||||||
|
# [^\u0000-\u001F\u007F-\u009F<>\"\\s{-}\\^⟨⟩`]+"
|
||||||
|
# hyperlinks: true
|
||||||
|
# command: xdg-open
|
||||||
|
# post_processing: true
|
||||||
|
# mouse:
|
||||||
|
# enabled: true
|
||||||
|
# mods: None
|
||||||
|
# binding:
|
||||||
|
# key: U
|
||||||
|
# mods: Control|Shift
|
||||||
|
|
||||||
|
# Mouse bindings
|
||||||
|
#
|
||||||
|
# Mouse bindings are specified as a list of objects, much like the key
|
||||||
|
# bindings further below.
|
||||||
|
#
|
||||||
|
# To trigger mouse bindings when an application running within Alacritty
|
||||||
|
# captures the mouse, the `Shift` modifier is automatically added as a
|
||||||
|
# requirement.
|
||||||
|
#
|
||||||
|
# Each mouse binding will specify a:
|
||||||
|
#
|
||||||
|
# - `mouse`:
|
||||||
|
#
|
||||||
|
# - Middle
|
||||||
|
# - Left
|
||||||
|
# - Right
|
||||||
|
# - Numeric identifier such as `5`
|
||||||
|
#
|
||||||
|
# - `action` (see key bindings for actions not exclusive to mouse mode)
|
||||||
|
#
|
||||||
|
# - Mouse exclusive actions:
|
||||||
|
#
|
||||||
|
# - ExpandSelection
|
||||||
|
# Expand the selection to the current mouse cursor location.
|
||||||
|
#
|
||||||
|
# And optionally:
|
||||||
|
#
|
||||||
|
# - `mods` (see key bindings)
|
||||||
|
#mouse_bindings:
|
||||||
|
# - { mouse: Right, action: ExpandSelection }
|
||||||
|
# - { mouse: Right, mods: Control, action: ExpandSelection }
|
||||||
|
# - { mouse: Middle, mode: ~Vi, action: PasteSelection }
|
||||||
|
|
||||||
|
# Key bindings
|
||||||
|
#
|
||||||
|
# Key bindings are specified as a list of objects. For example, this is the
|
||||||
|
# default paste binding:
|
||||||
|
#
|
||||||
|
# `- { key: V, mods: Control|Shift, action: Paste }`
|
||||||
|
#
|
||||||
|
# Each key binding will specify a:
|
||||||
|
#
|
||||||
|
# - `key`: Identifier of the key pressed
|
||||||
|
#
|
||||||
|
# - A-Z
|
||||||
|
# - F1-F24
|
||||||
|
# - Key0-Key9
|
||||||
|
#
|
||||||
|
# A full list with available key codes can be found here:
|
||||||
|
# https://docs.rs/winit/*/winit/event/enum.VirtualKeyCode.html#variants
|
||||||
|
#
|
||||||
|
# Instead of using the name of the keys, the `key` field also supports using
|
||||||
|
# the scancode of the desired key. Scancodes have to be specified as a
|
||||||
|
# decimal number. This command will allow you to display the hex scancodes
|
||||||
|
# for certain keys:
|
||||||
|
#
|
||||||
|
# `showkey --scancodes`.
|
||||||
|
#
|
||||||
|
# Then exactly one of:
|
||||||
|
#
|
||||||
|
# - `chars`: Send a byte sequence to the running application
|
||||||
|
#
|
||||||
|
# The `chars` field writes the specified string to the terminal. This makes
|
||||||
|
# it possible to pass escape sequences. To find escape codes for bindings
|
||||||
|
# like `PageUp` (`"\x1b[5~"`), you can run the command `showkey -a` outside
|
||||||
|
# of tmux. Note that applications use terminfo to map escape sequences back
|
||||||
|
# to keys. It is therefore required to update the terminfo when changing an
|
||||||
|
# escape sequence.
|
||||||
|
#
|
||||||
|
# - `action`: Execute a predefined action
|
||||||
|
#
|
||||||
|
# - ToggleViMode
|
||||||
|
# - SearchForward
|
||||||
|
# Start searching toward the right of the search origin.
|
||||||
|
# - SearchBackward
|
||||||
|
# Start searching toward the left of the search origin.
|
||||||
|
# - Copy
|
||||||
|
# - Paste
|
||||||
|
# - IncreaseFontSize
|
||||||
|
# - DecreaseFontSize
|
||||||
|
# - ResetFontSize
|
||||||
|
# - ScrollPageUp
|
||||||
|
# - ScrollPageDown
|
||||||
|
# - ScrollHalfPageUp
|
||||||
|
# - ScrollHalfPageDown
|
||||||
|
# - ScrollLineUp
|
||||||
|
# - ScrollLineDown
|
||||||
|
# - ScrollToTop
|
||||||
|
# - ScrollToBottom
|
||||||
|
# - ClearHistory
|
||||||
|
# Remove the terminal's scrollback history.
|
||||||
|
# - Hide
|
||||||
|
# Hide the Alacritty window.
|
||||||
|
# - Minimize
|
||||||
|
# Minimize the Alacritty window.
|
||||||
|
# - Quit
|
||||||
|
# Quit Alacritty.
|
||||||
|
# - ToggleFullscreen
|
||||||
|
# - ToggleMaximized
|
||||||
|
# - SpawnNewInstance
|
||||||
|
# Spawn a new instance of Alacritty.
|
||||||
|
# - CreateNewWindow
|
||||||
|
# Create a new Alacritty window from the current process.
|
||||||
|
# - ClearLogNotice
|
||||||
|
# Clear Alacritty's UI warning and error notice.
|
||||||
|
# - ClearSelection
|
||||||
|
# Remove the active selection.
|
||||||
|
# - ReceiveChar
|
||||||
|
# - None
|
||||||
|
#
|
||||||
|
# - Vi mode exclusive actions:
|
||||||
|
#
|
||||||
|
# - Open
|
||||||
|
# Perform the action of the first matching hint under the vi mode cursor
|
||||||
|
# with `mouse.enabled` set to `true`.
|
||||||
|
# - ToggleNormalSelection
|
||||||
|
# - ToggleLineSelection
|
||||||
|
# - ToggleBlockSelection
|
||||||
|
# - ToggleSemanticSelection
|
||||||
|
# Toggle semantic selection based on `selection.semantic_escape_chars`.
|
||||||
|
# - CenterAroundViCursor
|
||||||
|
# Center view around vi mode cursor
|
||||||
|
#
|
||||||
|
# - Vi mode exclusive cursor motion actions:
|
||||||
|
#
|
||||||
|
# - Up
|
||||||
|
# One line up.
|
||||||
|
# - Down
|
||||||
|
# One line down.
|
||||||
|
# - Left
|
||||||
|
# One character left.
|
||||||
|
# - Right
|
||||||
|
# One character right.
|
||||||
|
# - First
|
||||||
|
# First column, or beginning of the line when already at the first column.
|
||||||
|
# - Last
|
||||||
|
# Last column, or beginning of the line when already at the last column.
|
||||||
|
# - FirstOccupied
|
||||||
|
# First non-empty cell in this terminal row, or first non-empty cell of
|
||||||
|
# the line when already at the first cell of the row.
|
||||||
|
# - High
|
||||||
|
# Top of the screen.
|
||||||
|
# - Middle
|
||||||
|
# Center of the screen.
|
||||||
|
# - Low
|
||||||
|
# Bottom of the screen.
|
||||||
|
# - SemanticLeft
|
||||||
|
# Start of the previous semantically separated word.
|
||||||
|
# - SemanticRight
|
||||||
|
# Start of the next semantically separated word.
|
||||||
|
# - SemanticLeftEnd
|
||||||
|
# End of the previous semantically separated word.
|
||||||
|
# - SemanticRightEnd
|
||||||
|
# End of the next semantically separated word.
|
||||||
|
# - WordLeft
|
||||||
|
# Start of the previous whitespace separated word.
|
||||||
|
# - WordRight
|
||||||
|
# Start of the next whitespace separated word.
|
||||||
|
# - WordLeftEnd
|
||||||
|
# End of the previous whitespace separated word.
|
||||||
|
# - WordRightEnd
|
||||||
|
# End of the next whitespace separated word.
|
||||||
|
# - Bracket
|
||||||
|
# Character matching the bracket at the cursor's location.
|
||||||
|
# - SearchNext
|
||||||
|
# Beginning of the next match.
|
||||||
|
# - SearchPrevious
|
||||||
|
# Beginning of the previous match.
|
||||||
|
# - SearchStart
|
||||||
|
# Start of the match to the left of the vi mode cursor.
|
||||||
|
# - SearchEnd
|
||||||
|
# End of the match to the right of the vi mode cursor.
|
||||||
|
#
|
||||||
|
# - Search mode exclusive actions:
|
||||||
|
# - SearchFocusNext
|
||||||
|
# Move the focus to the next search match.
|
||||||
|
# - SearchFocusPrevious
|
||||||
|
# Move the focus to the previous search match.
|
||||||
|
# - SearchConfirm
|
||||||
|
# - SearchCancel
|
||||||
|
# - SearchClear
|
||||||
|
# Reset the search regex.
|
||||||
|
# - SearchDeleteWord
|
||||||
|
# Delete the last word in the search regex.
|
||||||
|
# - SearchHistoryPrevious
|
||||||
|
# Go to the previous regex in the search history.
|
||||||
|
# - SearchHistoryNext
|
||||||
|
# Go to the next regex in the search history.
|
||||||
|
#
|
||||||
|
# - macOS exclusive actions:
|
||||||
|
# - ToggleSimpleFullscreen
|
||||||
|
# Enter fullscreen without occupying another space.
|
||||||
|
#
|
||||||
|
# - Linux/BSD exclusive actions:
|
||||||
|
#
|
||||||
|
# - CopySelection
|
||||||
|
# Copy from the selection buffer.
|
||||||
|
# - PasteSelection
|
||||||
|
# Paste from the selection buffer.
|
||||||
|
#
|
||||||
|
# - `command`: Fork and execute a specified command plus arguments
|
||||||
|
#
|
||||||
|
# The `command` field must be a map containing a `program` string and an
|
||||||
|
# `args` array of command line parameter strings. For example:
|
||||||
|
# `{ program: "alacritty", args: ["-e", "vttest"] }`
|
||||||
|
#
|
||||||
|
# And optionally:
|
||||||
|
#
|
||||||
|
# - `mods`: Key modifiers to filter binding actions
|
||||||
|
#
|
||||||
|
# - Command
|
||||||
|
# - Control
|
||||||
|
# - Option
|
||||||
|
# - Super
|
||||||
|
# - Shift
|
||||||
|
# - Alt
|
||||||
|
#
|
||||||
|
# Multiple `mods` can be combined using `|` like this:
|
||||||
|
# `mods: Control|Shift`.
|
||||||
|
# Whitespace and capitalization are relevant and must match the example.
|
||||||
|
#
|
||||||
|
# - `mode`: Indicate a binding for only specific terminal reported modes
|
||||||
|
#
|
||||||
|
# This is mainly used to send applications the correct escape sequences
|
||||||
|
# when in different modes.
|
||||||
|
#
|
||||||
|
# - AppCursor
|
||||||
|
# - AppKeypad
|
||||||
|
# - Search
|
||||||
|
# - Alt
|
||||||
|
# - Vi
|
||||||
|
#
|
||||||
|
# A `~` operator can be used before a mode to apply the binding whenever
|
||||||
|
# the mode is *not* active, e.g. `~Alt`.
|
||||||
|
#
|
||||||
|
# Bindings are always filled by default, but will be replaced when a new
|
||||||
|
# binding with the same triggers is defined. To unset a default binding, it can
|
||||||
|
# be mapped to the `ReceiveChar` action. Alternatively, you can use `None` for
|
||||||
|
# a no-op if you do not wish to receive input characters for that binding.
|
||||||
|
#
|
||||||
|
# If the same trigger is assigned to multiple actions, all of them are executed
|
||||||
|
# in the order they were defined in.
|
||||||
|
#key_bindings:
|
||||||
|
#- { key: Paste, action: Paste }
|
||||||
|
#- { key: Copy, action: Copy }
|
||||||
|
#- { key: L, mods: Control, action: ClearLogNotice }
|
||||||
|
#- { key: L, mods: Control, mode: ~Vi|~Search, chars: "\x0c" }
|
||||||
|
#- { key: PageUp, mods: Shift, mode: ~Alt, action: ScrollPageUp }
|
||||||
|
#- { key: PageDown, mods: Shift, mode: ~Alt, action: ScrollPageDown }
|
||||||
|
#- { key: Home, mods: Shift, mode: ~Alt, action: ScrollToTop }
|
||||||
|
#- { key: End, mods: Shift, mode: ~Alt, action: ScrollToBottom }
|
||||||
|
|
||||||
|
# Vi Mode
|
||||||
|
#- { key: Space, mods: Shift|Control, mode: ~Search, action: ToggleViMode }
|
||||||
|
#- { key: Space, mods: Shift|Control, mode: Vi|~Search, action: ScrollToBottom }
|
||||||
|
#- { key: Escape, mode: Vi|~Search, action: ClearSelection }
|
||||||
|
#- { key: I, mode: Vi|~Search, action: ToggleViMode }
|
||||||
|
#- { key: I, mode: Vi|~Search, action: ScrollToBottom }
|
||||||
|
#- { key: C, mods: Control, mode: Vi|~Search, action: ToggleViMode }
|
||||||
|
#- { key: Y, mods: Control, mode: Vi|~Search, action: ScrollLineUp }
|
||||||
|
#- { key: E, mods: Control, mode: Vi|~Search, action: ScrollLineDown }
|
||||||
|
#- { key: G, mode: Vi|~Search, action: ScrollToTop }
|
||||||
|
#- { key: G, mods: Shift, mode: Vi|~Search, action: ScrollToBottom }
|
||||||
|
#- { key: B, mods: Control, mode: Vi|~Search, action: ScrollPageUp }
|
||||||
|
#- { key: F, mods: Control, mode: Vi|~Search, action: ScrollPageDown }
|
||||||
|
#- { key: U, mods: Control, mode: Vi|~Search, action: ScrollHalfPageUp }
|
||||||
|
#- { key: D, mods: Control, mode: Vi|~Search, action: ScrollHalfPageDown }
|
||||||
|
#- { key: Y, mode: Vi|~Search, action: Copy }
|
||||||
|
#- { key: Y, mode: Vi|~Search, action: ClearSelection }
|
||||||
|
#- { key: Copy, mode: Vi|~Search, action: ClearSelection }
|
||||||
|
#- { key: V, mode: Vi|~Search, action: ToggleNormalSelection }
|
||||||
|
#- { key: V, mods: Shift, mode: Vi|~Search, action: ToggleLineSelection }
|
||||||
|
#- { key: V, mods: Control, mode: Vi|~Search, action: ToggleBlockSelection }
|
||||||
|
#- { key: V, mods: Alt, mode: Vi|~Search, action: ToggleSemanticSelection }
|
||||||
|
#- { key: Return, mode: Vi|~Search, action: Open }
|
||||||
|
#- { key: Z, mode: Vi|~Search, action: CenterAroundViCursor }
|
||||||
|
#- { key: K, mode: Vi|~Search, action: Up }
|
||||||
|
#- { key: J, mode: Vi|~Search, action: Down }
|
||||||
|
#- { key: H, mode: Vi|~Search, action: Left }
|
||||||
|
#- { key: L, mode: Vi|~Search, action: Right }
|
||||||
|
#- { key: Up, mode: Vi|~Search, action: Up }
|
||||||
|
#- { key: Down, mode: Vi|~Search, action: Down }
|
||||||
|
#- { key: Left, mode: Vi|~Search, action: Left }
|
||||||
|
#- { key: Right, mode: Vi|~Search, action: Right }
|
||||||
|
#- { key: Key0, mode: Vi|~Search, action: First }
|
||||||
|
#- { key: Key4, mods: Shift, mode: Vi|~Search, action: Last }
|
||||||
|
#- { key: Key6, mods: Shift, mode: Vi|~Search, action: FirstOccupied }
|
||||||
|
#- { key: H, mods: Shift, mode: Vi|~Search, action: High }
|
||||||
|
#- { key: M, mods: Shift, mode: Vi|~Search, action: Middle }
|
||||||
|
#- { key: L, mods: Shift, mode: Vi|~Search, action: Low }
|
||||||
|
#- { key: B, mode: Vi|~Search, action: SemanticLeft }
|
||||||
|
#- { key: W, mode: Vi|~Search, action: SemanticRight }
|
||||||
|
#- { key: E, mode: Vi|~Search, action: SemanticRightEnd }
|
||||||
|
#- { key: B, mods: Shift, mode: Vi|~Search, action: WordLeft }
|
||||||
|
#- { key: W, mods: Shift, mode: Vi|~Search, action: WordRight }
|
||||||
|
#- { key: E, mods: Shift, mode: Vi|~Search, action: WordRightEnd }
|
||||||
|
#- { key: Key5, mods: Shift, mode: Vi|~Search, action: Bracket }
|
||||||
|
#- { key: Slash, mode: Vi|~Search, action: SearchForward }
|
||||||
|
#- { key: Slash, mods: Shift, mode: Vi|~Search, action: SearchBackward }
|
||||||
|
#- { key: N, mode: Vi|~Search, action: SearchNext }
|
||||||
|
#- { key: N, mods: Shift, mode: Vi|~Search, action: SearchPrevious }
|
||||||
|
|
||||||
|
# Search Mode
|
||||||
|
#- { key: Return, mode: Search|Vi, action: SearchConfirm }
|
||||||
|
#- { key: Escape, mode: Search, action: SearchCancel }
|
||||||
|
#- { key: C, mods: Control, mode: Search, action: SearchCancel }
|
||||||
|
#- { key: U, mods: Control, mode: Search, action: SearchClear }
|
||||||
|
#- { key: W, mods: Control, mode: Search, action: SearchDeleteWord }
|
||||||
|
#- { key: P, mods: Control, mode: Search, action: SearchHistoryPrevious }
|
||||||
|
#- { key: N, mods: Control, mode: Search, action: SearchHistoryNext }
|
||||||
|
#- { key: Up, mode: Search, action: SearchHistoryPrevious }
|
||||||
|
#- { key: Down, mode: Search, action: SearchHistoryNext }
|
||||||
|
#- { key: Return, mode: Search|~Vi, action: SearchFocusNext }
|
||||||
|
#- { key: Return, mods: Shift, mode: Search|~Vi, action: SearchFocusPrevious }
|
||||||
|
|
||||||
|
# (Windows, Linux, and BSD only)
|
||||||
|
#- { key: V, mods: Control|Shift, mode: ~Vi, action: Paste }
|
||||||
|
#- { key: C, mods: Control|Shift, action: Copy }
|
||||||
|
#- { key: F, mods: Control|Shift, mode: ~Search, action: SearchForward }
|
||||||
|
#- { key: B, mods: Control|Shift, mode: ~Search, action: SearchBackward }
|
||||||
|
#- { key: C, mods: Control|Shift, mode: Vi|~Search, action: ClearSelection }
|
||||||
|
#- { key: Insert, mods: Shift, action: PasteSelection }
|
||||||
|
#- { key: Key0, mods: Control, action: ResetFontSize }
|
||||||
|
#- { key: Equals, mods: Control, action: IncreaseFontSize }
|
||||||
|
#- { key: Plus, mods: Control, action: IncreaseFontSize }
|
||||||
|
#- { key: NumpadAdd, mods: Control, action: IncreaseFontSize }
|
||||||
|
#- { key: Minus, mods: Control, action: DecreaseFontSize }
|
||||||
|
#- { key: NumpadSubtract, mods: Control, action: DecreaseFontSize }
|
||||||
|
|
||||||
|
# (Windows only)
|
||||||
|
#- { key: Return, mods: Alt, action: ToggleFullscreen }
|
||||||
|
|
||||||
|
# (macOS only)
|
||||||
|
#- { key: K, mods: Command, mode: ~Vi|~Search, chars: "\x0c" }
|
||||||
|
#- { key: K, mods: Command, mode: ~Vi|~Search, action: ClearHistory }
|
||||||
|
#- { key: Key0, mods: Command, action: ResetFontSize }
|
||||||
|
#- { key: Equals, mods: Command, action: IncreaseFontSize }
|
||||||
|
#- { key: Plus, mods: Command, action: IncreaseFontSize }
|
||||||
|
#- { key: NumpadAdd, mods: Command, action: IncreaseFontSize }
|
||||||
|
#- { key: Minus, mods: Command, action: DecreaseFontSize }
|
||||||
|
#- { key: NumpadSubtract, mods: Command, action: DecreaseFontSize }
|
||||||
|
#- { key: V, mods: Command, action: Paste }
|
||||||
|
#- { key: C, mods: Command, action: Copy }
|
||||||
|
#- { key: C, mods: Command, mode: Vi|~Search, action: ClearSelection }
|
||||||
|
#- { key: H, mods: Command, action: Hide }
|
||||||
|
#- { key: H, mods: Command|Alt, action: HideOtherApplications }
|
||||||
|
#- { key: M, mods: Command, action: Minimize }
|
||||||
|
#- { key: Q, mods: Command, action: Quit }
|
||||||
|
#- { key: W, mods: Command, action: Quit }
|
||||||
|
#- { key: N, mods: Command, action: CreateNewWindow }
|
||||||
|
#- { key: F, mods: Command|Control, action: ToggleFullscreen }
|
||||||
|
#- { key: F, mods: Command, mode: ~Search, action: SearchForward }
|
||||||
|
#- { key: B, mods: Command, mode: ~Search, action: SearchBackward }
|
||||||
|
|
||||||
|
#debug:
|
||||||
|
# Display the time it takes to redraw each frame.
|
||||||
|
#render_timer: false
|
||||||
|
|
||||||
|
# Keep the log file after quitting Alacritty.
|
||||||
|
#persistent_logging: false
|
||||||
|
|
||||||
|
# Log level
|
||||||
|
#
|
||||||
|
# Values for `log_level`:
|
||||||
|
# - Off
|
||||||
|
# - Error
|
||||||
|
# - Warn
|
||||||
|
# - Info
|
||||||
|
# - Debug
|
||||||
|
# - Trace
|
||||||
|
#log_level: Warn
|
||||||
|
|
||||||
|
# Renderer override.
|
||||||
|
# - glsl3
|
||||||
|
# - gles2
|
||||||
|
# - gles2_pure
|
||||||
|
#renderer: None
|
||||||
|
|
||||||
|
# Print all received window events.
|
||||||
|
#print_events: false
|
||||||
|
|
||||||
|
# Highlight window damage information.
|
||||||
|
#highlight_damage: false
|
||||||
|
|
||||||
|
import:
|
||||||
|
# uncomment the flavour you want below:
|
||||||
|
- ~/.config/alacritty/catppuccin/catppuccin-mocha.yml
|
||||||
|
# - ~/.config/alacritty/catppuccin/catppuccin-macchiato.yml
|
||||||
|
# - ~/.config/alacritty/catppuccin/catppuccin-frappe.yml
|
||||||
|
# - ~/.config/alacritty/catppuccin/catppuccin-latte.yml
|
1
alacritty/catppuccin
Submodule
1
alacritty/catppuccin
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 3c808cbb4f9c87be43ba5241bc57373c793d2f17
|
|
@ -1,39 +0,0 @@
|
||||||
{
|
|
||||||
runtimeInputs,
|
|
||||||
npmDepsHash,
|
|
||||||
src,
|
|
||||||
lib,
|
|
||||||
buildNpmPackage,
|
|
||||||
makeWrapper,
|
|
||||||
nodejs_latest,
|
|
||||||
jq,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) concatMapStringsSep;
|
|
||||||
inherit (builtins) fromJSON readFile;
|
|
||||||
|
|
||||||
packageJSON = fromJSON (readFile "${src}/package.json");
|
|
||||||
in
|
|
||||||
buildNpmPackage rec {
|
|
||||||
pname = packageJSON.name;
|
|
||||||
inherit (packageJSON) version;
|
|
||||||
|
|
||||||
inherit src runtimeInputs npmDepsHash;
|
|
||||||
|
|
||||||
prePatch = ''
|
|
||||||
mv ./tsconfig.json ./project.json
|
|
||||||
sed 's/^ *\/\/.*//' ${./config/tsconfig.json} > ./base.json
|
|
||||||
${jq}/bin/jq -sr '.[0] * .[1] | del(.extends)' ./project.json ./base.json > ./tsconfig.json
|
|
||||||
rm base.json project.json
|
|
||||||
'';
|
|
||||||
|
|
||||||
nativeBuildInputs = [makeWrapper];
|
|
||||||
|
|
||||||
postInstall = ''
|
|
||||||
wrapProgram $out/bin/${pname} \
|
|
||||||
--prefix PATH : ${concatMapStringsSep ":" (p: p + "/bin") runtimeInputs}
|
|
||||||
'';
|
|
||||||
|
|
||||||
nodejs = nodejs_latest;
|
|
||||||
meta.mainProgram = pname;
|
|
||||||
}
|
|
|
@ -1,2 +0,0 @@
|
||||||
use flake $FLAKE#node
|
|
||||||
npm ci
|
|
|
@ -1,451 +0,0 @@
|
||||||
import eslint from '@eslint/js';
|
|
||||||
import jsdoc from 'eslint-plugin-jsdoc';
|
|
||||||
import stylistic from '@stylistic/eslint-plugin';
|
|
||||||
import tseslint from 'typescript-eslint';
|
|
||||||
|
|
||||||
|
|
||||||
export default tseslint.config({
|
|
||||||
files: ['**/*.js', '**/*.ts'],
|
|
||||||
ignores: ['node_modules/**', 'types/**'],
|
|
||||||
|
|
||||||
extends: [
|
|
||||||
eslint.configs.recommended,
|
|
||||||
jsdoc.configs['flat/recommended-typescript'],
|
|
||||||
stylistic.configs['recommended-flat'],
|
|
||||||
...tseslint.configs.recommended,
|
|
||||||
...tseslint.configs.stylistic,
|
|
||||||
],
|
|
||||||
|
|
||||||
rules: {
|
|
||||||
// JSDoc settings
|
|
||||||
'jsdoc/tag-lines': ['warn', 'any', { startLines: 1 }],
|
|
||||||
'jsdoc/check-line-alignment': ['warn', 'always', {
|
|
||||||
tags: ['param', 'arg', 'argument', 'property', 'prop'],
|
|
||||||
}],
|
|
||||||
'jsdoc/no-types': 'off',
|
|
||||||
|
|
||||||
// Newer settings
|
|
||||||
'@typescript-eslint/no-extraneous-class': ['off'],
|
|
||||||
'@typescript-eslint/no-implied-eval': ['off'],
|
|
||||||
'class-methods-use-this': 'off',
|
|
||||||
'@stylistic/no-multiple-empty-lines': 'off',
|
|
||||||
|
|
||||||
// Pre-flat config
|
|
||||||
'@typescript-eslint/no-unused-vars': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
args: 'all',
|
|
||||||
argsIgnorePattern: '^_',
|
|
||||||
caughtErrors: 'all',
|
|
||||||
caughtErrorsIgnorePattern: '^_',
|
|
||||||
destructuredArrayIgnorePattern: '^_',
|
|
||||||
varsIgnorePattern: '^_',
|
|
||||||
ignoreRestSiblings: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
'array-callback-return': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
allowImplicit: true,
|
|
||||||
checkForEach: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'no-constructor-return': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-unreachable-loop': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
ignore: [
|
|
||||||
'ForInStatement',
|
|
||||||
'ForOfStatement',
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'no-use-before-define': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
functions: false,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'block-scoped-var': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'curly': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'default-case-last': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'default-param-last': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'eqeqeq': [
|
|
||||||
'error',
|
|
||||||
'smart',
|
|
||||||
],
|
|
||||||
'func-names': [
|
|
||||||
'warn',
|
|
||||||
'never',
|
|
||||||
],
|
|
||||||
'func-style': [
|
|
||||||
'warn',
|
|
||||||
'expression',
|
|
||||||
],
|
|
||||||
'logical-assignment-operators': [
|
|
||||||
'warn',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'no-array-constructor': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-empty-function': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-empty-static-block': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-extend-native': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-extra-bind': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-implicit-coercion': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-iterator': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-labels': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-lone-blocks': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-lonely-if': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-loop-func': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-magic-numbers': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
ignore: [
|
|
||||||
-1,
|
|
||||||
0.1,
|
|
||||||
0,
|
|
||||||
1,
|
|
||||||
2,
|
|
||||||
3,
|
|
||||||
4,
|
|
||||||
5,
|
|
||||||
10,
|
|
||||||
12,
|
|
||||||
33,
|
|
||||||
66,
|
|
||||||
100,
|
|
||||||
255,
|
|
||||||
360,
|
|
||||||
450,
|
|
||||||
500,
|
|
||||||
1000,
|
|
||||||
],
|
|
||||||
ignoreDefaultValues: true,
|
|
||||||
ignoreClassFieldInitialValues: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'no-multi-assign': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-new-wrappers': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-object-constructor': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-proto': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-return-assign': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-sequences': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-shadow': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
builtinGlobals: true,
|
|
||||||
allow: [
|
|
||||||
'Window',
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'no-undef-init': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-undefined': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-useless-constructor': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-useless-escape': [
|
|
||||||
'off',
|
|
||||||
],
|
|
||||||
'no-useless-return': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-var': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'no-void': [
|
|
||||||
'off',
|
|
||||||
],
|
|
||||||
'no-with': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'object-shorthand': [
|
|
||||||
'error',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'one-var': [
|
|
||||||
'error',
|
|
||||||
'never',
|
|
||||||
],
|
|
||||||
'operator-assignment': [
|
|
||||||
'warn',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'prefer-arrow-callback': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'prefer-const': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'prefer-object-has-own': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'prefer-regex-literals': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'prefer-template': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'no-prototype-builtins': 'off',
|
|
||||||
'@typescript-eslint/no-var-requires': [
|
|
||||||
'off',
|
|
||||||
],
|
|
||||||
'@stylistic/array-bracket-newline': [
|
|
||||||
'warn',
|
|
||||||
'consistent',
|
|
||||||
],
|
|
||||||
'@stylistic/array-bracket-spacing': [
|
|
||||||
'warn',
|
|
||||||
'never',
|
|
||||||
],
|
|
||||||
'@stylistic/arrow-parens': [
|
|
||||||
'warn',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'@stylistic/brace-style': [
|
|
||||||
'warn',
|
|
||||||
'stroustrup',
|
|
||||||
],
|
|
||||||
'@stylistic/comma-dangle': [
|
|
||||||
'warn',
|
|
||||||
'always-multiline',
|
|
||||||
],
|
|
||||||
'@stylistic/comma-spacing': [
|
|
||||||
'warn',
|
|
||||||
{
|
|
||||||
before: false,
|
|
||||||
after: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/comma-style': [
|
|
||||||
'error',
|
|
||||||
'last',
|
|
||||||
],
|
|
||||||
'@stylistic/dot-location': [
|
|
||||||
'error',
|
|
||||||
'property',
|
|
||||||
],
|
|
||||||
'@stylistic/function-call-argument-newline': [
|
|
||||||
'warn',
|
|
||||||
'consistent',
|
|
||||||
],
|
|
||||||
'@stylistic/function-paren-newline': [
|
|
||||||
'warn',
|
|
||||||
'consistent',
|
|
||||||
],
|
|
||||||
'@stylistic/indent': [
|
|
||||||
'warn',
|
|
||||||
4,
|
|
||||||
{
|
|
||||||
SwitchCase: 1,
|
|
||||||
ignoreComments: true,
|
|
||||||
ignoredNodes: ['TemplateLiteral > *'],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/key-spacing': [
|
|
||||||
'warn',
|
|
||||||
{
|
|
||||||
beforeColon: false,
|
|
||||||
afterColon: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/keyword-spacing': [
|
|
||||||
'warn',
|
|
||||||
{
|
|
||||||
before: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/linebreak-style': [
|
|
||||||
'error',
|
|
||||||
'unix',
|
|
||||||
],
|
|
||||||
'@stylistic/lines-between-class-members': [
|
|
||||||
'warn',
|
|
||||||
'always',
|
|
||||||
{
|
|
||||||
exceptAfterSingleLine: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/max-len': [
|
|
||||||
'warn',
|
|
||||||
{
|
|
||||||
code: 105,
|
|
||||||
ignoreComments: true,
|
|
||||||
ignoreTrailingComments: true,
|
|
||||||
ignoreUrls: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/multiline-ternary': [
|
|
||||||
'warn',
|
|
||||||
'always-multiline',
|
|
||||||
],
|
|
||||||
'@stylistic/new-parens': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'@stylistic/no-mixed-operators': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'@stylistic/no-mixed-spaces-and-tabs': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'@stylistic/no-multi-spaces': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'@stylistic/no-tabs': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'@stylistic/no-trailing-spaces': [
|
|
||||||
'error',
|
|
||||||
],
|
|
||||||
'@stylistic/no-whitespace-before-property': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'@stylistic/nonblock-statement-body-position': [
|
|
||||||
'error',
|
|
||||||
'below',
|
|
||||||
],
|
|
||||||
'@stylistic/object-curly-newline': [
|
|
||||||
'warn',
|
|
||||||
{
|
|
||||||
consistent: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/object-curly-spacing': [
|
|
||||||
'warn',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'@stylistic/operator-linebreak': [
|
|
||||||
'warn',
|
|
||||||
'after',
|
|
||||||
],
|
|
||||||
'@stylistic/padded-blocks': [
|
|
||||||
'error',
|
|
||||||
'never',
|
|
||||||
],
|
|
||||||
'@stylistic/padding-line-between-statements': [
|
|
||||||
'warn',
|
|
||||||
{
|
|
||||||
blankLine: 'always',
|
|
||||||
prev: '*',
|
|
||||||
next: 'return',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
blankLine: 'always',
|
|
||||||
prev: [
|
|
||||||
'const',
|
|
||||||
'let',
|
|
||||||
'var',
|
|
||||||
],
|
|
||||||
next: '*',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
blankLine: 'any',
|
|
||||||
prev: [
|
|
||||||
'const',
|
|
||||||
'let',
|
|
||||||
'var',
|
|
||||||
],
|
|
||||||
next: [
|
|
||||||
'const',
|
|
||||||
'let',
|
|
||||||
'var',
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
blankLine: 'always',
|
|
||||||
prev: [
|
|
||||||
'case',
|
|
||||||
'default',
|
|
||||||
],
|
|
||||||
next: '*',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/quote-props': [
|
|
||||||
'error',
|
|
||||||
'consistent-as-needed',
|
|
||||||
],
|
|
||||||
'@stylistic/quotes': [
|
|
||||||
'error',
|
|
||||||
'single',
|
|
||||||
{
|
|
||||||
avoidEscape: true,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@stylistic/semi': [
|
|
||||||
'error',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'@stylistic/semi-spacing': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'@stylistic/space-before-blocks': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'@stylistic/space-before-function-paren': [
|
|
||||||
'warn',
|
|
||||||
'never',
|
|
||||||
],
|
|
||||||
'@stylistic/space-infix-ops': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'@stylistic/spaced-comment': [
|
|
||||||
'warn',
|
|
||||||
'always',
|
|
||||||
],
|
|
||||||
'@stylistic/switch-colon-spacing': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
'@stylistic/wrap-regex': [
|
|
||||||
'warn',
|
|
||||||
],
|
|
||||||
},
|
|
||||||
});
|
|
|
@ -1,3 +0,0 @@
|
||||||
import eslintConf from './eslint.config';
|
|
||||||
|
|
||||||
export default eslintConf;
|
|
BIN
apps/config/package-lock.json
generated
BIN
apps/config/package-lock.json
generated
Binary file not shown.
|
@ -1,16 +0,0 @@
|
||||||
{
|
|
||||||
"name": "eslint-conf",
|
|
||||||
"version": "0.0.0",
|
|
||||||
"type": "module",
|
|
||||||
"exports": "./index.ts",
|
|
||||||
"devDependencies": {
|
|
||||||
"@eslint/js": "9.17.0",
|
|
||||||
"@stylistic/eslint-plugin": "2.12.1",
|
|
||||||
"eslint": "9.17.0",
|
|
||||||
"eslint-plugin-jsdoc": "50.6.1",
|
|
||||||
"jiti": "2.4.2",
|
|
||||||
"pkg-types": "1.3.0",
|
|
||||||
"typescript": "5.7.3",
|
|
||||||
"typescript-eslint": "8.19.1"
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,25 +0,0 @@
|
||||||
{
|
|
||||||
"$schema": "https://json.schemastore.org/tsconfig",
|
|
||||||
"compilerOptions": {
|
|
||||||
// Env
|
|
||||||
"target": "ESNext",
|
|
||||||
"lib": [
|
|
||||||
"ESNext"
|
|
||||||
],
|
|
||||||
// Module
|
|
||||||
"module": "nodenext",
|
|
||||||
"moduleResolution": "bundler",
|
|
||||||
"baseUrl": ".",
|
|
||||||
// Emit
|
|
||||||
"noEmit": true,
|
|
||||||
"newLine": "LF",
|
|
||||||
// Interop
|
|
||||||
"esModuleInterop": true,
|
|
||||||
"allowSyntheticDefaultImports": true,
|
|
||||||
"forceConsistentCasingInFileNames": true,
|
|
||||||
"isolatedModules": true,
|
|
||||||
// Type Checking
|
|
||||||
"strict": true,
|
|
||||||
"noImplicitAny": false
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,10 +0,0 @@
|
||||||
{
|
|
||||||
"$schema": "https://json.schemastore.org/tsconfig",
|
|
||||||
"extends": "./tsconfig.base.json",
|
|
||||||
"includes": [
|
|
||||||
"*.ts",
|
|
||||||
"**/*.ts",
|
|
||||||
"*.js",
|
|
||||||
"**/*.js"
|
|
||||||
]
|
|
||||||
}
|
|
|
@ -1,20 +0,0 @@
|
||||||
{
|
|
||||||
inputs,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (pkgs.lib) getExe listToAttrs nameValuePair;
|
|
||||||
|
|
||||||
buildApp = attrs: (pkgs.callPackage ./buildApp.nix ({} // inputs // attrs));
|
|
||||||
|
|
||||||
mkApp = file: {
|
|
||||||
program = getExe (pkgs.callPackage file ({inherit buildApp;} // inputs));
|
|
||||||
type = "app";
|
|
||||||
};
|
|
||||||
|
|
||||||
mkApps = apps: listToAttrs (map (x: nameValuePair x (mkApp ./${x})) apps);
|
|
||||||
in
|
|
||||||
mkApps [
|
|
||||||
"extract-subs"
|
|
||||||
"update-sources"
|
|
||||||
]
|
|
|
@ -1,2 +0,0 @@
|
||||||
use flake $FLAKE#subtitles-dev
|
|
||||||
npm ci
|
|
|
@ -1,13 +0,0 @@
|
||||||
{
|
|
||||||
buildApp,
|
|
||||||
ffmpeg-full,
|
|
||||||
...
|
|
||||||
}:
|
|
||||||
buildApp {
|
|
||||||
src = ./.;
|
|
||||||
npmDepsHash = "sha256-VPvEbpLIuPx7Oax5UNEIf8/Gd34UMZkuQGt4bCnUwQQ=";
|
|
||||||
|
|
||||||
runtimeInputs = [
|
|
||||||
ffmpeg-full
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,3 +0,0 @@
|
||||||
import eslintConf from 'eslint-conf';
|
|
||||||
|
|
||||||
export default eslintConf;
|
|
BIN
apps/extract-subs/package-lock.json
generated
BIN
apps/extract-subs/package-lock.json
generated
Binary file not shown.
|
@ -1,21 +0,0 @@
|
||||||
{
|
|
||||||
"name": "extract-subs",
|
|
||||||
"version": "0.0.0",
|
|
||||||
"bin": "out/bin/app.cjs",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"build": "node_ver=$(node -v); esbuild src/app.ts --bundle --platform=node --target=\"node${node_ver:1:2}\" --outfile=out/bin/app.cjs"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
|
||||||
"@types/fluent-ffmpeg": "2.1.27",
|
|
||||||
"fluent-ffmpeg": "2.1.3"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"eslint-conf": "file:../config",
|
|
||||||
"@types/node": "22.10.5",
|
|
||||||
"esbuild": "0.24.2",
|
|
||||||
"eslint": "9.17.0",
|
|
||||||
"jiti": "2.4.2",
|
|
||||||
"typescript": "5.7.3"
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,157 +0,0 @@
|
||||||
import { spawnSync as spawn } from 'child_process';
|
|
||||||
|
|
||||||
import ffprobe from './ffprobe';
|
|
||||||
import { ISO6393To1 } from './lang-codes';
|
|
||||||
|
|
||||||
/* Types */
|
|
||||||
import { FfprobeStream } from 'fluent-ffmpeg';
|
|
||||||
|
|
||||||
|
|
||||||
const SPAWN_OPTS = {
|
|
||||||
stdio: [process.stdin, process.stdout, process.stderr],
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* These are the cli arguments
|
|
||||||
*
|
|
||||||
* @param videoPath the directory in which we want to sync the subtitles
|
|
||||||
* @param languages a comma-separated list of languages (3 letters) to sync the subtitles
|
|
||||||
*/
|
|
||||||
const video = process.argv[2];
|
|
||||||
const languages = process.argv[3]?.split(',');
|
|
||||||
|
|
||||||
|
|
||||||
// Global Vars
|
|
||||||
const subIndexes: number[] = [];
|
|
||||||
let videoPath: string;
|
|
||||||
let baseName: string;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the relative path to the subtitle file of a ffmpeg stream.
|
|
||||||
*
|
|
||||||
* @param sub the stream of the subtitles to extract
|
|
||||||
* @returns the path of the subtitle file
|
|
||||||
*/
|
|
||||||
const getSubPath = (sub: FfprobeStream): string => {
|
|
||||||
const language = ISO6393To1.get(sub.tags.language);
|
|
||||||
|
|
||||||
const forced = sub.disposition?.forced === 0 ?
|
|
||||||
'' :
|
|
||||||
'.forced';
|
|
||||||
|
|
||||||
const hearingImpaired = sub.disposition?.hearing_impaired === 0 ?
|
|
||||||
'' :
|
|
||||||
'.sdh';
|
|
||||||
|
|
||||||
return `${baseName}${forced}.${language}${hearingImpaired}.srt`;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Removes all subtitles streams from the video file.
|
|
||||||
*/
|
|
||||||
const removeContainerSubs = (): void => {
|
|
||||||
spawn('mv', [
|
|
||||||
videoPath,
|
|
||||||
`${videoPath}.bak`,
|
|
||||||
], SPAWN_OPTS);
|
|
||||||
|
|
||||||
spawn('ffmpeg', [
|
|
||||||
'-i', `${videoPath}.bak`,
|
|
||||||
'-map', '0',
|
|
||||||
...subIndexes.map((i) => ['-map', `-0:${i}`]).flat(),
|
|
||||||
'-c', 'copy', videoPath,
|
|
||||||
], SPAWN_OPTS);
|
|
||||||
|
|
||||||
spawn('rm', [
|
|
||||||
`${videoPath}.bak`,
|
|
||||||
], SPAWN_OPTS);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Extracts a sub of a video file to a subtitle file.
|
|
||||||
*
|
|
||||||
* @param sub the stream of the subtitles to extract
|
|
||||||
*/
|
|
||||||
const extractSub = (sub: FfprobeStream): void => {
|
|
||||||
const subFile = getSubPath(sub);
|
|
||||||
|
|
||||||
spawn('ffmpeg', [
|
|
||||||
'-i', videoPath,
|
|
||||||
'-map', `0:${sub.index}`, subFile,
|
|
||||||
], SPAWN_OPTS);
|
|
||||||
|
|
||||||
subIndexes.push(sub.index);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sorts the list of streams to only keep subtitles
|
|
||||||
* that can be extracted.
|
|
||||||
*
|
|
||||||
* @param lang the language of the subtitles
|
|
||||||
* @param streams the streams
|
|
||||||
* @returns the streams that represent subtitles
|
|
||||||
*/
|
|
||||||
const findSubs = (
|
|
||||||
lang: string,
|
|
||||||
streams: FfprobeStream[],
|
|
||||||
): FfprobeStream[] => {
|
|
||||||
const subs = streams.filter((s) => s.tags?.language &&
|
|
||||||
s.tags.language === lang &&
|
|
||||||
s.codec_type === 'subtitle');
|
|
||||||
|
|
||||||
const pgs = subs.filter((s) => s.codec_name === 'hdmv_pgs_subtitle');
|
|
||||||
|
|
||||||
// If we only have PGS subs, warn user
|
|
||||||
if (pgs.length === subs.length) {
|
|
||||||
console.warn(`No SRT subtitle tracks were found for ${lang}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove PGS streams from subs
|
|
||||||
return subs.filter((s) => s.codec_name !== 'hdmv_pgs_subtitle');
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Where the magic happens.
|
|
||||||
*/
|
|
||||||
const main = async(): Promise<void> => {
|
|
||||||
// Get rid of video extension
|
|
||||||
baseName = videoPath.split('/').at(-1)!.replace(/\.[^.]*$/, '');
|
|
||||||
|
|
||||||
// ffprobe the video file to see available sub tracks
|
|
||||||
const data = await ffprobe(videoPath);
|
|
||||||
|
|
||||||
if (!data?.streams) {
|
|
||||||
console.error('Couldn\'t find streams in video file');
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check for languages wanted
|
|
||||||
languages.forEach((lang) => {
|
|
||||||
const subs = findSubs(lang, data.streams);
|
|
||||||
|
|
||||||
if (subs.length === 0) {
|
|
||||||
console.warn(`No subtitle tracks were found for ${lang}`);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Extract all subs
|
|
||||||
subs.forEach((sub) => { extractSub(sub); });
|
|
||||||
});
|
|
||||||
|
|
||||||
removeContainerSubs();
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
// Check if there are 2 params
|
|
||||||
if (video && languages) {
|
|
||||||
videoPath = video;
|
|
||||||
main();
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
console.error('Error: no argument passed');
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
|
@ -1,8 +0,0 @@
|
||||||
import Ffmpeg from 'fluent-ffmpeg';
|
|
||||||
|
|
||||||
|
|
||||||
export default (videoPath: string) => new Promise<Ffmpeg.FfprobeData>((resolve) => {
|
|
||||||
Ffmpeg.ffprobe(videoPath, (_e, data) => {
|
|
||||||
resolve(data);
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,373 +0,0 @@
|
||||||
export const ISO6391To3 = new Map([
|
|
||||||
['aa', 'aar'],
|
|
||||||
['ab', 'abk'],
|
|
||||||
['af', 'afr'],
|
|
||||||
['ak', 'aka'],
|
|
||||||
['am', 'amh'],
|
|
||||||
['ar', 'ara'],
|
|
||||||
['an', 'arg'],
|
|
||||||
['as', 'asm'],
|
|
||||||
['av', 'ava'],
|
|
||||||
['ae', 'ave'],
|
|
||||||
['ay', 'aym'],
|
|
||||||
['az', 'aze'],
|
|
||||||
['ba', 'bak'],
|
|
||||||
['bm', 'bam'],
|
|
||||||
['be', 'bel'],
|
|
||||||
['bn', 'ben'],
|
|
||||||
['bi', 'bis'],
|
|
||||||
['bo', 'bod'],
|
|
||||||
['bs', 'bos'],
|
|
||||||
['br', 'bre'],
|
|
||||||
['bg', 'bul'],
|
|
||||||
['ca', 'cat'],
|
|
||||||
['cs', 'ces'],
|
|
||||||
['ch', 'cha'],
|
|
||||||
['ce', 'che'],
|
|
||||||
['cu', 'chu'],
|
|
||||||
['cv', 'chv'],
|
|
||||||
['kw', 'cor'],
|
|
||||||
['co', 'cos'],
|
|
||||||
['cr', 'cre'],
|
|
||||||
['cy', 'cym'],
|
|
||||||
['da', 'dan'],
|
|
||||||
['de', 'deu'],
|
|
||||||
['dv', 'div'],
|
|
||||||
['dz', 'dzo'],
|
|
||||||
['el', 'ell'],
|
|
||||||
['en', 'eng'],
|
|
||||||
['eo', 'epo'],
|
|
||||||
['et', 'est'],
|
|
||||||
['eu', 'eus'],
|
|
||||||
['ee', 'ewe'],
|
|
||||||
['fo', 'fao'],
|
|
||||||
['fa', 'fas'],
|
|
||||||
['fj', 'fij'],
|
|
||||||
['fi', 'fin'],
|
|
||||||
['fr', 'fre'],
|
|
||||||
['fy', 'fry'],
|
|
||||||
['ff', 'ful'],
|
|
||||||
['gd', 'gla'],
|
|
||||||
['ga', 'gle'],
|
|
||||||
['gl', 'glg'],
|
|
||||||
['gv', 'glv'],
|
|
||||||
['gn', 'grn'],
|
|
||||||
['gu', 'guj'],
|
|
||||||
['ht', 'hat'],
|
|
||||||
['ha', 'hau'],
|
|
||||||
['sh', 'hbs'],
|
|
||||||
['he', 'heb'],
|
|
||||||
['hz', 'her'],
|
|
||||||
['hi', 'hin'],
|
|
||||||
['ho', 'hmo'],
|
|
||||||
['hr', 'hrv'],
|
|
||||||
['hu', 'hun'],
|
|
||||||
['hy', 'hye'],
|
|
||||||
['ig', 'ibo'],
|
|
||||||
['io', 'ido'],
|
|
||||||
['ii', 'iii'],
|
|
||||||
['iu', 'iku'],
|
|
||||||
['ie', 'ile'],
|
|
||||||
['ia', 'ina'],
|
|
||||||
['id', 'ind'],
|
|
||||||
['ik', 'ipk'],
|
|
||||||
['is', 'isl'],
|
|
||||||
['it', 'ita'],
|
|
||||||
['jv', 'jav'],
|
|
||||||
['ja', 'jpn'],
|
|
||||||
['kl', 'kal'],
|
|
||||||
['kn', 'kan'],
|
|
||||||
['ks', 'kas'],
|
|
||||||
['ka', 'kat'],
|
|
||||||
['kr', 'kau'],
|
|
||||||
['kk', 'kaz'],
|
|
||||||
['km', 'khm'],
|
|
||||||
['ki', 'kik'],
|
|
||||||
['rw', 'kin'],
|
|
||||||
['ky', 'kir'],
|
|
||||||
['kv', 'kom'],
|
|
||||||
['kg', 'kon'],
|
|
||||||
['ko', 'kor'],
|
|
||||||
['kj', 'kua'],
|
|
||||||
['ku', 'kur'],
|
|
||||||
['lo', 'lao'],
|
|
||||||
['la', 'lat'],
|
|
||||||
['lv', 'lav'],
|
|
||||||
['li', 'lim'],
|
|
||||||
['ln', 'lin'],
|
|
||||||
['lt', 'lit'],
|
|
||||||
['lb', 'ltz'],
|
|
||||||
['lu', 'lub'],
|
|
||||||
['lg', 'lug'],
|
|
||||||
['mh', 'mah'],
|
|
||||||
['ml', 'mal'],
|
|
||||||
['mr', 'mar'],
|
|
||||||
['mk', 'mkd'],
|
|
||||||
['mg', 'mlg'],
|
|
||||||
['mt', 'mlt'],
|
|
||||||
['mn', 'mon'],
|
|
||||||
['mi', 'mri'],
|
|
||||||
['ms', 'msa'],
|
|
||||||
['my', 'mya'],
|
|
||||||
['na', 'nau'],
|
|
||||||
['nv', 'nav'],
|
|
||||||
['nr', 'nbl'],
|
|
||||||
['nd', 'nde'],
|
|
||||||
['ng', 'ndo'],
|
|
||||||
['ne', 'nep'],
|
|
||||||
['nl', 'nld'],
|
|
||||||
['nn', 'nno'],
|
|
||||||
['nb', 'nob'],
|
|
||||||
['no', 'nor'],
|
|
||||||
['ny', 'nya'],
|
|
||||||
['oc', 'oci'],
|
|
||||||
['oj', 'oji'],
|
|
||||||
['or', 'ori'],
|
|
||||||
['om', 'orm'],
|
|
||||||
['os', 'oss'],
|
|
||||||
['pa', 'pan'],
|
|
||||||
['pi', 'pli'],
|
|
||||||
['pl', 'pol'],
|
|
||||||
['pt', 'por'],
|
|
||||||
['ps', 'pus'],
|
|
||||||
['qu', 'que'],
|
|
||||||
['rm', 'roh'],
|
|
||||||
['ro', 'ron'],
|
|
||||||
['rn', 'run'],
|
|
||||||
['ru', 'rus'],
|
|
||||||
['sg', 'sag'],
|
|
||||||
['sa', 'san'],
|
|
||||||
['si', 'sin'],
|
|
||||||
['sk', 'slk'],
|
|
||||||
['sl', 'slv'],
|
|
||||||
['se', 'sme'],
|
|
||||||
['sm', 'smo'],
|
|
||||||
['sn', 'sna'],
|
|
||||||
['sd', 'snd'],
|
|
||||||
['so', 'som'],
|
|
||||||
['st', 'sot'],
|
|
||||||
['es', 'spa'],
|
|
||||||
['sq', 'sqi'],
|
|
||||||
['sc', 'srd'],
|
|
||||||
['sr', 'srp'],
|
|
||||||
['ss', 'ssw'],
|
|
||||||
['su', 'sun'],
|
|
||||||
['sw', 'swa'],
|
|
||||||
['sv', 'swe'],
|
|
||||||
['ty', 'tah'],
|
|
||||||
['ta', 'tam'],
|
|
||||||
['tt', 'tat'],
|
|
||||||
['te', 'tel'],
|
|
||||||
['tg', 'tgk'],
|
|
||||||
['tl', 'tgl'],
|
|
||||||
['th', 'tha'],
|
|
||||||
['ti', 'tir'],
|
|
||||||
['to', 'ton'],
|
|
||||||
['tn', 'tsn'],
|
|
||||||
['ts', 'tso'],
|
|
||||||
['tk', 'tuk'],
|
|
||||||
['tr', 'tur'],
|
|
||||||
['tw', 'twi'],
|
|
||||||
['ug', 'uig'],
|
|
||||||
['uk', 'ukr'],
|
|
||||||
['ur', 'urd'],
|
|
||||||
['uz', 'uzb'],
|
|
||||||
['ve', 'ven'],
|
|
||||||
['vi', 'vie'],
|
|
||||||
['vo', 'vol'],
|
|
||||||
['wa', 'wln'],
|
|
||||||
['wo', 'wol'],
|
|
||||||
['xh', 'xho'],
|
|
||||||
['yi', 'yid'],
|
|
||||||
['yo', 'yor'],
|
|
||||||
['za', 'zha'],
|
|
||||||
['zh', 'zho'],
|
|
||||||
['zu', 'zul'],
|
|
||||||
]);
|
|
||||||
|
|
||||||
export const ISO6393To1 = new Map([
|
|
||||||
['aar', 'aa'],
|
|
||||||
['abk', 'ab'],
|
|
||||||
['afr', 'af'],
|
|
||||||
['aka', 'ak'],
|
|
||||||
['amh', 'am'],
|
|
||||||
['ara', 'ar'],
|
|
||||||
['arg', 'an'],
|
|
||||||
['asm', 'as'],
|
|
||||||
['ava', 'av'],
|
|
||||||
['ave', 'ae'],
|
|
||||||
['aym', 'ay'],
|
|
||||||
['aze', 'az'],
|
|
||||||
['bak', 'ba'],
|
|
||||||
['bam', 'bm'],
|
|
||||||
['bel', 'be'],
|
|
||||||
['ben', 'bn'],
|
|
||||||
['bis', 'bi'],
|
|
||||||
['bod', 'bo'],
|
|
||||||
['bos', 'bs'],
|
|
||||||
['bre', 'br'],
|
|
||||||
['bul', 'bg'],
|
|
||||||
['cat', 'ca'],
|
|
||||||
['ces', 'cs'],
|
|
||||||
['cha', 'ch'],
|
|
||||||
['che', 'ce'],
|
|
||||||
['chu', 'cu'],
|
|
||||||
['chv', 'cv'],
|
|
||||||
['cor', 'kw'],
|
|
||||||
['cos', 'co'],
|
|
||||||
['cre', 'cr'],
|
|
||||||
['cym', 'cy'],
|
|
||||||
['dan', 'da'],
|
|
||||||
['deu', 'de'],
|
|
||||||
['div', 'dv'],
|
|
||||||
['dzo', 'dz'],
|
|
||||||
['ell', 'el'],
|
|
||||||
['eng', 'en'],
|
|
||||||
['epo', 'eo'],
|
|
||||||
['est', 'et'],
|
|
||||||
['eus', 'eu'],
|
|
||||||
['ewe', 'ee'],
|
|
||||||
['fao', 'fo'],
|
|
||||||
['fas', 'fa'],
|
|
||||||
['fij', 'fj'],
|
|
||||||
['fin', 'fi'],
|
|
||||||
['fre', 'fr'],
|
|
||||||
['fry', 'fy'],
|
|
||||||
['ful', 'ff'],
|
|
||||||
['gla', 'gd'],
|
|
||||||
['gle', 'ga'],
|
|
||||||
['glg', 'gl'],
|
|
||||||
['glv', 'gv'],
|
|
||||||
['grn', 'gn'],
|
|
||||||
['guj', 'gu'],
|
|
||||||
['hat', 'ht'],
|
|
||||||
['hau', 'ha'],
|
|
||||||
['hbs', 'sh'],
|
|
||||||
['heb', 'he'],
|
|
||||||
['her', 'hz'],
|
|
||||||
['hin', 'hi'],
|
|
||||||
['hmo', 'ho'],
|
|
||||||
['hrv', 'hr'],
|
|
||||||
['hun', 'hu'],
|
|
||||||
['hye', 'hy'],
|
|
||||||
['ibo', 'ig'],
|
|
||||||
['ido', 'io'],
|
|
||||||
['iii', 'ii'],
|
|
||||||
['iku', 'iu'],
|
|
||||||
['ile', 'ie'],
|
|
||||||
['ina', 'ia'],
|
|
||||||
['ind', 'id'],
|
|
||||||
['ipk', 'ik'],
|
|
||||||
['isl', 'is'],
|
|
||||||
['ita', 'it'],
|
|
||||||
['jav', 'jv'],
|
|
||||||
['jpn', 'ja'],
|
|
||||||
['kal', 'kl'],
|
|
||||||
['kan', 'kn'],
|
|
||||||
['kas', 'ks'],
|
|
||||||
['kat', 'ka'],
|
|
||||||
['kau', 'kr'],
|
|
||||||
['kaz', 'kk'],
|
|
||||||
['khm', 'km'],
|
|
||||||
['kik', 'ki'],
|
|
||||||
['kin', 'rw'],
|
|
||||||
['kir', 'ky'],
|
|
||||||
['kom', 'kv'],
|
|
||||||
['kon', 'kg'],
|
|
||||||
['kor', 'ko'],
|
|
||||||
['kua', 'kj'],
|
|
||||||
['kur', 'ku'],
|
|
||||||
['lao', 'lo'],
|
|
||||||
['lat', 'la'],
|
|
||||||
['lav', 'lv'],
|
|
||||||
['lim', 'li'],
|
|
||||||
['lin', 'ln'],
|
|
||||||
['lit', 'lt'],
|
|
||||||
['ltz', 'lb'],
|
|
||||||
['lub', 'lu'],
|
|
||||||
['lug', 'lg'],
|
|
||||||
['mah', 'mh'],
|
|
||||||
['mal', 'ml'],
|
|
||||||
['mar', 'mr'],
|
|
||||||
['mkd', 'mk'],
|
|
||||||
['mlg', 'mg'],
|
|
||||||
['mlt', 'mt'],
|
|
||||||
['mon', 'mn'],
|
|
||||||
['mri', 'mi'],
|
|
||||||
['msa', 'ms'],
|
|
||||||
['mya', 'my'],
|
|
||||||
['nau', 'na'],
|
|
||||||
['nav', 'nv'],
|
|
||||||
['nbl', 'nr'],
|
|
||||||
['nde', 'nd'],
|
|
||||||
['ndo', 'ng'],
|
|
||||||
['nep', 'ne'],
|
|
||||||
['nld', 'nl'],
|
|
||||||
['nno', 'nn'],
|
|
||||||
['nob', 'nb'],
|
|
||||||
['nor', 'no'],
|
|
||||||
['nya', 'ny'],
|
|
||||||
['oci', 'oc'],
|
|
||||||
['oji', 'oj'],
|
|
||||||
['ori', 'or'],
|
|
||||||
['orm', 'om'],
|
|
||||||
['oss', 'os'],
|
|
||||||
['pan', 'pa'],
|
|
||||||
['pli', 'pi'],
|
|
||||||
['pol', 'pl'],
|
|
||||||
['por', 'pt'],
|
|
||||||
['pus', 'ps'],
|
|
||||||
['que', 'qu'],
|
|
||||||
['roh', 'rm'],
|
|
||||||
['ron', 'ro'],
|
|
||||||
['run', 'rn'],
|
|
||||||
['rus', 'ru'],
|
|
||||||
['sag', 'sg'],
|
|
||||||
['san', 'sa'],
|
|
||||||
['sin', 'si'],
|
|
||||||
['slk', 'sk'],
|
|
||||||
['slv', 'sl'],
|
|
||||||
['sme', 'se'],
|
|
||||||
['smo', 'sm'],
|
|
||||||
['sna', 'sn'],
|
|
||||||
['snd', 'sd'],
|
|
||||||
['som', 'so'],
|
|
||||||
['sot', 'st'],
|
|
||||||
['spa', 'es'],
|
|
||||||
['sqi', 'sq'],
|
|
||||||
['srd', 'sc'],
|
|
||||||
['srp', 'sr'],
|
|
||||||
['ssw', 'ss'],
|
|
||||||
['sun', 'su'],
|
|
||||||
['swa', 'sw'],
|
|
||||||
['swe', 'sv'],
|
|
||||||
['tah', 'ty'],
|
|
||||||
['tam', 'ta'],
|
|
||||||
['tat', 'tt'],
|
|
||||||
['tel', 'te'],
|
|
||||||
['tgk', 'tg'],
|
|
||||||
['tgl', 'tl'],
|
|
||||||
['tha', 'th'],
|
|
||||||
['tir', 'ti'],
|
|
||||||
['ton', 'to'],
|
|
||||||
['tsn', 'tn'],
|
|
||||||
['tso', 'ts'],
|
|
||||||
['tuk', 'tk'],
|
|
||||||
['tur', 'tr'],
|
|
||||||
['twi', 'tw'],
|
|
||||||
['uig', 'ug'],
|
|
||||||
['ukr', 'uk'],
|
|
||||||
['urd', 'ur'],
|
|
||||||
['uzb', 'uz'],
|
|
||||||
['ven', 've'],
|
|
||||||
['vie', 'vi'],
|
|
||||||
['vol', 'vo'],
|
|
||||||
['wln', 'wa'],
|
|
||||||
['wol', 'wo'],
|
|
||||||
['xho', 'xh'],
|
|
||||||
['yid', 'yi'],
|
|
||||||
['yor', 'yo'],
|
|
||||||
['zha', 'za'],
|
|
||||||
['zho', 'zh'],
|
|
||||||
['zul', 'zu'],
|
|
||||||
]);
|
|
|
@ -1,10 +0,0 @@
|
||||||
{
|
|
||||||
"$schema": "https://json.schemastore.org/tsconfig",
|
|
||||||
"extends": "../config/tsconfig.base.json",
|
|
||||||
"includes": [
|
|
||||||
"*.ts",
|
|
||||||
"**/*.ts",
|
|
||||||
"*.js",
|
|
||||||
"**/*.js"
|
|
||||||
]
|
|
||||||
}
|
|
|
@ -1,14 +0,0 @@
|
||||||
{
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (pkgs.lib) mapAttrs removeSuffix;
|
|
||||||
in
|
|
||||||
mapAttrs (
|
|
||||||
name: app: (pkgs.symlinkJoin {
|
|
||||||
name = "app-${name}";
|
|
||||||
paths = [(removeSuffix "/bin/${name}" (toString app.program))];
|
|
||||||
})
|
|
||||||
)
|
|
||||||
(removeAttrs self.apps.${pkgs.system} ["genflake"])
|
|
|
@ -1,2 +0,0 @@
|
||||||
use flake $FLAKE#node
|
|
||||||
npm ci
|
|
|
@ -1,21 +0,0 @@
|
||||||
{
|
|
||||||
buildApp,
|
|
||||||
callPackage,
|
|
||||||
go,
|
|
||||||
nix-update,
|
|
||||||
nodejs_latest,
|
|
||||||
prefetch-npm-deps,
|
|
||||||
...
|
|
||||||
}:
|
|
||||||
buildApp {
|
|
||||||
src = ./.;
|
|
||||||
npmDepsHash = "sha256-nAMu7riJm/6w+ixjzHm4W4YlqOkAwCapM3PHPW2BnnA=";
|
|
||||||
|
|
||||||
runtimeInputs = [
|
|
||||||
go
|
|
||||||
nix-update
|
|
||||||
nodejs_latest
|
|
||||||
prefetch-npm-deps
|
|
||||||
(callPackage ../../modules/docker/updateImage.nix {})
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,3 +0,0 @@
|
||||||
import eslintConf from 'eslint-conf';
|
|
||||||
|
|
||||||
export default eslintConf;
|
|
BIN
apps/update-sources/package-lock.json
generated
BIN
apps/update-sources/package-lock.json
generated
Binary file not shown.
|
@ -1,18 +0,0 @@
|
||||||
{
|
|
||||||
"name": "update-sources",
|
|
||||||
"version": "0.0.0",
|
|
||||||
"bin": "out/bin/app.cjs",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"build": "node_ver=$(node -v); esbuild src/app.ts --bundle --platform=node --target=\"node${node_ver:1:2}\" --outfile=out/bin/app.cjs"
|
|
||||||
},
|
|
||||||
"devDependencies": {
|
|
||||||
"eslint-conf": "file:../config",
|
|
||||||
"@types/node": "22.10.5",
|
|
||||||
"esbuild": "0.24.2",
|
|
||||||
"eslint": "9.17.0",
|
|
||||||
"jiti": "2.4.2",
|
|
||||||
"pkg-types": "1.3.0",
|
|
||||||
"typescript": "5.7.3"
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,175 +0,0 @@
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
import { writeFileSync } from 'node:fs';
|
|
||||||
|
|
||||||
import { parseArgs } from './lib';
|
|
||||||
|
|
||||||
import { updateDocker } from './docker';
|
|
||||||
import { updateFirefoxAddons } from '././firefox';
|
|
||||||
import { updateFlakeInputs } from './flake';
|
|
||||||
import updateNodeModules from './node-modules';
|
|
||||||
|
|
||||||
import {
|
|
||||||
runNixUpdate,
|
|
||||||
updateCaddyPlugins,
|
|
||||||
updateCustomPackage,
|
|
||||||
updateVuetorrent,
|
|
||||||
} from './misc';
|
|
||||||
|
|
||||||
|
|
||||||
/* Constants */
|
|
||||||
const FLAKE = process.env.FLAKE;
|
|
||||||
|
|
||||||
if (!FLAKE) {
|
|
||||||
console.error('Env var FLAKE not found');
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
const args = parseArgs();
|
|
||||||
|
|
||||||
const main = async() => {
|
|
||||||
if (args['d'] || args['docker']) {
|
|
||||||
console.log(updateDocker());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['i'] || args['inputs']) {
|
|
||||||
console.log(updateFlakeInputs());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['f'] || args['firefox']) {
|
|
||||||
console.log(updateFirefoxAddons());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['v'] || args['vuetorrent']) {
|
|
||||||
console.log(updateVuetorrent());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['c'] || args['custom-sidebar']) {
|
|
||||||
console.log(updateCustomPackage(
|
|
||||||
'scopedPackages.x86_64-linux.lovelace-components.custom-sidebar',
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['m'] || args['material-rounded-theme']) {
|
|
||||||
console.log(updateCustomPackage(
|
|
||||||
'scopedPackages.x86_64-linux.lovelace-components.material-rounded-theme',
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['s'] || args['some-sass-language-server']) {
|
|
||||||
console.log(updateCustomPackage('some-sass-language-server'));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['n'] || args['node_modules']) {
|
|
||||||
console.log(await updateNodeModules());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['h'] || args['homepage']) {
|
|
||||||
console.log(runNixUpdate('homepage'));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['cp'] || args['caddy-plugins']) {
|
|
||||||
console.log(updateCaddyPlugins());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['a'] || args['all']) {
|
|
||||||
// Update this first because of nix run cmd
|
|
||||||
const firefoxOutput = updateFirefoxAddons();
|
|
||||||
|
|
||||||
console.log(firefoxOutput);
|
|
||||||
|
|
||||||
|
|
||||||
const flakeOutput = updateFlakeInputs();
|
|
||||||
|
|
||||||
console.log(flakeOutput);
|
|
||||||
|
|
||||||
|
|
||||||
const dockerOutput = updateDocker();
|
|
||||||
|
|
||||||
console.log(dockerOutput);
|
|
||||||
|
|
||||||
|
|
||||||
const nodeModulesOutput = await updateNodeModules();
|
|
||||||
|
|
||||||
console.log(nodeModulesOutput);
|
|
||||||
|
|
||||||
|
|
||||||
const vuetorrentOutput = updateVuetorrent();
|
|
||||||
|
|
||||||
console.log(vuetorrentOutput);
|
|
||||||
|
|
||||||
|
|
||||||
const caddyPluginsOutput = updateCaddyPlugins();
|
|
||||||
|
|
||||||
console.log(caddyPluginsOutput);
|
|
||||||
|
|
||||||
|
|
||||||
// This doesn't need to be added to commit msgs
|
|
||||||
console.log(updateCustomPackage(
|
|
||||||
'scopedPackages.x86_64-linux.lovelace-components.custom-sidebar',
|
|
||||||
));
|
|
||||||
console.log(updateCustomPackage(
|
|
||||||
'scopedPackages.x86_64-linux.lovelace-components.material-rounded-theme',
|
|
||||||
));
|
|
||||||
console.log(updateCustomPackage('some-sass-language-server'));
|
|
||||||
|
|
||||||
// nix-update executions
|
|
||||||
let nixUpdateOutputs = '';
|
|
||||||
|
|
||||||
const updatePackage = (pkg: string): void => {
|
|
||||||
const execution = runNixUpdate(pkg);
|
|
||||||
|
|
||||||
nixUpdateOutputs += execution.stdout;
|
|
||||||
console.log(execution.stderr);
|
|
||||||
console.log(execution.stdout);
|
|
||||||
};
|
|
||||||
|
|
||||||
updatePackage('homepage');
|
|
||||||
|
|
||||||
|
|
||||||
spawnSync('nixFastBuild', [], {
|
|
||||||
shell: true,
|
|
||||||
stdio: [process.stdin, process.stdout, process.stderr],
|
|
||||||
});
|
|
||||||
|
|
||||||
const indentOutput = (output: string): string => {
|
|
||||||
return ` ${output.split('\n').join('\n ')}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
const output = [
|
|
||||||
'chore: update sources\n\n',
|
|
||||||
];
|
|
||||||
|
|
||||||
if (flakeOutput.length > 5) {
|
|
||||||
output.push(`Flake Inputs:\n${indentOutput(flakeOutput)}\n\n`);
|
|
||||||
}
|
|
||||||
if (dockerOutput.length > 5) {
|
|
||||||
output.push(`Docker Images:\n${indentOutput(dockerOutput)}\n`);
|
|
||||||
}
|
|
||||||
if (firefoxOutput.length > 5) {
|
|
||||||
output.push(`Firefox Addons:\n${indentOutput(firefoxOutput)}\n\n`);
|
|
||||||
}
|
|
||||||
if (nodeModulesOutput.length > 5) {
|
|
||||||
output.push(`Node modules:\n${indentOutput(nodeModulesOutput)}\n`);
|
|
||||||
}
|
|
||||||
if (vuetorrentOutput.length > 5) {
|
|
||||||
output.push(`Misc Sources:\n${indentOutput(vuetorrentOutput)}\n\n`);
|
|
||||||
}
|
|
||||||
if (caddyPluginsOutput.length > 5) {
|
|
||||||
output.push(`Caddy Plugins:\n${indentOutput(caddyPluginsOutput)}\n\n`);
|
|
||||||
}
|
|
||||||
if (nixUpdateOutputs.length > 5) {
|
|
||||||
output.push(`nix-update executions:\n${indentOutput(nixUpdateOutputs)}\n`);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (args['f']) {
|
|
||||||
writeFileSync(args['f'] as string, output.join(''));
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
console.log(output.join(''));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
spawnSync('alejandra', ['-q', FLAKE], { shell: true });
|
|
||||||
};
|
|
||||||
|
|
||||||
main();
|
|
|
@ -1,33 +0,0 @@
|
||||||
import { readdirSync } from 'node:fs';
|
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
|
|
||||||
|
|
||||||
/* Constants */
|
|
||||||
const FLAKE = process.env.FLAKE;
|
|
||||||
|
|
||||||
const updateImages = (imagePath: string): string | undefined => {
|
|
||||||
console.log(`Updating ${imagePath.split('/').at(-1)} images`);
|
|
||||||
|
|
||||||
const out = spawnSync('updateImages', [imagePath], { shell: true }).stdout.toString();
|
|
||||||
|
|
||||||
if (!out.startsWith('# Locked')) {
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const updateDocker = () => {
|
|
||||||
let updates = '';
|
|
||||||
|
|
||||||
updates += updateImages(`${FLAKE}/configurations/nos/modules/jellyfin`) ?? '';
|
|
||||||
updates += updateImages(`${FLAKE}/configurations/homie/modules/home-assistant/netdaemon`) ?? '';
|
|
||||||
|
|
||||||
const DIR = `${FLAKE}/configurations/nos/modules/docker`;
|
|
||||||
|
|
||||||
readdirSync(DIR, { withFileTypes: true, recursive: true }).forEach((path) => {
|
|
||||||
if (path.name === 'compose.nix') {
|
|
||||||
updates += updateImages(path.parentPath) ?? '';
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return updates;
|
|
||||||
};
|
|
|
@ -1,61 +0,0 @@
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
import { readFileSync } from 'node:fs';
|
|
||||||
|
|
||||||
|
|
||||||
/* Constants */
|
|
||||||
const FLAKE = process.env.FLAKE;
|
|
||||||
|
|
||||||
export const updateFirefoxAddons = () => {
|
|
||||||
console.log('Updating firefox addons using mozilla-addons-to-nix');
|
|
||||||
|
|
||||||
const DIR = `${FLAKE}/scopedPackages/firefox-addons`;
|
|
||||||
const GENERATED_FILE = `${DIR}/generated-firefox-addons.nix`;
|
|
||||||
const SLUGS = `${DIR}/addons.json`;
|
|
||||||
|
|
||||||
const nameMap = Object.fromEntries([...JSON.parse(readFileSync(SLUGS, 'utf-8'))]
|
|
||||||
.map((addon) => [addon.slug, addon.pname || addon.slug]));
|
|
||||||
|
|
||||||
const nixExpr = `'
|
|
||||||
x: let
|
|
||||||
inherit (builtins) attrValues filter hasAttr isAttrs map;
|
|
||||||
in
|
|
||||||
map (d: d.name) (filter (y:
|
|
||||||
isAttrs y &&
|
|
||||||
hasAttr "type" y &&
|
|
||||||
y.type == "derivation") (attrValues x))
|
|
||||||
'`;
|
|
||||||
|
|
||||||
const OLD_VERS = Object.fromEntries([...JSON.parse(spawnSync('nix', [
|
|
||||||
'eval',
|
|
||||||
'.#scopedPackages.x86_64-linux.firefoxAddons',
|
|
||||||
'--apply',
|
|
||||||
nixExpr,
|
|
||||||
'--json',
|
|
||||||
], { shell: true }).stdout.toString())]
|
|
||||||
.map((p) => {
|
|
||||||
const pname = p.replace(/-[0-9].*$/, '');
|
|
||||||
|
|
||||||
return [pname, p.replace(`${pname}-`, '')];
|
|
||||||
})
|
|
||||||
.filter((pinfo) => pinfo[0] !== 'frankerfacez'));
|
|
||||||
|
|
||||||
const NEW_VERS = Object.fromEntries(spawnSync(
|
|
||||||
'nix',
|
|
||||||
['run', 'sourcehut:~rycee/mozilla-addons-to-nix',
|
|
||||||
SLUGS, GENERATED_FILE],
|
|
||||||
{ shell: true },
|
|
||||||
).stdout
|
|
||||||
.toString()
|
|
||||||
.split('\n')
|
|
||||||
.map((p) => {
|
|
||||||
const pinfo = p.replace('Fetched ', '').split(' ');
|
|
||||||
|
|
||||||
return [nameMap[pinfo[0]], pinfo[2]];
|
|
||||||
}));
|
|
||||||
|
|
||||||
return Object.keys(OLD_VERS)
|
|
||||||
.sort()
|
|
||||||
.filter((pname) => OLD_VERS[pname] !== NEW_VERS[pname])
|
|
||||||
.map((pname) => `${pname}: ${OLD_VERS[pname]} -> ${NEW_VERS[pname]}`)
|
|
||||||
.join('\n');
|
|
||||||
};
|
|
|
@ -1,44 +0,0 @@
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
|
|
||||||
|
|
||||||
/* Constants */
|
|
||||||
const FLAKE = process.env.FLAKE;
|
|
||||||
|
|
||||||
export const updateFlakeInputs = () => {
|
|
||||||
const output = spawnSync(
|
|
||||||
`git restore flake.lock &> /dev/null; nix flake update --flake ${FLAKE}` +
|
|
||||||
' |& grep -v "warning: updating lock file" |& grep -v "unpacking"',
|
|
||||||
[],
|
|
||||||
{ shell: true },
|
|
||||||
).stdout
|
|
||||||
.toString()
|
|
||||||
// Add an extra blank line between inputs
|
|
||||||
.split('\n•')
|
|
||||||
// Filter out some inputs
|
|
||||||
.filter((input) => ![
|
|
||||||
'systems',
|
|
||||||
'flake-utils',
|
|
||||||
'flake-parts',
|
|
||||||
'treefmt-nix',
|
|
||||||
'lib-aggregate',
|
|
||||||
'lib-aggregate/nixpkgs-lib',
|
|
||||||
'sops-nix/nixpkgs-stable',
|
|
||||||
'discord-overlay/Vencord-src',
|
|
||||||
'nix-gaming/umu',
|
|
||||||
].some((inputName) => input.startsWith(` Updated input '${inputName}'`)))
|
|
||||||
.join('\n\n•')
|
|
||||||
// help readability of git revs
|
|
||||||
.split('\n')
|
|
||||||
.map((l) => l
|
|
||||||
.replace(
|
|
||||||
/\/(.{40})\?narHash=sha256[^']*(.*)/,
|
|
||||||
(_, backref1, backref2) => `${backref2} rev: ${backref1}`,
|
|
||||||
)
|
|
||||||
.replace(
|
|
||||||
/\?ref.*&rev=(.{40})[^'&]*(.*)/,
|
|
||||||
(_, backref1, backref2) => `${backref2} rev: ${backref1}`,
|
|
||||||
))
|
|
||||||
.join('\n');
|
|
||||||
|
|
||||||
return output;
|
|
||||||
};
|
|
|
@ -1,43 +0,0 @@
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
import { readFileSync, writeFileSync } from 'node:fs';
|
|
||||||
|
|
||||||
|
|
||||||
export const parseArgs = () => {
|
|
||||||
const args = {} as Record<string, unknown>;
|
|
||||||
let lastFlag: string | null = null;
|
|
||||||
|
|
||||||
for (let i = 2; i < process.argv.length; ++i) {
|
|
||||||
const arg = process.argv[i];
|
|
||||||
|
|
||||||
if (arg.toString().startsWith('-')) {
|
|
||||||
lastFlag = arg.toString().replace(/^-{1,2}/, '');
|
|
||||||
args[lastFlag] = true;
|
|
||||||
}
|
|
||||||
else if (lastFlag) {
|
|
||||||
args[lastFlag] = arg;
|
|
||||||
lastFlag = null;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
console.error(`Could not parse args: ${arg.toString()}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return args;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const parseFetchurl = (url: string) => JSON.parse(spawnSync(
|
|
||||||
'nix', ['store', 'prefetch-file', '--refresh', '--json',
|
|
||||||
'--hash-type', 'sha256', url, '--name', '"escaped"'], { shell: true },
|
|
||||||
).stdout.toString()).hash;
|
|
||||||
|
|
||||||
export const replaceInFile = (replace: RegExp, replacement: string, file: string) => {
|
|
||||||
const fileContents = readFileSync(file);
|
|
||||||
|
|
||||||
const replaced = fileContents.toString().replace(replace, replacement);
|
|
||||||
|
|
||||||
writeFileSync(file, replaced);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const npmRun = (args: string[], workspaceDir: string) => spawnSync(
|
|
||||||
'npm', args, { cwd: workspaceDir },
|
|
||||||
).stdout.toString();
|
|
|
@ -1,148 +0,0 @@
|
||||||
import { writeFileSync } from 'node:fs';
|
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
|
|
||||||
import { parseFetchurl, replaceInFile } from './lib';
|
|
||||||
|
|
||||||
|
|
||||||
/* Constants */
|
|
||||||
const FLAKE = process.env.FLAKE;
|
|
||||||
|
|
||||||
const genVueText = (
|
|
||||||
version: string,
|
|
||||||
hash: string,
|
|
||||||
url: string,
|
|
||||||
) => `# This file was autogenerated. DO NOT EDIT!
|
|
||||||
{
|
|
||||||
version = "${version}";
|
|
||||||
url = "${url}";
|
|
||||||
hash = "${hash}";
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
export const updateVuetorrent = () => {
|
|
||||||
const FILE = `${FLAKE}/configurations/nos/modules/qbittorrent/vuetorrent.nix`;
|
|
||||||
|
|
||||||
const OLD_VERSION = JSON.parse(spawnSync('nix',
|
|
||||||
['eval', '-f', FILE, '--json'],
|
|
||||||
{ shell: true }).stdout.toString()).version;
|
|
||||||
|
|
||||||
const VERSION = JSON.parse(spawnSync('curl',
|
|
||||||
['-s', 'https://api.github.com/repos/VueTorrent/VueTorrent/releases/latest'],
|
|
||||||
{ shell: true }).stdout.toString()).tag_name.replace('v', '');
|
|
||||||
|
|
||||||
const URL = `https://github.com/VueTorrent/VueTorrent/releases/download/v${VERSION}/vuetorrent.zip`;
|
|
||||||
const HASH = parseFetchurl(URL);
|
|
||||||
|
|
||||||
const fileText = genVueText(VERSION, HASH, URL);
|
|
||||||
|
|
||||||
writeFileSync(FILE, fileText);
|
|
||||||
|
|
||||||
return OLD_VERSION !== VERSION ? `Vuetorrent: ${OLD_VERSION} -> ${VERSION}` : '';
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
export const updateCustomPackage = (pkg: string) => spawnSync(
|
|
||||||
`nix run ${FLAKE}#${pkg}.update`,
|
|
||||||
[],
|
|
||||||
{ shell: true },
|
|
||||||
).stderr.toString();
|
|
||||||
|
|
||||||
|
|
||||||
const getAttrVersion = (attr: string): string => spawnSync('nix',
|
|
||||||
['eval', '--raw', `${FLAKE}#${attr}.version`],
|
|
||||||
{ shell: true }).stdout.toString();
|
|
||||||
|
|
||||||
export const runNixUpdate = (
|
|
||||||
attr: string,
|
|
||||||
options: string[] = [],
|
|
||||||
): { stdout: string, stderr: string } => {
|
|
||||||
const OLD_VERSION = getAttrVersion(attr);
|
|
||||||
|
|
||||||
const execution = spawnSync(
|
|
||||||
`nix-update --flake ${attr} --write-commit-message >(head -n 1 -) > /dev/null`,
|
|
||||||
options,
|
|
||||||
{ shell: true, cwd: FLAKE },
|
|
||||||
);
|
|
||||||
|
|
||||||
const NEW_VERSION = getAttrVersion(attr);
|
|
||||||
|
|
||||||
return {
|
|
||||||
stdout: OLD_VERSION !== NEW_VERSION ? execution.stdout.toString() : '',
|
|
||||||
stderr: execution.stderr.toString(),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
const genPluginsText = (
|
|
||||||
plugins: Record<string, { url: string, version: string }>,
|
|
||||||
) => `# This file was autogenerated. DO NOT EDIT!
|
|
||||||
{
|
|
||||||
plugins = {
|
|
||||||
${Object.entries(plugins)
|
|
||||||
.map(([key, value]) => `
|
|
||||||
${key} = {
|
|
||||||
url = "${value.url}";
|
|
||||||
version = "${value.version}";
|
|
||||||
};
|
|
||||||
`)
|
|
||||||
.join('')}
|
|
||||||
};
|
|
||||||
|
|
||||||
hash = "";
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
export const updateCaddyPlugins = () => {
|
|
||||||
let updates = '';
|
|
||||||
const dir = `${FLAKE}/configurations/cluster/modules/caddy`;
|
|
||||||
|
|
||||||
// Setup workspace
|
|
||||||
spawnSync(
|
|
||||||
[
|
|
||||||
'rm -rf /tmp/update-caddy',
|
|
||||||
'mkdir -p /tmp/update-caddy',
|
|
||||||
'cd /tmp/update-caddy || exit 1',
|
|
||||||
'go mod init temp',
|
|
||||||
].join('; '),
|
|
||||||
[],
|
|
||||||
{ shell: true, cwd: '/tmp' },
|
|
||||||
);
|
|
||||||
|
|
||||||
const plugins = JSON.parse(spawnSync('nix',
|
|
||||||
['eval', '-f', `${dir}/plugins.nix`, '--json'],
|
|
||||||
{ shell: true }).stdout.toString()).plugins as Record<string, { url: string, version: string }>;
|
|
||||||
|
|
||||||
// Get most recent versions of plugins
|
|
||||||
Object.entries(plugins).forEach(([key, value]) => {
|
|
||||||
const NEW_VERSION = spawnSync([
|
|
||||||
'go mod init temp > /dev/null',
|
|
||||||
`go get ${value.url} > /dev/null`,
|
|
||||||
`grep '${value.url}' go.mod`,
|
|
||||||
].join('; '), [], { shell: true, cwd: '/tmp/update-caddy' })
|
|
||||||
.stdout
|
|
||||||
.toString()
|
|
||||||
.trim()
|
|
||||||
.replace(' // indirect', '')
|
|
||||||
.split(' ')[1];
|
|
||||||
|
|
||||||
if (plugins[key].version !== NEW_VERSION) {
|
|
||||||
updates += `${key}: ${plugins[key].version} -> ${NEW_VERSION}`;
|
|
||||||
plugins[key].version = NEW_VERSION;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
writeFileSync(`${dir}/plugins.nix`, genPluginsText(plugins));
|
|
||||||
|
|
||||||
// Get new hash
|
|
||||||
const caddyPkgAttr = 'nixosConfigurations.thingone.config.services.caddy.package';
|
|
||||||
|
|
||||||
const NEW_HASH = spawnSync(
|
|
||||||
`nix build "$FLAKE#${caddyPkgAttr}" |& sed -n 's/.*got: *//p'`,
|
|
||||||
[],
|
|
||||||
{ shell: true },
|
|
||||||
).stdout.toString().trim();
|
|
||||||
|
|
||||||
replaceInFile(/hash = ".*";/, `hash = "${NEW_HASH}";`, `${dir}/plugins.nix`);
|
|
||||||
|
|
||||||
return updates;
|
|
||||||
};
|
|
|
@ -1,83 +0,0 @@
|
||||||
import { readPackageJSON, writePackageJSON } from 'pkg-types';
|
|
||||||
import { existsSync, readdirSync } from 'node:fs';
|
|
||||||
import { spawnSync } from 'node:child_process';
|
|
||||||
|
|
||||||
import { replaceInFile, npmRun } from './lib';
|
|
||||||
|
|
||||||
|
|
||||||
/* Constants */
|
|
||||||
const FLAKE = process.env.FLAKE as string;
|
|
||||||
|
|
||||||
|
|
||||||
const updatePackageJson = async(workspaceDir: string, updates: object) => {
|
|
||||||
const currentPackageJson = await readPackageJSON(`${workspaceDir}/package.json`);
|
|
||||||
|
|
||||||
const outdated = JSON.parse(npmRun(['outdated', '--json'], workspaceDir));
|
|
||||||
|
|
||||||
const updateDeps = (deps: string) => {
|
|
||||||
Object.keys(currentPackageJson[deps]).forEach((dep) => {
|
|
||||||
const versions = outdated[dep];
|
|
||||||
const current = versions?.wanted || versions?.current;
|
|
||||||
|
|
||||||
if (!current) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (current !== versions.latest) {
|
|
||||||
updates[dep] = `${current} -> ${versions.latest}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
currentPackageJson[deps][dep] = versions.latest;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
if (currentPackageJson.dependencies) {
|
|
||||||
updateDeps('dependencies');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentPackageJson.devDependencies) {
|
|
||||||
updateDeps('devDependencies');
|
|
||||||
}
|
|
||||||
|
|
||||||
await writePackageJSON(`${workspaceDir}/package.json`, currentPackageJson);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
const prefetchNpmDeps = (workspaceDir: string): string => {
|
|
||||||
npmRun(['update', '--package-lock-only'], workspaceDir);
|
|
||||||
|
|
||||||
return spawnSync(
|
|
||||||
'prefetch-npm-deps',
|
|
||||||
[`${workspaceDir}/package-lock.json`],
|
|
||||||
).stdout.toString().replace('\n', '');
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
export default async() => {
|
|
||||||
const updates = {};
|
|
||||||
|
|
||||||
const packages = readdirSync(FLAKE, { withFileTypes: true, recursive: true });
|
|
||||||
|
|
||||||
for (const path of packages) {
|
|
||||||
if (
|
|
||||||
path.name === 'package.json' &&
|
|
||||||
!path.parentPath.includes('node_modules')
|
|
||||||
) {
|
|
||||||
await updatePackageJson(path.parentPath, updates);
|
|
||||||
|
|
||||||
if (existsSync(`${path.parentPath}/default.nix`)) {
|
|
||||||
const hash = prefetchNpmDeps(path.parentPath);
|
|
||||||
|
|
||||||
replaceInFile(
|
|
||||||
/npmDepsHash = ".*";/,
|
|
||||||
`npmDepsHash = "${hash}";`,
|
|
||||||
`${path.parentPath}/default.nix`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Object.entries(updates)
|
|
||||||
.map(([key, dep]) => `${key}: ${dep}`)
|
|
||||||
.join('\n');
|
|
||||||
};
|
|
|
@ -1,10 +0,0 @@
|
||||||
{
|
|
||||||
"$schema": "https://json.schemastore.org/tsconfig",
|
|
||||||
"extends": "../config/tsconfig.base.json",
|
|
||||||
"includes": [
|
|
||||||
"*.ts",
|
|
||||||
"**/*.ts",
|
|
||||||
"*.js",
|
|
||||||
"**/*.js"
|
|
||||||
]
|
|
||||||
}
|
|
|
@ -1,16 +0,0 @@
|
||||||
# Devices
|
|
||||||
|
|
||||||
This directory encompasses every device's main configuration file.
|
|
||||||
|
|
||||||
## List of my Devices
|
|
||||||
|
|
||||||
| Name | Description |
|
|
||||||
| ----------- | ------------------------------------------------------------------------------------------------------- |
|
|
||||||
| `android` | My [Nix-On-Droid](https://github.com/nix-community/nix-on-droid) configuration for my OnePlus 9 Pro |
|
|
||||||
| `bbsteamie` | My wife's SteamDeck that has a pink case |
|
|
||||||
| `binto` | My desktop PC with a multi-monitor setup and an NVIDIA (cringe) 3070 |
|
|
||||||
| `cluster` | Two Lenovo mini PCs that make use of [NixOS-pcsd](https://github.com/matt1432/nixos-pcsd) to form a cluster |
|
|
||||||
| `homie` | My Lenovo mini PC that will serve as a Home-assistant server |
|
|
||||||
| `nos` | My custom built NAS |
|
|
||||||
| `servivi` | A gaming PC in a previous life, it is now used as a build farm and hosts game servers |
|
|
||||||
| `wim` | My 2-1 Lenovo Laptop that I use for uni |
|
|
|
@ -1,56 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) attrValues concatStringsSep;
|
|
||||||
in {
|
|
||||||
imports = [./nix-on-droid.nix];
|
|
||||||
|
|
||||||
environment.variables.FLAKE = "/data/data/com.termux.nix/files/home/.nix";
|
|
||||||
|
|
||||||
terminal.font = "${
|
|
||||||
pkgs.nerd-fonts.jetbrains-mono
|
|
||||||
}/share/fonts/truetype/NerdFonts/JetBrainsMono/JetBrainsMonoNerdFontMono-Regular.ttf";
|
|
||||||
|
|
||||||
environment.packages = [
|
|
||||||
(pkgs.writeShellApplication {
|
|
||||||
name = "switch";
|
|
||||||
|
|
||||||
runtimeInputs = attrValues {
|
|
||||||
inherit
|
|
||||||
(pkgs)
|
|
||||||
coreutils
|
|
||||||
nix-output-monitor
|
|
||||||
nvd
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
oldProfile=$(realpath /nix/var/nix/profiles/per-user/nix-on-droid/profile)
|
|
||||||
|
|
||||||
nix-on-droid ${concatStringsSep " " [
|
|
||||||
"switch"
|
|
||||||
"--flake ${config.environment.variables.FLAKE}"
|
|
||||||
"--builders ssh-ng://matt@100.64.0.7"
|
|
||||||
''"$@"''
|
|
||||||
"|&"
|
|
||||||
"nom"
|
|
||||||
]} &&
|
|
||||||
|
|
||||||
nvd diff "$oldProfile" "$(realpath /nix/var/nix/profiles/per-user/nix-on-droid/profile)"
|
|
||||||
'';
|
|
||||||
})
|
|
||||||
];
|
|
||||||
|
|
||||||
environment.etcBackupExtension = ".bak";
|
|
||||||
environment.motd = null;
|
|
||||||
home-manager.backupFileExtension = "hm-bak";
|
|
||||||
|
|
||||||
# Set your time zone.
|
|
||||||
time.timeZone = "America/Montreal";
|
|
||||||
|
|
||||||
# No touchy
|
|
||||||
system.stateVersion = "23.05";
|
|
||||||
}
|
|
|
@ -1,90 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
imports = [
|
|
||||||
self.nixosModules.base-droid
|
|
||||||
{
|
|
||||||
roles.base = {
|
|
||||||
enable = true;
|
|
||||||
user = "nix-on-droid";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
self.nixosModules.tmux
|
|
||||||
{programs.tmux.enableCustomConf = true;}
|
|
||||||
];
|
|
||||||
|
|
||||||
nix = {
|
|
||||||
# Edit nix.conf
|
|
||||||
extraOptions = ''
|
|
||||||
experimental-features = nix-command flakes
|
|
||||||
keep-outputs = true
|
|
||||||
keep-derivations = true
|
|
||||||
warn-dirty = false
|
|
||||||
'';
|
|
||||||
|
|
||||||
substituters = [
|
|
||||||
# Nix-community
|
|
||||||
"https://nix-community.cachix.org"
|
|
||||||
];
|
|
||||||
trustedPublicKeys = [
|
|
||||||
# Nix-community
|
|
||||||
"nix-community.cachix.org-1:mB9FSh9qf2dCimDSUo8Zy7bkq5CX+/rkCWyvRCYg3Fs="
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
# Global hm settings
|
|
||||||
home-manager.config = {
|
|
||||||
imports = [
|
|
||||||
self.homeManagerModules.neovim
|
|
||||||
{
|
|
||||||
programs.neovim = {
|
|
||||||
enable = true;
|
|
||||||
user = "nix-on-droid";
|
|
||||||
|
|
||||||
ideConfig = {
|
|
||||||
enableJava = false;
|
|
||||||
enableNix = false;
|
|
||||||
enablePython = false;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
self.homeManagerModules.shell
|
|
||||||
{programs.bash.enable = true;}
|
|
||||||
|
|
||||||
{
|
|
||||||
programs.bash.sessionVariables = {
|
|
||||||
FLAKE = config.environment.variables.FLAKE;
|
|
||||||
};
|
|
||||||
|
|
||||||
programs.bash.shellAliases = {
|
|
||||||
# Make ping work on nix-on-droid
|
|
||||||
# https://github.com/nix-community/nix-on-droid/issues/185#issuecomment-1659294700
|
|
||||||
ping = "/android/system/bin/linker64 /android/system/bin/ping";
|
|
||||||
|
|
||||||
# SSH
|
|
||||||
# Desktop
|
|
||||||
pc = "ssh -t matt@100.64.0.6 'tmux -2u new -At phone'";
|
|
||||||
|
|
||||||
# NAS
|
|
||||||
nos = "ssh -t matt@100.64.0.4 'tmux -2u new -At phone'";
|
|
||||||
|
|
||||||
# Experimenting server
|
|
||||||
servivi = "ssh -t matt@100.64.0.7 'tmux -2u new -At phone'";
|
|
||||||
|
|
||||||
# Home-assistant
|
|
||||||
homie = "ssh -t matt@100.64.0.10 'tmux -2u new -At phone'";
|
|
||||||
|
|
||||||
# Cluster nodes
|
|
||||||
thingone = "ssh -t matt@100.64.0.8 'tmux -2u new -At phone'";
|
|
||||||
thingtwo = "ssh -t matt@100.64.0.9 'tmux -2u new -At phone'";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
home.stateVersion = "23.05";
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,74 +0,0 @@
|
||||||
{
|
|
||||||
mainUser,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
# ------------------------------------------------
|
|
||||||
# Imports
|
|
||||||
# ------------------------------------------------
|
|
||||||
imports = [
|
|
||||||
./hardware-configuration.nix
|
|
||||||
|
|
||||||
./modules
|
|
||||||
|
|
||||||
self.nixosModules.base
|
|
||||||
self.nixosModules.kmscon
|
|
||||||
self.nixosModules.plymouth
|
|
||||||
self.nixosModules.server
|
|
||||||
];
|
|
||||||
|
|
||||||
# State Version: DO NOT CHANGE
|
|
||||||
system.stateVersion = "24.11";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# User Settings
|
|
||||||
# ------------------------------------------------
|
|
||||||
users.users.${mainUser} = {
|
|
||||||
isNormalUser = true;
|
|
||||||
extraGroups = [
|
|
||||||
"wheel"
|
|
||||||
"adm"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
networking = {
|
|
||||||
hostName = "bbsteamie";
|
|
||||||
networkmanager.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
time.timeZone = "America/Montreal";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# `Self` Modules configuration
|
|
||||||
# ------------------------------------------------
|
|
||||||
roles.base = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
|
|
||||||
roles.server = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
sshd.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
boot.plymouth = {
|
|
||||||
enable = true;
|
|
||||||
theme = "bgrt";
|
|
||||||
};
|
|
||||||
|
|
||||||
services.kmscon.enable = true;
|
|
||||||
|
|
||||||
home-manager.users.${mainUser} = {
|
|
||||||
imports = [
|
|
||||||
self.homeManagerModules.shell
|
|
||||||
];
|
|
||||||
|
|
||||||
programs = {
|
|
||||||
bash = {
|
|
||||||
enable = true;
|
|
||||||
promptMainColor = "pink";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,61 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
jovian,
|
|
||||||
modulesPath,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
nixpkgs.hostPlatform = "x86_64-linux";
|
|
||||||
nixpkgs.overlays = [jovian.overlays.default];
|
|
||||||
imports = [
|
|
||||||
(modulesPath + "/installer/scan/not-detected.nix")
|
|
||||||
jovian.nixosModules.default
|
|
||||||
];
|
|
||||||
|
|
||||||
jovian = {
|
|
||||||
steamos.useSteamOSConfig = true;
|
|
||||||
|
|
||||||
devices.steamdeck = {
|
|
||||||
enable = true;
|
|
||||||
enableGyroDsuService = true;
|
|
||||||
};
|
|
||||||
hardware.has.amd.gpu = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
boot = {
|
|
||||||
kernelModules = ["kvm-amd"];
|
|
||||||
initrd.availableKernelModules = ["nvme" "xhci_pci" "usbhid" "sdhci_pci"];
|
|
||||||
|
|
||||||
loader = {
|
|
||||||
efi.canTouchEfiVariables = true;
|
|
||||||
|
|
||||||
systemd-boot = {
|
|
||||||
enable = true;
|
|
||||||
configurationLimit = 30;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
virtualisation.waydroid.enable = true;
|
|
||||||
|
|
||||||
fileSystems = {
|
|
||||||
"/" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "ext4";
|
|
||||||
};
|
|
||||||
|
|
||||||
"/boot" = {
|
|
||||||
device = "/dev/disk/by-label/NIXBOOT";
|
|
||||||
fsType = "vfat";
|
|
||||||
options = ["fmask=0022" "dmask=0022"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
swapDevices = [
|
|
||||||
{
|
|
||||||
device = "/.swapfile";
|
|
||||||
size = 16 * 1024; # 16GB
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
hardware.cpu.amd.updateMicrocode = config.hardware.enableRedistributableFirmware;
|
|
||||||
}
|
|
|
@ -1,5 +0,0 @@
|
||||||
{...}: {
|
|
||||||
imports = [
|
|
||||||
./desktop
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,40 +0,0 @@
|
||||||
{pkgs, ...}: let
|
|
||||||
defaultSession = "plasma";
|
|
||||||
in {
|
|
||||||
imports = [
|
|
||||||
(import ./session-switching.nix defaultSession)
|
|
||||||
(import ./steam.nix defaultSession)
|
|
||||||
];
|
|
||||||
|
|
||||||
services.desktopManager.plasma6.enable = true;
|
|
||||||
|
|
||||||
programs = {
|
|
||||||
kdeconnect.enable = true;
|
|
||||||
xwayland.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
# Flatpak support for Discover
|
|
||||||
services.flatpak.enable = true;
|
|
||||||
services.packagekit.enable = true;
|
|
||||||
|
|
||||||
# Wayland env vars
|
|
||||||
environment.variables = {
|
|
||||||
NIXOS_OZONE_WL = "1";
|
|
||||||
ELECTRON_OZONE_PLATFORM_HINT = "auto";
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = builtins.attrValues {
|
|
||||||
inherit
|
|
||||||
(pkgs)
|
|
||||||
firefox
|
|
||||||
wl-clipboard
|
|
||||||
xclip
|
|
||||||
;
|
|
||||||
|
|
||||||
inherit
|
|
||||||
(pkgs.kdePackages)
|
|
||||||
discover
|
|
||||||
krfb
|
|
||||||
;
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,81 +0,0 @@
|
||||||
# How to install Palia Map and Overwolf on Linux
|
|
||||||
|
|
||||||
Dependencies:
|
|
||||||
- latest GE-Proton: https://github.com/GloriousEggroll/proton-ge-custom
|
|
||||||
- protontricks: https://github.com/Matoking/protontricks
|
|
||||||
- protonhax: https://github.com/jcnils/protonhax
|
|
||||||
|
|
||||||
|
|
||||||
## First Step: Install and run Palia at least once
|
|
||||||
|
|
||||||
|
|
||||||
## Second Step: Setup Palia WINEPREFIX with latest GE-Proton
|
|
||||||
|
|
||||||
1. Delete prefix at /home/"$USER"/.steam/steam/steamapps/compatdata/2707930/pfx
|
|
||||||
|
|
||||||
```bash
|
|
||||||
rm -r /home/"$USER"/.steam/steam/steamapps/compatdata/2707930/pfx
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Launch Palia with GE-Proton by going to game properties -> Compatibility -> Force the use ... -> Select GE-Proton...
|
|
||||||
|
|
||||||
3. Install dotnet48 and other Windows deps to allow for Overwolf installation
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Force proton version of protontricks
|
|
||||||
export PROTON_VERSION="GE-Proton9-10"
|
|
||||||
protontricks 2707930 dotnet48
|
|
||||||
|
|
||||||
# If VC is needed
|
|
||||||
protontricks 2707930 vcrun2010
|
|
||||||
protontricks 2707930 vcrun2012
|
|
||||||
protontricks 2707930 vcrun2013
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## Third Step: Install Overwolf
|
|
||||||
|
|
||||||
1. Get this older version that worked for me here: https://overwolf.en.uptodown.com/windows/download/4714215
|
|
||||||
|
|
||||||
2. Install it with protontricks. Follow the GUI installer as if you were on Windows
|
|
||||||
|
|
||||||
```bash
|
|
||||||
export PROTON_VERSION="GE-Proton9-10"
|
|
||||||
protontricks-launch --appid 2707930 Downloads/overwolf-0-195-0-18.exe
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
## Fourth Step: Install Palia Map
|
|
||||||
|
|
||||||
1. Get the installer from here: https://www.overwolf.com/app/Leon_Machens-Palia_Map
|
|
||||||
|
|
||||||
2. Install it with protontricks.
|
|
||||||
|
|
||||||
```bash
|
|
||||||
export PROTON_VERSION="GE-Proton9-10"
|
|
||||||
protontricks-launch --appid 2707930 Downloads/Palia\ Map\ -\ Installer.exe
|
|
||||||
```
|
|
||||||
|
|
||||||
3. An error should popup saying `Installation failed` or something along those lines.
|
|
||||||
Close it and wait. You should see the Overwolf overlay on the left side of your screen
|
|
||||||
with the Palia Map icon saying `Installing` and then when it's done: `Palia Map`.
|
|
||||||
|
|
||||||
If nothing happens, try rebooting your PC and retrying the above shell commands until it works.
|
|
||||||
|
|
||||||
|
|
||||||
## Final Step: Setting it up in Steam
|
|
||||||
|
|
||||||
1. Add this line to Palia launch options
|
|
||||||
```bash
|
|
||||||
protonhax init %command%
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Add random non-steam game to your library and then edit its properties:
|
|
||||||
|
|
||||||
Change the name to `Palia Map` or anything you like
|
|
||||||
|
|
||||||
Change the target to this command:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
protonhax run 2707930 "/home/$USER/.local/share/Steam/steamapps/compatdata/2707930/pfx/drive_c/users/steamuser/AppData/Roaming/Microsoft/Windows/Start Menu/Programs/Overwolf/Palia Map.lnk"
|
|
||||||
```
|
|
|
@ -1,114 +0,0 @@
|
||||||
defaultSession: {
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
mainUser,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
config = let
|
|
||||||
inherit (lib) findFirst getExe mkForce;
|
|
||||||
|
|
||||||
switch-session = pkgs.writeShellApplication {
|
|
||||||
name = "switch-session";
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
mkdir -p /etc/sddm.conf.d
|
|
||||||
|
|
||||||
cat <<EOF | tee /etc/sddm.conf.d/autologin.conf
|
|
||||||
[Autologin]
|
|
||||||
User=${mainUser}
|
|
||||||
Session=$1
|
|
||||||
Relogin=true
|
|
||||||
EOF
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
gaming-mode = pkgs.writeShellScriptBin "gaming-mode" ''
|
|
||||||
sudo ${pkgs.systemd}/bin/systemctl start to-gaming-mode.service
|
|
||||||
'';
|
|
||||||
in {
|
|
||||||
services.displayManager.sddm = {
|
|
||||||
enable = true;
|
|
||||||
autoLogin.relogin = true;
|
|
||||||
|
|
||||||
wayland = {
|
|
||||||
enable = true;
|
|
||||||
compositorCommand = "kwin";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# Sets the default session at launch
|
|
||||||
systemd.services."set-session" = {
|
|
||||||
wantedBy = ["multi-user.target"];
|
|
||||||
before = ["display-manager.service"];
|
|
||||||
|
|
||||||
path = [switch-session];
|
|
||||||
|
|
||||||
script = ''
|
|
||||||
switch-session "${defaultSession}"
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
# Allows switching to gaming mode
|
|
||||||
systemd.services."to-gaming-mode" = {
|
|
||||||
wantedBy = mkForce [];
|
|
||||||
|
|
||||||
path = [switch-session];
|
|
||||||
|
|
||||||
script = ''
|
|
||||||
switch-session "gamescope-wayland"
|
|
||||||
systemctl restart display-manager
|
|
||||||
sleep 10
|
|
||||||
switch-session "${defaultSession}"
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
# Make it so we don't need root to switch to gaming mode
|
|
||||||
security.sudo.extraRules = [
|
|
||||||
{
|
|
||||||
users = [mainUser];
|
|
||||||
groups = [100];
|
|
||||||
commands = [
|
|
||||||
{
|
|
||||||
command = "${pkgs.systemd}/bin/systemctl start to-gaming-mode.service";
|
|
||||||
options = ["SETENV" "NOPASSWD"];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
home-manager.users.${mainUser} = {
|
|
||||||
# Add desktop entry to make it GUI friendly
|
|
||||||
xdg.desktopEntries."Gaming Mode" = {
|
|
||||||
name = "Gaming Mode";
|
|
||||||
exec = getExe gaming-mode;
|
|
||||||
icon = "steam";
|
|
||||||
terminal = false;
|
|
||||||
type = "Application";
|
|
||||||
};
|
|
||||||
|
|
||||||
home.file."Desktop/Gaming Mode.desktop".source =
|
|
||||||
(
|
|
||||||
findFirst
|
|
||||||
(x: x.meta.name == "Gaming Mode.desktop") {}
|
|
||||||
config.home-manager.users.mariah.home.packages
|
|
||||||
)
|
|
||||||
+ "/share/applications/Gaming Mode.desktop";
|
|
||||||
|
|
||||||
# Fix remote control prompt showing up everytime
|
|
||||||
xdg.configFile = let
|
|
||||||
mkAutostart = name: flags: {
|
|
||||||
"autostart/${name}.desktop".text = "[Desktop Entry]\nType=Application\nExec=${name} ${flags}";
|
|
||||||
};
|
|
||||||
in (
|
|
||||||
# Needs xdg-desktop-portal-kde patch provided by `self.overlays.xdg-desktop-portal-kde`
|
|
||||||
{"plasmaremotedesktoprc".text = "[Sharing]\nUnattended=true";}
|
|
||||||
// (mkAutostart "krfb" "--nodialog %c")
|
|
||||||
// (mkAutostart "steam" "-silent %U")
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# For accurate stack trace
|
|
||||||
_file = ./session-switching.nix;
|
|
||||||
}
|
|
|
@ -1,77 +0,0 @@
|
||||||
defaultSession: {
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
mainUser,
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) attrValues makeSearchPathOutput;
|
|
||||||
in {
|
|
||||||
config = {
|
|
||||||
# Normal Steam Stuff
|
|
||||||
programs.steam = {
|
|
||||||
enable = true;
|
|
||||||
protontricks.enable = true;
|
|
||||||
|
|
||||||
remotePlay.openFirewall = true;
|
|
||||||
extraCompatPackages = [
|
|
||||||
self.packages.${pkgs.system}.proton-ge-latest
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
# Jovian Steam settings
|
|
||||||
jovian.steam = {
|
|
||||||
# Steam > Settings > System > Enable Developer Mode
|
|
||||||
# Steam > Developer > CEF Remote Debugging
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
|
|
||||||
environment = {
|
|
||||||
STEAM_EXTRA_COMPAT_TOOLS_PATHS =
|
|
||||||
makeSearchPathOutput
|
|
||||||
"steamcompattool"
|
|
||||||
""
|
|
||||||
config.programs.steam.extraCompatPackages;
|
|
||||||
};
|
|
||||||
|
|
||||||
desktopSession = defaultSession;
|
|
||||||
};
|
|
||||||
|
|
||||||
# Decky settings
|
|
||||||
jovian.decky-loader = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
stateDir = "/home/${mainUser}/.local/share/decky"; # Keep scoped to user
|
|
||||||
# https://github.com/Jovian-Experiments/Jovian-NixOS/blob/1171169117f63f1de9ef2ea36efd8dcf377c6d5a/modules/decky-loader.nix#L80-L84
|
|
||||||
|
|
||||||
extraPackages = attrValues {
|
|
||||||
inherit
|
|
||||||
(pkgs)
|
|
||||||
curl
|
|
||||||
unzip
|
|
||||||
util-linux
|
|
||||||
gnugrep
|
|
||||||
readline
|
|
||||||
procps
|
|
||||||
pciutils
|
|
||||||
libpulseaudio
|
|
||||||
;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# Misc Packages
|
|
||||||
environment.systemPackages = [
|
|
||||||
pkgs.steam-rom-manager
|
|
||||||
pkgs.r2modman
|
|
||||||
|
|
||||||
self.packages.${pkgs.system}.protonhax
|
|
||||||
|
|
||||||
# Ryujinx ACNH crashes on Vulkan
|
|
||||||
pkgs.ryujinx
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
# For accurate stack trace
|
|
||||||
_file = ./steam.nix;
|
|
||||||
}
|
|
|
@ -1,98 +0,0 @@
|
||||||
{
|
|
||||||
mainUser,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
# ------------------------------------------------
|
|
||||||
# Imports
|
|
||||||
# ------------------------------------------------
|
|
||||||
imports = [
|
|
||||||
./hardware-configuration.nix
|
|
||||||
|
|
||||||
./modules
|
|
||||||
|
|
||||||
self.nixosModules.base
|
|
||||||
self.nixosModules.desktop
|
|
||||||
self.nixosModules.kmscon
|
|
||||||
self.nixosModules.server
|
|
||||||
];
|
|
||||||
|
|
||||||
# State Version: DO NOT CHANGE
|
|
||||||
system.stateVersion = "23.11";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# User Settings
|
|
||||||
# ------------------------------------------------
|
|
||||||
users.users.${mainUser} = {
|
|
||||||
isNormalUser = true;
|
|
||||||
extraGroups = [
|
|
||||||
"wheel"
|
|
||||||
"input"
|
|
||||||
"uinput"
|
|
||||||
"adm"
|
|
||||||
"video"
|
|
||||||
"libvirtd"
|
|
||||||
"adbusers"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
programs.adb.enable = true;
|
|
||||||
|
|
||||||
networking = {
|
|
||||||
hostName = "binto";
|
|
||||||
networkmanager.enable = true;
|
|
||||||
firewall.enable = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
time.timeZone = "America/Montreal";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# `Self` Modules configuration
|
|
||||||
# ------------------------------------------------
|
|
||||||
roles.base = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
|
|
||||||
roles.desktop = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
|
|
||||||
ags.enable = true;
|
|
||||||
mainMonitor = "desc:GIGA-BYTE TECHNOLOGY CO. LTD. G27QC 0x00000B1D";
|
|
||||||
displayManager.duplicateScreen = false;
|
|
||||||
|
|
||||||
fontSize = 12.5;
|
|
||||||
};
|
|
||||||
|
|
||||||
roles.server = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
tailscale.enable = true;
|
|
||||||
sshd.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
services.kmscon.enable = true;
|
|
||||||
|
|
||||||
home-manager.users.${mainUser} = {
|
|
||||||
imports = [
|
|
||||||
self.homeManagerModules.firefox
|
|
||||||
self.homeManagerModules.neovim
|
|
||||||
self.homeManagerModules.shell
|
|
||||||
];
|
|
||||||
|
|
||||||
programs = {
|
|
||||||
bash = {
|
|
||||||
enable = true;
|
|
||||||
promptMainColor = "purple";
|
|
||||||
};
|
|
||||||
|
|
||||||
firefox.enableCustomConf = true;
|
|
||||||
|
|
||||||
neovim = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,117 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
modulesPath,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
nixpkgs.hostPlatform = "x86_64-linux";
|
|
||||||
imports = [(modulesPath + "/installer/scan/not-detected.nix")];
|
|
||||||
|
|
||||||
boot = {
|
|
||||||
kernelPackages = pkgs.linuxPackages_zen;
|
|
||||||
|
|
||||||
kernelParams = [
|
|
||||||
"amd_pstate=active"
|
|
||||||
|
|
||||||
# Remove these if I use plymouth module
|
|
||||||
"quiet"
|
|
||||||
"splash"
|
|
||||||
"boot.shell_on_fail"
|
|
||||||
"i915.fastboot=1"
|
|
||||||
"loglevel=3"
|
|
||||||
"rd.systemd.show_status=false"
|
|
||||||
"rd.udev.log_level=3"
|
|
||||||
"udev.log_priority=3"
|
|
||||||
];
|
|
||||||
kernelModules = ["kvm-amd"];
|
|
||||||
|
|
||||||
# Zenpower for ryzen cpu monitoring
|
|
||||||
extraModulePackages = builtins.attrValues {
|
|
||||||
inherit
|
|
||||||
(config.boot.kernelPackages)
|
|
||||||
v4l2loopback
|
|
||||||
zenpower
|
|
||||||
;
|
|
||||||
};
|
|
||||||
blacklistedKernelModules = ["k10temp"];
|
|
||||||
|
|
||||||
supportedFilesystems = ["ntfs"];
|
|
||||||
|
|
||||||
consoleLogLevel = 0;
|
|
||||||
|
|
||||||
initrd = {
|
|
||||||
verbose = false;
|
|
||||||
systemd.enable = true;
|
|
||||||
availableKernelModules = ["nvme" "xhci_pci" "ahci" "usbhid" "usb_storage" "sd_mod"];
|
|
||||||
};
|
|
||||||
|
|
||||||
loader = {
|
|
||||||
efi.canTouchEfiVariables = true;
|
|
||||||
timeout = 0;
|
|
||||||
|
|
||||||
systemd-boot = {
|
|
||||||
enable = true;
|
|
||||||
consoleMode = "max";
|
|
||||||
configurationLimit = 30;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
fileSystems = {
|
|
||||||
"/" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "btrfs";
|
|
||||||
};
|
|
||||||
|
|
||||||
# sudo btrfs subvolume create /@swap
|
|
||||||
"/swap" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "btrfs";
|
|
||||||
# Idk why this is the subvol
|
|
||||||
options = ["subvol=@/@swap"];
|
|
||||||
};
|
|
||||||
|
|
||||||
"/boot" = {
|
|
||||||
device = "/dev/disk/by-label/NIXBOOT";
|
|
||||||
fsType = "vfat";
|
|
||||||
};
|
|
||||||
|
|
||||||
"/run/media/matt/Games" = {
|
|
||||||
device = "/dev/disk/by-uuid/da62f4ee-d4a6-4fdd-ab12-9c5e131c6f30";
|
|
||||||
fsType = "ext4";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
swapDevices = [
|
|
||||||
{
|
|
||||||
device = "/swap/swapfile";
|
|
||||||
size = 16 * 1024;
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
zramSwap.enable = true;
|
|
||||||
|
|
||||||
hardware = {
|
|
||||||
cpu.amd.updateMicrocode = config.hardware.enableRedistributableFirmware;
|
|
||||||
uinput.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
virtualisation = {
|
|
||||||
libvirtd.enable = true;
|
|
||||||
spiceUSBRedirection.enable = true;
|
|
||||||
};
|
|
||||||
environment.systemPackages = builtins.attrValues {
|
|
||||||
inherit
|
|
||||||
(pkgs)
|
|
||||||
qemu
|
|
||||||
virtiofsd
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
nvidia = {
|
|
||||||
enable = true;
|
|
||||||
enableNvidiaSettings = true;
|
|
||||||
enableWayland = true;
|
|
||||||
enableCUDA = true;
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,6 +0,0 @@
|
||||||
{...}: {
|
|
||||||
imports = [
|
|
||||||
./gpu-replay.nix
|
|
||||||
./nix-gaming.nix
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,85 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
mainUser,
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) concatStringsSep getExe removePrefix;
|
|
||||||
inherit (self.packages.${pkgs.system}) gpu-screen-recorder gsr-kms-server;
|
|
||||||
|
|
||||||
hyprPkgs = config.home-manager.users.${mainUser}.wayland.windowManager.hyprland.finalPackage;
|
|
||||||
|
|
||||||
cfgDesktop = config.roles.desktop;
|
|
||||||
in {
|
|
||||||
security.wrappers = {
|
|
||||||
gpu-screen-recorder = {
|
|
||||||
owner = "root";
|
|
||||||
group = "video";
|
|
||||||
capabilities = "cap_sys_nice+ep";
|
|
||||||
source = getExe gpu-screen-recorder;
|
|
||||||
};
|
|
||||||
|
|
||||||
gsr-kms-server = {
|
|
||||||
owner = "root";
|
|
||||||
group = "video";
|
|
||||||
capabilities = "cap_sys_admin+ep";
|
|
||||||
source = getExe gsr-kms-server;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
home-manager.users.${mainUser} = {
|
|
||||||
home.packages = [
|
|
||||||
(pkgs.writeShellApplication {
|
|
||||||
name = "gpu-save-replay";
|
|
||||||
|
|
||||||
runtimeInputs = with pkgs; [procps];
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
pkill --signal SIGUSR1 -f gpu-screen-recorder
|
|
||||||
'';
|
|
||||||
})
|
|
||||||
|
|
||||||
(pkgs.writeShellApplication {
|
|
||||||
name = "gsr-start";
|
|
||||||
|
|
||||||
runtimeInputs = [
|
|
||||||
pkgs.pulseaudio
|
|
||||||
pkgs.xorg.xrandr
|
|
||||||
|
|
||||||
hyprPkgs
|
|
||||||
];
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
main="${removePrefix "desc:" cfgDesktop.mainMonitor}"
|
|
||||||
WINDOW=$(hyprctl -j monitors | jq '.[] |= (.description |= gsub(","; ""))' | jq -r ".[] | select(.description | test(\"$main\")) | .name")
|
|
||||||
|
|
||||||
# Fix fullscreen game resolution
|
|
||||||
xrandr --output "$WINDOW" --primary
|
|
||||||
|
|
||||||
gpu-screen-recorder ${concatStringsSep " " [
|
|
||||||
''-v no''
|
|
||||||
''-r 1200''
|
|
||||||
''-df yes''
|
|
||||||
''-o /home/matt/Videos/Replay''
|
|
||||||
# Audio settings
|
|
||||||
''-ac aac''
|
|
||||||
''-a "$(pactl get-default-sink).monitor"''
|
|
||||||
''-a "$(pactl get-default-source)"''
|
|
||||||
# Video settings
|
|
||||||
''-w "$WINDOW"''
|
|
||||||
''-f 60''
|
|
||||||
''-c mkv''
|
|
||||||
''-k hevc''
|
|
||||||
''-q very_high''
|
|
||||||
]}
|
|
||||||
'';
|
|
||||||
})
|
|
||||||
];
|
|
||||||
|
|
||||||
wayland.windowManager.hyprland.settings = {
|
|
||||||
bind = [",F8, exec, ags request 'save-replay'"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,38 +0,0 @@
|
||||||
{
|
|
||||||
nix-gaming,
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
imports = [
|
|
||||||
nix-gaming.nixosModules.platformOptimizations
|
|
||||||
];
|
|
||||||
|
|
||||||
programs = {
|
|
||||||
steam = {
|
|
||||||
enable = true;
|
|
||||||
protontricks.enable = true;
|
|
||||||
remotePlay.openFirewall = true;
|
|
||||||
|
|
||||||
extraCompatPackages = [
|
|
||||||
self.packages.${pkgs.system}.proton-ge-latest
|
|
||||||
];
|
|
||||||
|
|
||||||
platformOptimizations.enable = true;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = [
|
|
||||||
(pkgs.lutris.override {
|
|
||||||
extraLibraries = pkgs: [
|
|
||||||
# List library dependencies here
|
|
||||||
];
|
|
||||||
extraPkgs = pkgs: [
|
|
||||||
# List extra packages available to lutris here
|
|
||||||
];
|
|
||||||
})
|
|
||||||
|
|
||||||
pkgs.r2modman
|
|
||||||
pkgs.ryujinx
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,97 +0,0 @@
|
||||||
deviceName: {
|
|
||||||
config,
|
|
||||||
mainUser,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
clusterIP = config.services.pcsd.virtualIps.caddy-vip.ip;
|
|
||||||
in {
|
|
||||||
# ------------------------------------------------
|
|
||||||
# Imports
|
|
||||||
# ------------------------------------------------
|
|
||||||
imports = [
|
|
||||||
./hardware-configuration.nix
|
|
||||||
|
|
||||||
./modules
|
|
||||||
|
|
||||||
self.nixosModules.base
|
|
||||||
self.nixosModules.kmscon
|
|
||||||
self.nixosModules.server
|
|
||||||
];
|
|
||||||
|
|
||||||
config = {
|
|
||||||
# State Version: DO NOT CHANGE
|
|
||||||
system.stateVersion = "24.05";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# User Settings
|
|
||||||
# ------------------------------------------------
|
|
||||||
users.users.${mainUser} = {
|
|
||||||
isNormalUser = true;
|
|
||||||
extraGroups = [
|
|
||||||
"wheel"
|
|
||||||
"adm"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
networking = {
|
|
||||||
hostName = deviceName;
|
|
||||||
resolvconf.enable = true;
|
|
||||||
nameservers = [
|
|
||||||
clusterIP
|
|
||||||
"1.0.0.1"
|
|
||||||
];
|
|
||||||
extraHosts = ''
|
|
||||||
10.0.0.244 thingone
|
|
||||||
10.0.0.159 thingtwo
|
|
||||||
'';
|
|
||||||
firewall.enable = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
time.timeZone = "America/Montreal";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# `Self` Modules configuration
|
|
||||||
# ------------------------------------------------
|
|
||||||
roles.base = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
|
|
||||||
roles.server = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
tailscale.enable = true;
|
|
||||||
sshd.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
services.kmscon.enable = true;
|
|
||||||
|
|
||||||
home-manager.users.${mainUser} = {
|
|
||||||
imports = [
|
|
||||||
self.homeManagerModules.neovim
|
|
||||||
self.homeManagerModules.shell
|
|
||||||
];
|
|
||||||
|
|
||||||
programs = {
|
|
||||||
bash = {
|
|
||||||
enable = true;
|
|
||||||
promptMainColor =
|
|
||||||
if deviceName == "thingone"
|
|
||||||
then "green"
|
|
||||||
else if deviceName == "thingtwo"
|
|
||||||
then "red"
|
|
||||||
else "purple";
|
|
||||||
};
|
|
||||||
|
|
||||||
neovim = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# For accurate stack trace
|
|
||||||
_file = ./default.nix;
|
|
||||||
}
|
|
|
@ -1,59 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
modulesPath,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
nixpkgs.hostPlatform = "x86_64-linux";
|
|
||||||
imports = [(modulesPath + "/installer/scan/not-detected.nix")];
|
|
||||||
|
|
||||||
boot = {
|
|
||||||
loader = {
|
|
||||||
efi.canTouchEfiVariables = true;
|
|
||||||
timeout = 2;
|
|
||||||
|
|
||||||
systemd-boot = {
|
|
||||||
enable = true;
|
|
||||||
consoleMode = "max";
|
|
||||||
configurationLimit = 30;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
initrd.availableKernelModules = [
|
|
||||||
"xhci_pci"
|
|
||||||
"ahci"
|
|
||||||
"usbhid"
|
|
||||||
"usb_storage"
|
|
||||||
"sd_mod"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
fileSystems = {
|
|
||||||
"/" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "btrfs";
|
|
||||||
};
|
|
||||||
|
|
||||||
# sudo btrfs subvolume create /@swap
|
|
||||||
"/swap" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "btrfs";
|
|
||||||
options = ["subvol=@swap"];
|
|
||||||
};
|
|
||||||
|
|
||||||
"/boot" = {
|
|
||||||
device = "/dev/disk/by-label/NIXBOOT";
|
|
||||||
fsType = "vfat";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
swapDevices = [
|
|
||||||
{
|
|
||||||
device = "/swap/swapfile";
|
|
||||||
size = 16 * 1024;
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
zramSwap.enable = true;
|
|
||||||
|
|
||||||
hardware.cpu.intel.updateMicrocode = config.hardware.enableRedistributableFirmware;
|
|
||||||
}
|
|
|
@ -1,24 +0,0 @@
|
||||||
{...}: {
|
|
||||||
services = {
|
|
||||||
blocky = {
|
|
||||||
enable = true;
|
|
||||||
|
|
||||||
settings = {
|
|
||||||
upstream = {
|
|
||||||
default = [
|
|
||||||
"127.0.0.1:5335"
|
|
||||||
"127.0.0.1:5335"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
blocking = {
|
|
||||||
blackLists = {
|
|
||||||
ads = [
|
|
||||||
"https://raw.githubusercontent.com/StevenBlack/hosts/master/hosts"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,175 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
mainUser,
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) attrValues removeAttrs;
|
|
||||||
|
|
||||||
inherit (config.sops) secrets;
|
|
||||||
inherit (config.networking) hostName;
|
|
||||||
in {
|
|
||||||
imports = [self.nixosModules.caddy-plus];
|
|
||||||
|
|
||||||
users.users.${mainUser}.extraGroups = ["caddy"];
|
|
||||||
|
|
||||||
boot.kernel.sysctl."net.ipv4.ip_nonlocal_bind" = 1;
|
|
||||||
|
|
||||||
systemd.services.caddy.serviceConfig = {
|
|
||||||
EnvironmentFile = secrets.caddy-cloudflare.path;
|
|
||||||
};
|
|
||||||
|
|
||||||
services.caddy = {
|
|
||||||
enable = true;
|
|
||||||
enableReload = false;
|
|
||||||
|
|
||||||
package = let
|
|
||||||
pluginsInfo = import ./plugins.nix;
|
|
||||||
in
|
|
||||||
pkgs.caddy.withPlugins {
|
|
||||||
plugins = map (x: "${x.url}@${x.version}") (attrValues pluginsInfo.plugins);
|
|
||||||
inherit (pluginsInfo) hash;
|
|
||||||
};
|
|
||||||
|
|
||||||
virtualHosts = let
|
|
||||||
clusterIP = config.services.pcsd.virtualIps.caddy-vip.ip;
|
|
||||||
nosIP = "10.0.0.121";
|
|
||||||
serviviIP = "10.0.0.249";
|
|
||||||
homieIP = "100.64.0.10";
|
|
||||||
|
|
||||||
tlsConf = ''
|
|
||||||
tls {
|
|
||||||
dns cloudflare {$CLOUDFLARE_API_TOKEN}
|
|
||||||
resolvers 1.0.0.1
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
|
|
||||||
mkPublicReverseProxy = subdomain: ip: extraConf:
|
|
||||||
{
|
|
||||||
hostName = "${subdomain}.nelim.org";
|
|
||||||
reverseProxy = ip;
|
|
||||||
listenAddresses = [clusterIP];
|
|
||||||
extraConfig = tlsConf + (extraConf.extraConfig or "");
|
|
||||||
}
|
|
||||||
// (removeAttrs extraConf ["extraConfig"]);
|
|
||||||
in {
|
|
||||||
# Public
|
|
||||||
"Home-Assistant" = mkPublicReverseProxy "homie" "${homieIP}:8123" {};
|
|
||||||
"Vaultwarden" = mkPublicReverseProxy "vault" "${nosIP}:8781" {};
|
|
||||||
"Hauk" = mkPublicReverseProxy "hauk" "${nosIP}:3003" {};
|
|
||||||
"Headscale" = mkPublicReverseProxy "headscale" "${clusterIP}:8085" {};
|
|
||||||
|
|
||||||
"Jellyfin" = mkPublicReverseProxy "jelly" "${nosIP}:8096" {
|
|
||||||
subDirectories.jfa-go = {
|
|
||||||
subDirName = "accounts";
|
|
||||||
reverseProxy = "${nosIP}:8056";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
"Jellyseer" = mkPublicReverseProxy "seerr" "${nosIP}:5055" {};
|
|
||||||
|
|
||||||
"Gameyfin" = mkPublicReverseProxy "games" "${nosIP}:8074" {};
|
|
||||||
|
|
||||||
"Forgejo" = mkPublicReverseProxy "git" "${nosIP}:3000" {};
|
|
||||||
|
|
||||||
"Nextcloud" = mkPublicReverseProxy "cloud" "${nosIP}:8042" {
|
|
||||||
extraConfig = ''
|
|
||||||
redir /.well-known/carddav /remote.php/dav 301
|
|
||||||
redir /.well-known/caldav /remote.php/dav 301
|
|
||||||
redir /.well-known/webfinger /index.php/.well-known/webfinger 301
|
|
||||||
redir /.well-known/nodeinfo /index.php/.well-known/nodeinfo 301
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
"OnlyOffice" = mkPublicReverseProxy "office" "http://${nosIP}:8055" {};
|
|
||||||
|
|
||||||
"Immich" = mkPublicReverseProxy "photos" "${nosIP}:2283" {};
|
|
||||||
|
|
||||||
"Binary Cache" = mkPublicReverseProxy "cache" "${serviviIP}:5000" {};
|
|
||||||
"Apt Binary Cache" = mkPublicReverseProxy "cache-apt" "${homieIP}:5000" {};
|
|
||||||
|
|
||||||
# Private
|
|
||||||
"nelim.org" = {
|
|
||||||
serverAliases = ["*.nelim.org"];
|
|
||||||
extraConfig = tlsConf;
|
|
||||||
listenAddresses = [
|
|
||||||
(
|
|
||||||
if hostName == "thingone"
|
|
||||||
then "100.64.0.8"
|
|
||||||
else "100.64.0.9"
|
|
||||||
)
|
|
||||||
];
|
|
||||||
|
|
||||||
subDomains = {
|
|
||||||
esphome.reverseProxy = "${homieIP}:6052";
|
|
||||||
pr-tracker.reverseProxy = "${serviviIP}:3000";
|
|
||||||
|
|
||||||
pcsd = {
|
|
||||||
extraConfig = ''
|
|
||||||
reverse_proxy https://${clusterIP}:2224 {
|
|
||||||
transport http {
|
|
||||||
tls_insecure_skip_verify
|
|
||||||
}
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
# Resume builder
|
|
||||||
resume.reverseProxy = "${nosIP}:3060";
|
|
||||||
resauth.reverseProxy = "${nosIP}:3100";
|
|
||||||
|
|
||||||
# FreshRSS & Co
|
|
||||||
bridge.reverseProxy = "${nosIP}:3006";
|
|
||||||
drss.reverseProxy = "${nosIP}:3007";
|
|
||||||
freshrss = {
|
|
||||||
subDomainName = "rss";
|
|
||||||
reverseProxy = "${nosIP}:2800";
|
|
||||||
};
|
|
||||||
|
|
||||||
wgui.reverseProxy = "${nosIP}:51821";
|
|
||||||
|
|
||||||
lan = {
|
|
||||||
reverseProxy = "${nosIP}:3020";
|
|
||||||
extraConfig = ''
|
|
||||||
redir /index.html /
|
|
||||||
'';
|
|
||||||
|
|
||||||
subDirectories = {
|
|
||||||
bazarr.reverseProxy = "${nosIP}:6767";
|
|
||||||
prowlarr.reverseProxy = "${nosIP}:9696";
|
|
||||||
radarr.reverseProxy = "${nosIP}:7878";
|
|
||||||
sabnzbd.reverseProxy = "${nosIP}:8382";
|
|
||||||
sonarr.reverseProxy = "${nosIP}:8989";
|
|
||||||
|
|
||||||
qbittorent = {
|
|
||||||
subDirName = "qbt";
|
|
||||||
experimental = true;
|
|
||||||
reverseProxy = "${nosIP}:8080";
|
|
||||||
};
|
|
||||||
|
|
||||||
vaultwarden = {
|
|
||||||
subDirName = "vault";
|
|
||||||
experimental = true;
|
|
||||||
reverseProxy = "${nosIP}:8780";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# Top secret Business
|
|
||||||
joal.extraConfig = ''
|
|
||||||
route {
|
|
||||||
rewrite * /joal/ui{uri}
|
|
||||||
reverse_proxy * ${nosIP}:5656
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
joalws.extraConfig = ''
|
|
||||||
route {
|
|
||||||
reverse_proxy ${nosIP}:5656
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,11 +0,0 @@
|
||||||
# This file was autogenerated. DO NOT EDIT!
|
|
||||||
{
|
|
||||||
plugins = {
|
|
||||||
cloudflare = {
|
|
||||||
url = "github.com/caddy-dns/cloudflare";
|
|
||||||
version = "v0.0.0-20240703190432-89f16b99c18e";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
hash = "sha256-JoujVXRXjKUam1Ej3/zKVvF0nX97dUizmISjy3M3Kr8=";
|
|
||||||
}
|
|
|
@ -1,10 +0,0 @@
|
||||||
{...}: {
|
|
||||||
imports = [
|
|
||||||
./blocky.nix
|
|
||||||
./caddy
|
|
||||||
./headscale
|
|
||||||
./nfs-client.nix
|
|
||||||
./pcsd.nix
|
|
||||||
./unbound.nix
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,70 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
mainUser,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (config.networking) hostName;
|
|
||||||
|
|
||||||
clusterIP = config.services.pcsd.virtualIps.caddy-vip.ip;
|
|
||||||
in {
|
|
||||||
users.users.${mainUser}.extraGroups = ["headscale"];
|
|
||||||
|
|
||||||
services.headscale = {
|
|
||||||
enable = true;
|
|
||||||
|
|
||||||
settings = {
|
|
||||||
server_url = "https://headscale.nelim.org";
|
|
||||||
listen_addr = "${clusterIP}:8085";
|
|
||||||
prefixes = {
|
|
||||||
v4 = "100.64.0.0/10";
|
|
||||||
v6 = "fd7a:115c:a1e0::/48";
|
|
||||||
};
|
|
||||||
metrics_listen_addr = "127.0.0.1:9090";
|
|
||||||
grpc_listen_addr = "0.0.0.0:50443";
|
|
||||||
grpc_allow_insecure = false;
|
|
||||||
disable_check_updates = true;
|
|
||||||
ephemeral_node_inactivity_timeout = "30m";
|
|
||||||
unix_socket = "/run/headscale/headscale.sock";
|
|
||||||
unix_socket_permission = "0770";
|
|
||||||
|
|
||||||
database = {
|
|
||||||
type = "sqlite";
|
|
||||||
sqlite.path = "/var/lib/headscale/db.sqlite";
|
|
||||||
};
|
|
||||||
|
|
||||||
private_key_path = "/var/lib/headscale/private.key";
|
|
||||||
noise.private_key_path = "/var/lib/headscale/noise_private.key";
|
|
||||||
|
|
||||||
dns = let
|
|
||||||
caddyIp =
|
|
||||||
if hostName == "thingone"
|
|
||||||
then "100.64.0.8"
|
|
||||||
else "100.64.0.9";
|
|
||||||
in {
|
|
||||||
magic_dns = false;
|
|
||||||
override_local_dns = true;
|
|
||||||
nameservers.global = [caddyIp];
|
|
||||||
};
|
|
||||||
|
|
||||||
log = {
|
|
||||||
format = "text";
|
|
||||||
level = "info";
|
|
||||||
};
|
|
||||||
|
|
||||||
derp = {
|
|
||||||
auto_update_enable = true;
|
|
||||||
update_frequency = "24h";
|
|
||||||
|
|
||||||
server = {
|
|
||||||
enabled = true;
|
|
||||||
stun_listen_addr = "${clusterIP}:3479";
|
|
||||||
private_key_path = "/var/lib/headscale/derp_server_private.key";
|
|
||||||
|
|
||||||
region_id = 995;
|
|
||||||
region_code = "mon";
|
|
||||||
region_name = "montreal";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,32 +0,0 @@
|
||||||
{pkgs, ...}: {
|
|
||||||
# NFS client setup
|
|
||||||
services.rpcbind.enable = true;
|
|
||||||
boot.supportedFilesystems = ["nfs"];
|
|
||||||
environment.systemPackages = builtins.attrValues {
|
|
||||||
inherit (pkgs) nfs-utils;
|
|
||||||
};
|
|
||||||
|
|
||||||
systemd.mounts = let
|
|
||||||
host = "10.0.0.249";
|
|
||||||
in [
|
|
||||||
{
|
|
||||||
type = "nfs";
|
|
||||||
mountConfig = {
|
|
||||||
Options = "noatime";
|
|
||||||
};
|
|
||||||
what = "${host}:/caddy";
|
|
||||||
where = "/var/lib/caddy";
|
|
||||||
requiredBy = ["caddy.service"];
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
type = "nfs";
|
|
||||||
mountConfig = {
|
|
||||||
Options = "noatime";
|
|
||||||
};
|
|
||||||
what = "${host}:/headscale";
|
|
||||||
where = "/var/lib/headscale";
|
|
||||||
requiredBy = ["headscale.service"];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,67 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
pcsd,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (config.sops) secrets;
|
|
||||||
in {
|
|
||||||
imports = [pcsd.nixosModules.default];
|
|
||||||
|
|
||||||
services.pcsd = {
|
|
||||||
enable = true;
|
|
||||||
enableBinaryCache = true;
|
|
||||||
enableWebUI = true;
|
|
||||||
|
|
||||||
clusterName = "thingies";
|
|
||||||
|
|
||||||
corosyncKeyFile = secrets.corosync.path;
|
|
||||||
clusterUserPasswordFile = secrets.pcs-pass.path;
|
|
||||||
|
|
||||||
virtualIps = {
|
|
||||||
"caddy-vip" = {
|
|
||||||
ip = "10.0.0.130";
|
|
||||||
interface = "eno1";
|
|
||||||
group = "caddy-grp";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
systemdResources = {
|
|
||||||
"unbound" = {
|
|
||||||
enable = true;
|
|
||||||
group = "caddy-grp";
|
|
||||||
startAfter = ["caddy-vip"];
|
|
||||||
};
|
|
||||||
|
|
||||||
"blocky" = {
|
|
||||||
enable = true;
|
|
||||||
group = "caddy-grp";
|
|
||||||
startAfter = ["unbound"];
|
|
||||||
};
|
|
||||||
|
|
||||||
"headscale" = {
|
|
||||||
enable = true;
|
|
||||||
group = "caddy-grp";
|
|
||||||
startAfter = ["blocky"];
|
|
||||||
};
|
|
||||||
|
|
||||||
"caddy" = {
|
|
||||||
enable = true;
|
|
||||||
group = "caddy-grp";
|
|
||||||
startAfter = ["headscale"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
nodes = [
|
|
||||||
{
|
|
||||||
name = "thingone";
|
|
||||||
nodeid = 1;
|
|
||||||
ring_addrs = ["10.0.0.244"];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
name = "thingtwo";
|
|
||||||
nodeid = 2;
|
|
||||||
ring_addrs = ["10.0.0.159"];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,94 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
mainUser,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (self.lib) mergeAttrsList;
|
|
||||||
|
|
||||||
inherit (lib) mapAttrsToList remove;
|
|
||||||
|
|
||||||
inherit (config.networking) hostName;
|
|
||||||
|
|
||||||
serviviIP = "100.64.0.7";
|
|
||||||
caddyIp =
|
|
||||||
if hostName == "thingone"
|
|
||||||
then "100.64.0.8"
|
|
||||||
else "100.64.0.9";
|
|
||||||
in {
|
|
||||||
# https://github.com/MatthewVance/unbound-docker-rpi/issues/4#issuecomment-1001879602
|
|
||||||
boot.kernel.sysctl."net.core.rmem_max" = 1048576;
|
|
||||||
|
|
||||||
users.users.${mainUser}.extraGroups = ["unbound"];
|
|
||||||
|
|
||||||
services.unbound = {
|
|
||||||
enable = true;
|
|
||||||
enableRootTrustAnchor = true;
|
|
||||||
resolveLocalQueries = false;
|
|
||||||
|
|
||||||
settings = {
|
|
||||||
server = let
|
|
||||||
mkLocalEntry = domain: ip: {
|
|
||||||
local-zone = ["${domain} redirect"];
|
|
||||||
local-data = ["\"${domain} IN A ${ip}\""];
|
|
||||||
};
|
|
||||||
|
|
||||||
mkMinecraftEntry = domain: port: {
|
|
||||||
local-zone = ["${domain} transparent"];
|
|
||||||
local-data = [
|
|
||||||
"\"${domain} IN A ${serviviIP}\""
|
|
||||||
"\"_minecraft._tcp.${domain}. 180 IN SRV 0 0 ${toString port} ${domain}.\""
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
forceResolveEntry = domain: {
|
|
||||||
local-zone = ["${domain} always_transparent"];
|
|
||||||
};
|
|
||||||
|
|
||||||
publicApps = remove "nelim.org" (mapAttrsToList (n: v: v.hostName) config.services.caddy.virtualHosts);
|
|
||||||
in
|
|
||||||
mergeAttrsList (
|
|
||||||
[(mkLocalEntry "cache-apt.nelim.org" "100.64.0.10")]
|
|
||||||
++ (map forceResolveEntry publicApps)
|
|
||||||
++ [
|
|
||||||
(mkMinecraftEntry "mc.nelim.org" 25569)
|
|
||||||
(mkMinecraftEntry "mc2.nelim.org" 25560)
|
|
||||||
(mkMinecraftEntry "cv.nelim.org" 25566)
|
|
||||||
|
|
||||||
(mkLocalEntry "nelim.org" caddyIp)
|
|
||||||
|
|
||||||
{
|
|
||||||
interface = ["127.0.0.1"];
|
|
||||||
port = 5335;
|
|
||||||
|
|
||||||
do-ip4 = true;
|
|
||||||
do-ip6 = false;
|
|
||||||
prefer-ip6 = false;
|
|
||||||
do-udp = true;
|
|
||||||
do-tcp = true;
|
|
||||||
|
|
||||||
# Performance
|
|
||||||
prefetch = true;
|
|
||||||
num-threads = 1;
|
|
||||||
|
|
||||||
private-address = [
|
|
||||||
"172.16.0.0/12"
|
|
||||||
"10.0.0.0/8"
|
|
||||||
"100.64.0.0/8"
|
|
||||||
"fd00::/8"
|
|
||||||
"fe80::/10"
|
|
||||||
];
|
|
||||||
|
|
||||||
# Default stuff
|
|
||||||
harden-glue = true;
|
|
||||||
harden-dnssec-stripped = true;
|
|
||||||
use-caps-for-id = false;
|
|
||||||
edns-buffer-size = 1232;
|
|
||||||
so-rcvbuf = "1m";
|
|
||||||
}
|
|
||||||
]
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,78 +0,0 @@
|
||||||
{
|
|
||||||
mainUser,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
# ------------------------------------------------
|
|
||||||
# Imports
|
|
||||||
# ------------------------------------------------
|
|
||||||
imports = [
|
|
||||||
./hardware-configuration.nix
|
|
||||||
|
|
||||||
./modules
|
|
||||||
|
|
||||||
self.nixosModules.base
|
|
||||||
self.nixosModules.docker
|
|
||||||
self.nixosModules.kmscon
|
|
||||||
self.nixosModules.server
|
|
||||||
];
|
|
||||||
|
|
||||||
# State Version: DO NOT CHANGE
|
|
||||||
system.stateVersion = "24.11";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# User Settings
|
|
||||||
# ------------------------------------------------
|
|
||||||
users.users.${mainUser} = {
|
|
||||||
isNormalUser = true;
|
|
||||||
extraGroups = [
|
|
||||||
"wheel"
|
|
||||||
"adm"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
networking = {
|
|
||||||
hostName = "homie";
|
|
||||||
resolvconf.enable = true;
|
|
||||||
firewall.enable = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
time.timeZone = "America/Montreal";
|
|
||||||
|
|
||||||
# ------------------------------------------------
|
|
||||||
# `Self` Modules configuration
|
|
||||||
# ------------------------------------------------
|
|
||||||
roles.base = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
|
|
||||||
roles.server = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
tailscale.enable = true;
|
|
||||||
sshd.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
khepri.enable = true;
|
|
||||||
services.kmscon.enable = true;
|
|
||||||
|
|
||||||
home-manager.users.${mainUser} = {
|
|
||||||
imports = [
|
|
||||||
self.homeManagerModules.neovim
|
|
||||||
self.homeManagerModules.shell
|
|
||||||
];
|
|
||||||
|
|
||||||
programs = {
|
|
||||||
bash = {
|
|
||||||
enable = true;
|
|
||||||
promptMainColor = "yellow";
|
|
||||||
};
|
|
||||||
|
|
||||||
neovim = {
|
|
||||||
enable = true;
|
|
||||||
user = mainUser;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,61 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
modulesPath,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
nixpkgs.hostPlatform = "x86_64-linux";
|
|
||||||
imports = [(modulesPath + "/installer/scan/not-detected.nix")];
|
|
||||||
|
|
||||||
boot = {
|
|
||||||
loader = {
|
|
||||||
efi.canTouchEfiVariables = true;
|
|
||||||
timeout = 2;
|
|
||||||
|
|
||||||
systemd-boot = {
|
|
||||||
enable = true;
|
|
||||||
consoleMode = "max";
|
|
||||||
configurationLimit = 30;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
initrd.availableKernelModules = [
|
|
||||||
"xhci_pci"
|
|
||||||
"ahci"
|
|
||||||
"nvme"
|
|
||||||
"usbhid"
|
|
||||||
"usb_storage"
|
|
||||||
"sd_mod"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
fileSystems = {
|
|
||||||
"/" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "btrfs";
|
|
||||||
};
|
|
||||||
|
|
||||||
# sudo btrfs subvolume create /@swap
|
|
||||||
"/swap" = {
|
|
||||||
device = "/dev/disk/by-label/NIXROOT";
|
|
||||||
fsType = "btrfs";
|
|
||||||
options = ["subvol=@swap"];
|
|
||||||
};
|
|
||||||
|
|
||||||
"/boot" = {
|
|
||||||
device = "/dev/disk/by-label/NIXBOOT";
|
|
||||||
fsType = "vfat";
|
|
||||||
options = ["fmask=0022" "dmask=0022"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
swapDevices = [
|
|
||||||
{
|
|
||||||
device = "/swap/swapfile";
|
|
||||||
size = 16 * 1024;
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
zramSwap.enable = true;
|
|
||||||
|
|
||||||
hardware.cpu.intel.updateMicrocode = config.hardware.enableRedistributableFirmware;
|
|
||||||
}
|
|
|
@ -1,102 +0,0 @@
|
||||||
# Unfortunately I had some hardware issues but this does work
|
|
||||||
{
|
|
||||||
lib,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) mkForce getExe;
|
|
||||||
|
|
||||||
connectControllers = getExe (pkgs.writeShellApplication {
|
|
||||||
name = "connectControllers";
|
|
||||||
runtimeInputs = with pkgs; [gnugrep usbutils];
|
|
||||||
text = ''
|
|
||||||
set +o errexit
|
|
||||||
|
|
||||||
for dev in /sys/bus/usb/devices/*; do
|
|
||||||
vendor="$(cat "$dev/idVendor" 2>/dev/null)"
|
|
||||||
prod="$(cat "$dev/idProduct" 2>/dev/null)"
|
|
||||||
|
|
||||||
if [[ "$vendor" != "" && "$prod" != "" ]]; then
|
|
||||||
if [[ "$(lsusb -d "$vendor:$prod" | grep "Microsoft Corp. Xbox Controller")" != "" ]]; then
|
|
||||||
echo 0 > "$dev/authorized"
|
|
||||||
echo 1 > "$dev/authorized"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
'';
|
|
||||||
});
|
|
||||||
|
|
||||||
hyprConf = pkgs.writeText "greetd-hypr-config" ''
|
|
||||||
cursor {
|
|
||||||
inactive_timeout = 1
|
|
||||||
}
|
|
||||||
|
|
||||||
misc {
|
|
||||||
disable_hyprland_logo = true
|
|
||||||
disable_splash_rendering = true
|
|
||||||
}
|
|
||||||
|
|
||||||
decoration {
|
|
||||||
blur {
|
|
||||||
enabled = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
animations {
|
|
||||||
enabled = false
|
|
||||||
first_launch_animation = false
|
|
||||||
}
|
|
||||||
|
|
||||||
bind = SUPER, Q, exec, kitty
|
|
||||||
|
|
||||||
windowrule = fullscreen, ^(.*)$
|
|
||||||
exec-once = waydroid show-full-ui
|
|
||||||
exec-once = sleep 10; sudo ${connectControllers}
|
|
||||||
'';
|
|
||||||
|
|
||||||
user = "matt";
|
|
||||||
command = "Hyprland --config ${hyprConf}";
|
|
||||||
|
|
||||||
session = {inherit command user;};
|
|
||||||
in {
|
|
||||||
# Make it so we don't need root to connect controllers
|
|
||||||
security.sudo.extraRules = [
|
|
||||||
{
|
|
||||||
users = [user];
|
|
||||||
groups = [user];
|
|
||||||
commands = [
|
|
||||||
{
|
|
||||||
command = connectControllers;
|
|
||||||
options = ["SETENV" "NOPASSWD"];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
# TODO: make the following declarative and also make the image declarative
|
|
||||||
# Add this to /var/lib/waydroid/waydroid.cfg for controller support
|
|
||||||
# persist.waydroid.udev = true
|
|
||||||
# persist.waydroid.uevent = true
|
|
||||||
virtualisation.waydroid.enable = true;
|
|
||||||
|
|
||||||
users.users."greeter" = {
|
|
||||||
home = "/var/lib/greeter";
|
|
||||||
};
|
|
||||||
|
|
||||||
programs.hyprland.enable = true;
|
|
||||||
|
|
||||||
services = {
|
|
||||||
greetd = {
|
|
||||||
enable = true;
|
|
||||||
|
|
||||||
settings = {
|
|
||||||
default_session = session;
|
|
||||||
initial_session = session;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
pipewire.enable = mkForce false;
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = [pkgs.kitty];
|
|
||||||
}
|
|
|
@ -1,34 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (builtins) attrValues;
|
|
||||||
inherit (config.sops) secrets;
|
|
||||||
|
|
||||||
nixFastBuild = pkgs.writeShellApplication {
|
|
||||||
name = "nixFastBuild";
|
|
||||||
|
|
||||||
runtimeInputs = attrValues {
|
|
||||||
inherit
|
|
||||||
(pkgs)
|
|
||||||
gnugrep
|
|
||||||
nix-fast-build
|
|
||||||
nix-output-monitor
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
cd "$FLAKE/results" || return
|
|
||||||
|
|
||||||
nix-fast-build -f ..#nixFastChecks.aptDevices "$@"
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
in {
|
|
||||||
services.nix-serve = {
|
|
||||||
enable = true;
|
|
||||||
secretKeyFile = secrets.apt-binary-cache-key.path;
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = [pkgs.nix-fast-build nixFastBuild];
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
{...}: {
|
|
||||||
imports = [
|
|
||||||
./binary-cache.nix
|
|
||||||
./home-assistant
|
|
||||||
./music
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,83 +0,0 @@
|
||||||
{
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
wakewords-src,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
imports = [
|
|
||||||
self.nixosModules.esphome-plus
|
|
||||||
self.nixosModules.wyoming-plus
|
|
||||||
];
|
|
||||||
|
|
||||||
services = {
|
|
||||||
home-assistant = {
|
|
||||||
package = pkgs.home-assistant.override {
|
|
||||||
packageOverrides = final: prev: {
|
|
||||||
# HassTimer has way too many collisions with my custom timer sentences
|
|
||||||
home-assistant-intents = prev.home-assistant-intents.overrideAttrs (o: {
|
|
||||||
nativeBuildInputs = o.nativeBuildInputs ++ [pkgs.findutils];
|
|
||||||
postPatch = ''
|
|
||||||
find ./. -name "*Timer*" -delete
|
|
||||||
find ./. -name "*Start*" -delete
|
|
||||||
'';
|
|
||||||
});
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
customComponents = builtins.attrValues {
|
|
||||||
inherit
|
|
||||||
(self.scopedPackages.${pkgs.system}.hass-components)
|
|
||||||
extended-ollama-conversation # url is without subdirectory
|
|
||||||
ha-fallback-conversation
|
|
||||||
tuya-local
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
extraComponents = [
|
|
||||||
"esphome"
|
|
||||||
"ollama"
|
|
||||||
"wyoming"
|
|
||||||
"scrape"
|
|
||||||
];
|
|
||||||
|
|
||||||
config = {
|
|
||||||
assist_pipeline = {};
|
|
||||||
conversation = {};
|
|
||||||
media_source = {};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
wyoming = {
|
|
||||||
piper.servers."en" = {
|
|
||||||
enable = true;
|
|
||||||
uri = "tcp://127.0.0.1:10200";
|
|
||||||
|
|
||||||
# see https://github.com/rhasspy/rhasspy3/blob/master/programs/tts/piper/script/download.py
|
|
||||||
voice = "en_US-hfc_male-medium";
|
|
||||||
speaker = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
openwakeword = {
|
|
||||||
enable = true;
|
|
||||||
uri = "tcp://127.0.0.1:10400";
|
|
||||||
|
|
||||||
threshold = 0.55;
|
|
||||||
vadThreshold = 0.50;
|
|
||||||
|
|
||||||
customModelsDirectories = ["${wakewords-src}/en/yo_homie"];
|
|
||||||
preloadModels = ["yo_homie"];
|
|
||||||
|
|
||||||
extraArgs = ["--debug"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
esphome = {
|
|
||||||
enable = true;
|
|
||||||
address = "100.64.0.10";
|
|
||||||
port = 6052;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# In case tailscale is down
|
|
||||||
boot.kernel.sysctl."net.ipv4.ip_nonlocal_bind" = 1;
|
|
||||||
}
|
|
|
@ -1,78 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
lib,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) getExe;
|
|
||||||
|
|
||||||
turnOnUE = pkgs.writeShellApplication {
|
|
||||||
name = "turnOnUE";
|
|
||||||
|
|
||||||
runtimeInputs = [config.hardware.bluetooth.package];
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
cmd=0x0003
|
|
||||||
bt_device_addr=88:C6:26:93:4B:77
|
|
||||||
|
|
||||||
# This is the MAC address on the first line of `bluetootctl show`
|
|
||||||
# without the `:` and with `01` added at the end
|
|
||||||
bt_controller_addr=E848B8C8200001
|
|
||||||
|
|
||||||
exec gatttool -b $bt_device_addr --char-write-req --handle=$cmd --value=$bt_controller_addr
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
in {
|
|
||||||
environment.systemPackages = [turnOnUE];
|
|
||||||
|
|
||||||
services.home-assistant = {
|
|
||||||
extraComponents = [
|
|
||||||
"mpd"
|
|
||||||
|
|
||||||
# BT components
|
|
||||||
"ibeacon"
|
|
||||||
"led_ble"
|
|
||||||
"kegtron"
|
|
||||||
"xiaomi_ble"
|
|
||||||
];
|
|
||||||
|
|
||||||
# Turn On the speaker automatically when openwakeword is used
|
|
||||||
config = {
|
|
||||||
shell_command.turn_on_ue = getExe turnOnUE;
|
|
||||||
script.turn_on_ue = {
|
|
||||||
alias = "Music - TurnOnUE";
|
|
||||||
description = "Script for turning on the UE Boom 2 speaker.";
|
|
||||||
icon = "mdi:music";
|
|
||||||
mode = "single";
|
|
||||||
|
|
||||||
sequence = [
|
|
||||||
{
|
|
||||||
alias = "Run shell command";
|
|
||||||
service = "shell_command.turn_on_ue";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
automation = [
|
|
||||||
{
|
|
||||||
alias = "Turn On UE";
|
|
||||||
mode = "single";
|
|
||||||
trigger = [
|
|
||||||
{
|
|
||||||
platform = "state";
|
|
||||||
entity_id = "wake_word.openwakeword";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
condition = [];
|
|
||||||
action = [
|
|
||||||
{
|
|
||||||
action = "shell_command.turn_on_ue";
|
|
||||||
metadata = {};
|
|
||||||
data = {};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,119 +0,0 @@
|
||||||
{
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: {
|
|
||||||
imports = [
|
|
||||||
./assist.nix
|
|
||||||
./bluetooth.nix
|
|
||||||
./firmware.nix
|
|
||||||
./frontend.nix
|
|
||||||
./zigbee.nix
|
|
||||||
|
|
||||||
./netdaemon
|
|
||||||
./spotify
|
|
||||||
./timer
|
|
||||||
|
|
||||||
self.nixosModules.ha-plus
|
|
||||||
];
|
|
||||||
|
|
||||||
services.home-assistant = {
|
|
||||||
enable = true;
|
|
||||||
|
|
||||||
extraComponents = [
|
|
||||||
"androidtv"
|
|
||||||
"androidtv_remote"
|
|
||||||
"caldav"
|
|
||||||
"cast"
|
|
||||||
"holiday"
|
|
||||||
"isal"
|
|
||||||
"met"
|
|
||||||
"switchbot"
|
|
||||||
"upnp"
|
|
||||||
"yamaha_musiccast"
|
|
||||||
];
|
|
||||||
|
|
||||||
customComponents = builtins.attrValues {
|
|
||||||
inherit
|
|
||||||
(self.scopedPackages.${pkgs.system}.hass-components)
|
|
||||||
yamaha-soundbar
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
config = {
|
|
||||||
homeassistant = {
|
|
||||||
name = "Home";
|
|
||||||
unit_system = "metric";
|
|
||||||
currency = "CAD";
|
|
||||||
country = "CA";
|
|
||||||
time_zone = "America/Montreal";
|
|
||||||
external_url = "https://homie.nelim.org";
|
|
||||||
};
|
|
||||||
|
|
||||||
media_player = [
|
|
||||||
{
|
|
||||||
platform = "yamaha_soundbar";
|
|
||||||
host = "192.168.0.96";
|
|
||||||
name = "Living Room Speaker";
|
|
||||||
sources = {
|
|
||||||
HDMI = "TV";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
# Proxy settings
|
|
||||||
http = {
|
|
||||||
server_host = "0.0.0.0";
|
|
||||||
trusted_proxies = ["100.64.0.8" "100.64.0.9"];
|
|
||||||
use_x_forwarded_for = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
# `default_config` enables too much stuff. this is what I want from it
|
|
||||||
config = {};
|
|
||||||
dhcp = {};
|
|
||||||
history = {};
|
|
||||||
image_upload = {};
|
|
||||||
logbook = {};
|
|
||||||
mobile_app = {};
|
|
||||||
my = {};
|
|
||||||
sun = {};
|
|
||||||
zeroconf = {};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = [
|
|
||||||
(pkgs.writeShellApplication {
|
|
||||||
name = "yaml2nix";
|
|
||||||
runtimeInputs = with pkgs; [yj];
|
|
||||||
text = ''
|
|
||||||
input="$(yj < "$1")"
|
|
||||||
output="''${2:-""}"
|
|
||||||
|
|
||||||
nixCode="$(nix eval --expr "builtins.fromJSON '''$input'''" --impure | alejandra -q | sed 's/ = null;/ = {};/g')"
|
|
||||||
|
|
||||||
if [[ "$output" != "" ]]; then
|
|
||||||
echo "$nixCode" > "$output"
|
|
||||||
else
|
|
||||||
echo "$nixCode"
|
|
||||||
fi
|
|
||||||
'';
|
|
||||||
})
|
|
||||||
|
|
||||||
(pkgs.writeShellApplication {
|
|
||||||
name = "nix2yaml";
|
|
||||||
runtimeInputs = with pkgs; [remarshal];
|
|
||||||
text = ''
|
|
||||||
input="$1"
|
|
||||||
output="''${2:-""}"
|
|
||||||
|
|
||||||
yamlCode="$(nix eval --json --file "$input" | remarshal --if json --of yaml)"
|
|
||||||
|
|
||||||
if [[ "$output" != "" ]]; then
|
|
||||||
echo "$yamlCode" > "$output"
|
|
||||||
else
|
|
||||||
echo "$yamlCode"
|
|
||||||
fi
|
|
||||||
'';
|
|
||||||
})
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,26 +0,0 @@
|
||||||
# I use nix2yaml from ../default.nix to convert this to YAML and place it in the functions of extended_ollama_conversation
|
|
||||||
[
|
|
||||||
{
|
|
||||||
spec = {
|
|
||||||
name = "get_attributes";
|
|
||||||
description = "Get attributes of any home assistant entity";
|
|
||||||
parameters = {
|
|
||||||
type = "object";
|
|
||||||
|
|
||||||
properties = {
|
|
||||||
entity_id = {
|
|
||||||
type = "string";
|
|
||||||
description = "entity_id";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
required = ["entity_id"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
function = {
|
|
||||||
type = "template";
|
|
||||||
value_template = "{{ states[entity_id] }}";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
]
|
|
|
@ -1,51 +0,0 @@
|
||||||
{%- set customize_glob_exposed_attributes = {
|
|
||||||
".*": {
|
|
||||||
"friendly_name": true,
|
|
||||||
"temperature": true,
|
|
||||||
"current_temperature": true,
|
|
||||||
"temperature_unit": true,
|
|
||||||
"brightness": true,
|
|
||||||
"humidity": true,
|
|
||||||
"unit_of_measurement": true,
|
|
||||||
"device_class": true,
|
|
||||||
"current_position": true,
|
|
||||||
"percentage": true,
|
|
||||||
"volume_level": true,
|
|
||||||
"media_title": true,
|
|
||||||
"media_artist": true,
|
|
||||||
"media_album_name": true,
|
|
||||||
},
|
|
||||||
} %}
|
|
||||||
|
|
||||||
{%- macro get_exposed_attributes(entity_id) -%}
|
|
||||||
{%- set ns = namespace(exposed_attributes = {}, result = {}) %}
|
|
||||||
{%- for pattern, attributes in customize_glob_exposed_attributes.items() -%}
|
|
||||||
{%- if entity_id | regex_match(pattern) -%}
|
|
||||||
{%- set ns.exposed_attributes = dict(ns.exposed_attributes, **attributes) -%}
|
|
||||||
{%- endif -%}
|
|
||||||
{%- endfor -%}
|
|
||||||
{%- for attribute_key, should_include in ns.exposed_attributes.items() -%}
|
|
||||||
{%- if should_include and state_attr(entity_id, attribute_key) != None -%}
|
|
||||||
{%- set temp = {attribute_key: state_attr(entity_id, attribute_key)} if should_include is boolean else {attribute_key: should_include} -%}
|
|
||||||
{%- set ns.result = dict(ns.result, **temp) -%}
|
|
||||||
{%- endif -%}
|
|
||||||
{%- endfor -%}
|
|
||||||
{%- set result = ns.result | to_json if ns.result!={} else None -%}
|
|
||||||
{{"'" + result + "'" if result != None else ''}}
|
|
||||||
{%- endmacro -%}
|
|
||||||
|
|
||||||
I want you to act as a personal assistant who is aware of my smart home.
|
|
||||||
You will truthfully answer in one sentence in everyday language.
|
|
||||||
|
|
||||||
Current Time: {{now()}}
|
|
||||||
|
|
||||||
Available Devices:
|
|
||||||
```csv
|
|
||||||
entity_id,name,state,aliases,attributes
|
|
||||||
{% for entity in exposed_entities -%}
|
|
||||||
{{ entity.entity_id }},{{ entity.name }},{{ entity.state }},{{entity.aliases | join('/')}},{{get_exposed_attributes(entity.entity_id)}}
|
|
||||||
{% endfor -%}
|
|
||||||
```
|
|
||||||
|
|
||||||
The current state of devices is provided in available devices.
|
|
||||||
Do not restate what user says.
|
|
|
@ -1,387 +0,0 @@
|
||||||
{config, ...}: {
|
|
||||||
services.esphome = {
|
|
||||||
secretsFile = config.sops.secrets.esphome.path;
|
|
||||||
|
|
||||||
firmwareConfigs = {
|
|
||||||
# -------------------------------------------------------------
|
|
||||||
"AtomEcho" = {
|
|
||||||
# Device specific settings
|
|
||||||
api.encryption.key = "!secret api_key";
|
|
||||||
|
|
||||||
ota = [
|
|
||||||
{
|
|
||||||
platform = "esphome";
|
|
||||||
password = "!secret ota_pass";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
wifi = {
|
|
||||||
ssid = "!secret wifi_ssid";
|
|
||||||
password = "!secret wifi_password";
|
|
||||||
|
|
||||||
manual_ip = {
|
|
||||||
static_ip = "192.168.0.92";
|
|
||||||
gateway = "192.168.0.1";
|
|
||||||
subnet = "255.255.255.0";
|
|
||||||
};
|
|
||||||
|
|
||||||
ap = {
|
|
||||||
ssid = "Esp1 Fallback Hotspot";
|
|
||||||
password = "!secret ap_fallback";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# Hardware Declaration
|
|
||||||
esphome = {
|
|
||||||
friendly_name = "M5Stack Atom Echo";
|
|
||||||
min_version = "2024.9.0";
|
|
||||||
name = "m5stack-atom-echo";
|
|
||||||
name_add_mac_suffix = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
esp32 = {
|
|
||||||
board = "m5stack-atom";
|
|
||||||
framework = {
|
|
||||||
type = "esp-idf";
|
|
||||||
version = "4.4.8";
|
|
||||||
platform_version = "5.4.0";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
esp_adf = {};
|
|
||||||
|
|
||||||
button = [
|
|
||||||
{
|
|
||||||
id = "button_safe_mode";
|
|
||||||
name = "Safe Mode Boot";
|
|
||||||
platform = "safe_mode";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
id = "factory_reset_btn";
|
|
||||||
name = "Factory reset";
|
|
||||||
platform = "factory_reset";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
microphone = [
|
|
||||||
{
|
|
||||||
adc_type = "external";
|
|
||||||
i2s_din_pin = "GPIO23";
|
|
||||||
id = "echo_microphone";
|
|
||||||
pdm = true;
|
|
||||||
platform = "i2s_audio";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
speaker = [
|
|
||||||
{
|
|
||||||
dac_type = "external";
|
|
||||||
i2s_dout_pin = "GPIO21"; # "GPIO22"; turn off speaker
|
|
||||||
id = "echo_speaker";
|
|
||||||
channel = "mono";
|
|
||||||
platform = "i2s_audio";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
i2s_audio = [
|
|
||||||
{
|
|
||||||
id = "i2s_audio_bus";
|
|
||||||
i2s_bclk_pin = "GPIO19";
|
|
||||||
i2s_lrclk_pin = "GPIO33";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
light = [
|
|
||||||
{
|
|
||||||
id = "led";
|
|
||||||
name = "None";
|
|
||||||
entity_category = "config";
|
|
||||||
|
|
||||||
chipset = "SK6812";
|
|
||||||
pin = "GPIO27";
|
|
||||||
platform = "esp32_rmt_led_strip";
|
|
||||||
|
|
||||||
default_transition_length = "0s";
|
|
||||||
disabled_by_default = false;
|
|
||||||
num_leds = 1;
|
|
||||||
rgb_order = "grb";
|
|
||||||
rmt_channel = 0;
|
|
||||||
|
|
||||||
effects = [
|
|
||||||
{
|
|
||||||
pulse = {
|
|
||||||
name = "Slow Pulse";
|
|
||||||
|
|
||||||
max_brightness = "100%";
|
|
||||||
min_brightness = "50%";
|
|
||||||
transition_length = "250ms";
|
|
||||||
update_interval = "250ms";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
{
|
|
||||||
pulse = {
|
|
||||||
name = "Fast Pulse";
|
|
||||||
|
|
||||||
max_brightness = "100%";
|
|
||||||
min_brightness = "50%";
|
|
||||||
transition_length = "100ms";
|
|
||||||
update_interval = "100ms";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
# Home-assistant buttons
|
|
||||||
switch = [
|
|
||||||
{
|
|
||||||
id = "use_wake_word";
|
|
||||||
name = "Use wake word";
|
|
||||||
entity_category = "config";
|
|
||||||
|
|
||||||
optimistic = true;
|
|
||||||
platform = "template";
|
|
||||||
restore_mode = "RESTORE_DEFAULT_ON";
|
|
||||||
|
|
||||||
on_turn_on = [
|
|
||||||
{lambda = "id(va).set_use_wake_word(true);";}
|
|
||||||
{
|
|
||||||
"if" = {
|
|
||||||
condition.not = ["voice_assistant.is_running"];
|
|
||||||
"then" = ["voice_assistant.start_continuous"];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_turn_off = [
|
|
||||||
"voice_assistant.stop"
|
|
||||||
{lambda = "id(va).set_use_wake_word(false);";}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
id = "use_listen_light";
|
|
||||||
name = "Use listen light";
|
|
||||||
entity_category = "config";
|
|
||||||
|
|
||||||
optimistic = true;
|
|
||||||
platform = "template";
|
|
||||||
restore_mode = "RESTORE_DEFAULT_ON";
|
|
||||||
|
|
||||||
on_turn_on = [{"script.execute" = "reset_led";}];
|
|
||||||
on_turn_off = [{"script.execute" = "reset_led";}];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
binary_sensor = [
|
|
||||||
{
|
|
||||||
id = "echo_button";
|
|
||||||
name = "Button";
|
|
||||||
entity_category = "diagnostic";
|
|
||||||
|
|
||||||
disabled_by_default = false;
|
|
||||||
|
|
||||||
platform = "gpio";
|
|
||||||
pin = {
|
|
||||||
inverted = true;
|
|
||||||
number = "GPIO39";
|
|
||||||
};
|
|
||||||
|
|
||||||
on_multi_click = [
|
|
||||||
{
|
|
||||||
timing = ["ON for at least 50ms" "OFF for at least 50ms"];
|
|
||||||
"then" = [
|
|
||||||
{
|
|
||||||
"if" = {
|
|
||||||
condition = {"switch.is_off" = "use_wake_word";};
|
|
||||||
"then" = [
|
|
||||||
{
|
|
||||||
"if" = {
|
|
||||||
condition = "voice_assistant.is_running";
|
|
||||||
"then" = [
|
|
||||||
{"voice_assistant.stop" = {};}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
];
|
|
||||||
"else" = [{"voice_assistant.start" = {};}];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
"else" = [
|
|
||||||
"voice_assistant.stop"
|
|
||||||
{delay = "1s";}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
{"script.wait" = "reset_led";}
|
|
||||||
{"voice_assistant.start_continuous" = {};}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
timing = ["ON for at least 10s"];
|
|
||||||
"then" = [{"button.press" = "factory_reset_btn";}];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
# Misc
|
|
||||||
logger = {};
|
|
||||||
|
|
||||||
external_components = [
|
|
||||||
{
|
|
||||||
source = "github://pr#5230";
|
|
||||||
components = ["esp_adf"];
|
|
||||||
refresh = "0s";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
# Configs
|
|
||||||
script = [
|
|
||||||
{
|
|
||||||
id = "reset_led";
|
|
||||||
"then" = [
|
|
||||||
{
|
|
||||||
"if" = {
|
|
||||||
condition = [
|
|
||||||
{"switch.is_on" = "use_wake_word";}
|
|
||||||
{"switch.is_on" = "use_listen_light";}
|
|
||||||
];
|
|
||||||
"then" = [
|
|
||||||
{
|
|
||||||
"light.turn_on" = {
|
|
||||||
id = "led";
|
|
||||||
brightness = "60%";
|
|
||||||
effect = "none";
|
|
||||||
|
|
||||||
red = "100%";
|
|
||||||
green = "89%";
|
|
||||||
blue = "71%";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
"else" = [{"light.turn_off" = "led";}];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
voice_assistant = {
|
|
||||||
id = "va";
|
|
||||||
speaker = "echo_speaker";
|
|
||||||
microphone = "echo_microphone";
|
|
||||||
auto_gain = "31dBFS";
|
|
||||||
|
|
||||||
noise_suppression_level = 2;
|
|
||||||
vad_threshold = 3;
|
|
||||||
volume_multiplier = 2;
|
|
||||||
|
|
||||||
on_listening = [
|
|
||||||
{
|
|
||||||
"light.turn_on" = {
|
|
||||||
id = "led";
|
|
||||||
effect = "Slow Pulse";
|
|
||||||
|
|
||||||
green = "0%";
|
|
||||||
red = "0%";
|
|
||||||
blue = "100%";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_stt_vad_end = [
|
|
||||||
{
|
|
||||||
"light.turn_on" = {
|
|
||||||
id = "led";
|
|
||||||
effect = "Fast Pulse";
|
|
||||||
|
|
||||||
red = "0%";
|
|
||||||
green = "0%";
|
|
||||||
blue = "100%";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_tts_start = [
|
|
||||||
{
|
|
||||||
"light.turn_on" = {
|
|
||||||
id = "led";
|
|
||||||
brightness = "100%";
|
|
||||||
effect = "none";
|
|
||||||
|
|
||||||
red = "0%";
|
|
||||||
green = "0%";
|
|
||||||
blue = "100%";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
# Play audio from bluetooth speaker
|
|
||||||
on_tts_end = [
|
|
||||||
{
|
|
||||||
"homeassistant.service" = {
|
|
||||||
service = "media_player.play_media";
|
|
||||||
data = {
|
|
||||||
entity_id = "media_player.music_player_daemon";
|
|
||||||
media_content_id = "!lambda \"return x;\"";
|
|
||||||
media_content_type = "music";
|
|
||||||
announce = "\"true\"";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_end = [
|
|
||||||
{delay = "100ms";}
|
|
||||||
{wait_until.not."speaker.is_playing" = {};}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_error = [
|
|
||||||
{
|
|
||||||
"light.turn_on" = {
|
|
||||||
id = "led";
|
|
||||||
brightness = "100%";
|
|
||||||
effect = "none";
|
|
||||||
|
|
||||||
red = "100%";
|
|
||||||
green = "0%";
|
|
||||||
blue = "0%";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
{delay = "1s";}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_client_connected = [
|
|
||||||
{
|
|
||||||
"if" = {
|
|
||||||
condition = {"switch.is_on" = "use_wake_word";};
|
|
||||||
"then" = [
|
|
||||||
{"voice_assistant.start_continuous" = {};}
|
|
||||||
{"script.execute" = "reset_led";}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
on_client_disconnected = [
|
|
||||||
{
|
|
||||||
"if" = {
|
|
||||||
condition = {"switch.is_on" = "use_wake_word";};
|
|
||||||
"then" = [
|
|
||||||
{"voice_assistant.stop" = {};}
|
|
||||||
{"light.turn_off" = "led";}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
# -------------------------------------------------------------
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,364 +0,0 @@
|
||||||
{
|
|
||||||
caule-themes-src,
|
|
||||||
dracul-ha-src,
|
|
||||||
material-rounded-theme-src,
|
|
||||||
lib,
|
|
||||||
pkgs,
|
|
||||||
self,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (lib) attrValues singleton;
|
|
||||||
inherit (pkgs.writers) writeYAML;
|
|
||||||
in {
|
|
||||||
services.home-assistant = {
|
|
||||||
configFiles = {
|
|
||||||
"themes/caule.yaml".source = "${caule-themes-src}/themes/caule-themes-pack-1.yaml";
|
|
||||||
"themes/dracul-ha.yaml".source = "${dracul-ha-src}/themes/dracul-ha.yaml";
|
|
||||||
"themes/material_rounded.yaml".source = "${material-rounded-theme-src}/themes/material_rounded.yaml";
|
|
||||||
|
|
||||||
"www/sidebar-config.yaml".source = writeYAML "sidebar" {
|
|
||||||
id = "my-sidebar";
|
|
||||||
|
|
||||||
order = [
|
|
||||||
# Top
|
|
||||||
{
|
|
||||||
item = "overview";
|
|
||||||
order = 1;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
match = "href";
|
|
||||||
item = "calendar";
|
|
||||||
order = 2;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
match = "href";
|
|
||||||
item = "todo";
|
|
||||||
order = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
# Bottom
|
|
||||||
{
|
|
||||||
bottom = true;
|
|
||||||
item = "esphome";
|
|
||||||
order = 5;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
bottom = true;
|
|
||||||
item = "logbook";
|
|
||||||
order = 7;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
bottom = true;
|
|
||||||
icon = "mdi:tools";
|
|
||||||
item = "developer tools";
|
|
||||||
name = "Developer tools";
|
|
||||||
order = 9;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
bottom = true;
|
|
||||||
item = "settings";
|
|
||||||
order = 11;
|
|
||||||
}
|
|
||||||
|
|
||||||
# Hidden
|
|
||||||
{
|
|
||||||
hide = true;
|
|
||||||
item = "map";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
hide = true;
|
|
||||||
item = "energy";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
hide = true;
|
|
||||||
item = "history";
|
|
||||||
}
|
|
||||||
{
|
|
||||||
hide = true;
|
|
||||||
item = "media";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
customComponents = attrValues {
|
|
||||||
inherit
|
|
||||||
(self.scopedPackages.${pkgs.system}.hass-components)
|
|
||||||
material-symbols
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
customLovelaceModules = attrValues {
|
|
||||||
inherit
|
|
||||||
(pkgs.home-assistant-custom-lovelace-modules)
|
|
||||||
card-mod
|
|
||||||
light-entity-card
|
|
||||||
universal-remote-card
|
|
||||||
;
|
|
||||||
|
|
||||||
inherit
|
|
||||||
(self.scopedPackages.${pkgs.system}.lovelace-components)
|
|
||||||
custom-sidebar
|
|
||||||
material-rounded-theme
|
|
||||||
;
|
|
||||||
};
|
|
||||||
|
|
||||||
config.frontend = {
|
|
||||||
themes = "!include_dir_merge_named themes";
|
|
||||||
extra_module_url = map (p: "/local/nixos-lovelace-modules/${p}.js") [
|
|
||||||
"card-mod"
|
|
||||||
"light-entity-card"
|
|
||||||
"custom-sidebar-yaml"
|
|
||||||
"material-rounded-theme"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
config.template = [
|
|
||||||
{
|
|
||||||
sensor = singleton {
|
|
||||||
name = "Material Rounded Base Color Matt";
|
|
||||||
unique_id = "material_rounded_base_color_matt";
|
|
||||||
state = ''{{ states("sensor.pixel_8_accent_color") }}'';
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
lovelaceConfig = {
|
|
||||||
title = "Our House";
|
|
||||||
# I don't want multiple views
|
|
||||||
views = singleton {
|
|
||||||
path = "home";
|
|
||||||
title = "Home";
|
|
||||||
cards = [
|
|
||||||
{
|
|
||||||
type = "entities";
|
|
||||||
entities = [
|
|
||||||
"switch.smartplug1"
|
|
||||||
"switch.smartplug3"
|
|
||||||
];
|
|
||||||
}
|
|
||||||
{
|
|
||||||
type = "entities";
|
|
||||||
entities = [
|
|
||||||
"timer.assist_timer1"
|
|
||||||
"timer.assist_timer2"
|
|
||||||
"timer.assist_timer3"
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
type = "custom:light-entity-card";
|
|
||||||
entity = "light.tz3210_katchgxy_ts0505b_light";
|
|
||||||
|
|
||||||
shorten_cards = false;
|
|
||||||
consolidate_entities = false;
|
|
||||||
child_card = false;
|
|
||||||
hide_header = false;
|
|
||||||
show_header_icon = false;
|
|
||||||
header = "";
|
|
||||||
color_wheel = true;
|
|
||||||
persist_features = true;
|
|
||||||
brightness = true;
|
|
||||||
color_temp = true;
|
|
||||||
white_value = true;
|
|
||||||
color_picker = true;
|
|
||||||
speed = true;
|
|
||||||
intensity = false;
|
|
||||||
force_features = false;
|
|
||||||
show_slider_percent = true;
|
|
||||||
full_width_sliders = true;
|
|
||||||
brightness_icon = "weather-sunny";
|
|
||||||
white_icon = "file-word-box";
|
|
||||||
temperature_icon = "thermometer";
|
|
||||||
speed_icon = "speedometer";
|
|
||||||
intensity_icon = "transit-connection-horizontal";
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
type = "custom:android-tv-card";
|
|
||||||
|
|
||||||
visibility = singleton {
|
|
||||||
condition = "state";
|
|
||||||
entity = "remote.onn_4k_streaming_box";
|
|
||||||
state_not = ["unavailable" "unknown"];
|
|
||||||
};
|
|
||||||
|
|
||||||
media_player_id = "media_player.living_room_speaker";
|
|
||||||
keyboard_id = "remote.android_tv_192_168_0_106";
|
|
||||||
remote_id = "remote.onn_4k_streaming_box";
|
|
||||||
|
|
||||||
rows = [
|
|
||||||
"navigation_buttons"
|
|
||||||
[null "slider" null]
|
|
||||||
[null]
|
|
||||||
["jellyfin" "home" "back" "keyboard"]
|
|
||||||
[null]
|
|
||||||
];
|
|
||||||
|
|
||||||
custom_actions = [
|
|
||||||
{
|
|
||||||
name = "center";
|
|
||||||
type = "button";
|
|
||||||
icon = "mdi:checkbox-blank-circle";
|
|
||||||
|
|
||||||
styles = ''
|
|
||||||
:host {
|
|
||||||
--icon-color: rgb(94, 94, 94);
|
|
||||||
--size: 200px;
|
|
||||||
background: rgb(31, 31, 31);
|
|
||||||
border-radius: 200px;
|
|
||||||
margin: -70px;
|
|
||||||
padding: 70px;
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
|
|
||||||
tap_action = {
|
|
||||||
action = "key";
|
|
||||||
key = "DPAD_CENTER";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
name = "up";
|
|
||||||
type = "button";
|
|
||||||
icon = "mdi:chevron-up";
|
|
||||||
|
|
||||||
styles = ''
|
|
||||||
:host {
|
|
||||||
--icon-color: rgb(197, 199, 197);
|
|
||||||
z-index: 2;
|
|
||||||
top: 25px;
|
|
||||||
height: 90px;
|
|
||||||
width: 300px;
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
|
|
||||||
hold_action = {action = "repeat";};
|
|
||||||
tap_action = {
|
|
||||||
action = "key";
|
|
||||||
key = "DPAD_UP";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
name = "down";
|
|
||||||
type = "button";
|
|
||||||
icon = "mdi:chevron-down";
|
|
||||||
|
|
||||||
styles = ''
|
|
||||||
:host {
|
|
||||||
--icon-color: rgb(197, 199, 197);
|
|
||||||
z-index: 2;
|
|
||||||
bottom: 25px;
|
|
||||||
height: 90px;
|
|
||||||
width: 300px;
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
|
|
||||||
hold_action = {action = "repeat";};
|
|
||||||
tap_action = {
|
|
||||||
action = "key";
|
|
||||||
key = "DPAD_DOWN";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
name = "left";
|
|
||||||
type = "button";
|
|
||||||
icon = "mdi:chevron-left";
|
|
||||||
|
|
||||||
styles = ''
|
|
||||||
:host {
|
|
||||||
--icon-color: rgb(197, 199, 197);
|
|
||||||
z-index: 2;
|
|
||||||
left: 30px;
|
|
||||||
height: 170px;
|
|
||||||
width: 90px;
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
|
|
||||||
hold_action = {action = "repeat";};
|
|
||||||
tap_action = {
|
|
||||||
action = "key";
|
|
||||||
key = "DPAD_LEFT";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
name = "right";
|
|
||||||
type = "button";
|
|
||||||
icon = "mdi:chevron-right";
|
|
||||||
|
|
||||||
styles = ''
|
|
||||||
:host {
|
|
||||||
--icon-color: rgb(197, 199, 197);
|
|
||||||
z-index: 2;
|
|
||||||
right: 30px;
|
|
||||||
height: 170px;
|
|
||||||
width: 90px;
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
|
|
||||||
hold_action = {action = "repeat";};
|
|
||||||
tap_action = {
|
|
||||||
action = "key";
|
|
||||||
key = "DPAD_RIGHT";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
name = "slider";
|
|
||||||
type = "slider";
|
|
||||||
icon = "mdi:volume-high";
|
|
||||||
|
|
||||||
range = [0 1];
|
|
||||||
step = 0.01;
|
|
||||||
|
|
||||||
tap_action = {
|
|
||||||
action = "perform-action";
|
|
||||||
perform_action = "media_player.volume_set";
|
|
||||||
data = {
|
|
||||||
volume_level = "{{ value | float }}";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
value_attribute = "volume_level";
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
styles = ''
|
|
||||||
#row-1 {
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
#row-2 {
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
#row-3 {
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
}
|
|
||||||
];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
config.lovelace.dashboards = {
|
|
||||||
esphome-dash = {
|
|
||||||
title = "ESPHome";
|
|
||||||
icon = "mdi:car-esp";
|
|
||||||
mode = "yaml";
|
|
||||||
|
|
||||||
show_in_sidebar = true;
|
|
||||||
require_admin = true;
|
|
||||||
|
|
||||||
filename = writeYAML "esphome.yaml" {
|
|
||||||
strategy = {
|
|
||||||
type = "iframe";
|
|
||||||
url = "https://esphome.nelim.org";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
|
@ -1,360 +0,0 @@
|
||||||
root = true
|
|
||||||
|
|
||||||
# All files
|
|
||||||
[*]
|
|
||||||
indent_style = space
|
|
||||||
|
|
||||||
# C# files
|
|
||||||
[*.cs]
|
|
||||||
|
|
||||||
#### Core EditorConfig Options ####
|
|
||||||
|
|
||||||
# Indentation and spacing
|
|
||||||
indent_size = 4
|
|
||||||
tab_width = 4
|
|
||||||
|
|
||||||
# New line preferences
|
|
||||||
end_of_line = lf
|
|
||||||
insert_final_newline = true
|
|
||||||
|
|
||||||
#### .NET Coding Conventions ####
|
|
||||||
[*.{cs,vb}]
|
|
||||||
|
|
||||||
# Organize usings
|
|
||||||
dotnet_separate_import_directive_groups = true
|
|
||||||
dotnet_sort_system_directives_first = true
|
|
||||||
file_header_template = unset
|
|
||||||
|
|
||||||
# this. and Me. preferences
|
|
||||||
dotnet_style_qualification_for_event = false:silent
|
|
||||||
dotnet_style_qualification_for_field = false:silent
|
|
||||||
dotnet_style_qualification_for_method = false:silent
|
|
||||||
dotnet_style_qualification_for_property = false:silent
|
|
||||||
|
|
||||||
# Language keywords vs BCL types preferences
|
|
||||||
dotnet_style_predefined_type_for_locals_parameters_members = true:silent
|
|
||||||
dotnet_style_predefined_type_for_member_access = true:silent
|
|
||||||
|
|
||||||
# Parentheses preferences
|
|
||||||
dotnet_style_parentheses_in_arithmetic_binary_operators = always_for_clarity:silent
|
|
||||||
dotnet_style_parentheses_in_other_binary_operators = always_for_clarity:silent
|
|
||||||
dotnet_style_parentheses_in_other_operators = never_if_unnecessary:silent
|
|
||||||
dotnet_style_parentheses_in_relational_binary_operators = always_for_clarity:silent
|
|
||||||
|
|
||||||
# Modifier preferences
|
|
||||||
dotnet_style_require_accessibility_modifiers = for_non_interface_members:silent
|
|
||||||
|
|
||||||
# Expression-level preferences
|
|
||||||
dotnet_style_coalesce_expression = true:suggestion
|
|
||||||
dotnet_style_collection_initializer = true:suggestion
|
|
||||||
dotnet_style_explicit_tuple_names = true:suggestion
|
|
||||||
dotnet_style_null_propagation = true:suggestion
|
|
||||||
dotnet_style_object_initializer = true:suggestion
|
|
||||||
dotnet_style_operator_placement_when_wrapping = beginning_of_line
|
|
||||||
dotnet_style_prefer_auto_properties = true:suggestion
|
|
||||||
dotnet_style_prefer_compound_assignment = true:suggestion
|
|
||||||
dotnet_style_prefer_conditional_expression_over_assignment = true:suggestion
|
|
||||||
dotnet_style_prefer_conditional_expression_over_return = true:suggestion
|
|
||||||
dotnet_style_prefer_inferred_anonymous_type_member_names = true:suggestion
|
|
||||||
dotnet_style_prefer_inferred_tuple_names = true:suggestion
|
|
||||||
dotnet_style_prefer_is_null_check_over_reference_equality_method = true:suggestion
|
|
||||||
dotnet_style_prefer_simplified_boolean_expressions = true:suggestion
|
|
||||||
dotnet_style_prefer_simplified_interpolation = true:suggestion
|
|
||||||
|
|
||||||
# Field preferences
|
|
||||||
dotnet_style_readonly_field = true:warning
|
|
||||||
|
|
||||||
# Parameter preferences
|
|
||||||
dotnet_code_quality_unused_parameters = all:suggestion
|
|
||||||
|
|
||||||
# Suppression preferences
|
|
||||||
dotnet_remove_unnecessary_suppression_exclusions = none
|
|
||||||
|
|
||||||
#### C# Coding Conventions ####
|
|
||||||
[*.cs]
|
|
||||||
|
|
||||||
# var preferences
|
|
||||||
csharp_style_var_elsewhere = false:silent
|
|
||||||
csharp_style_var_for_built_in_types = false:silent
|
|
||||||
csharp_style_var_when_type_is_apparent = false:silent
|
|
||||||
|
|
||||||
# Expression-bodied members
|
|
||||||
csharp_style_expression_bodied_accessors = true:silent
|
|
||||||
csharp_style_expression_bodied_constructors = false:silent
|
|
||||||
csharp_style_expression_bodied_indexers = true:silent
|
|
||||||
csharp_style_expression_bodied_lambdas = true:suggestion
|
|
||||||
csharp_style_expression_bodied_local_functions = false:silent
|
|
||||||
csharp_style_expression_bodied_methods = false:silent
|
|
||||||
csharp_style_expression_bodied_operators = false:silent
|
|
||||||
csharp_style_expression_bodied_properties = true:silent
|
|
||||||
|
|
||||||
# Pattern matching preferences
|
|
||||||
csharp_style_pattern_matching_over_as_with_null_check = true:suggestion
|
|
||||||
csharp_style_pattern_matching_over_is_with_cast_check = true:suggestion
|
|
||||||
csharp_style_prefer_not_pattern = true:suggestion
|
|
||||||
csharp_style_prefer_pattern_matching = true:silent
|
|
||||||
csharp_style_prefer_switch_expression = true:suggestion
|
|
||||||
|
|
||||||
# Null-checking preferences
|
|
||||||
csharp_style_conditional_delegate_call = true:suggestion
|
|
||||||
|
|
||||||
# Modifier preferences
|
|
||||||
csharp_prefer_static_local_function = true:warning
|
|
||||||
csharp_preferred_modifier_order = public,private,protected,internal,static,extern,new,virtual,abstract,sealed,override,readonly,unsafe,volatile,async:silent
|
|
||||||
|
|
||||||
# Code-block preferences
|
|
||||||
csharp_prefer_braces = true:silent
|
|
||||||
csharp_prefer_simple_using_statement = true:suggestion
|
|
||||||
|
|
||||||
# Expression-level preferences
|
|
||||||
csharp_prefer_simple_default_expression = true:suggestion
|
|
||||||
csharp_style_deconstructed_variable_declaration = true:suggestion
|
|
||||||
csharp_style_inlined_variable_declaration = true:suggestion
|
|
||||||
csharp_style_pattern_local_over_anonymous_function = true:suggestion
|
|
||||||
csharp_style_prefer_index_operator = true:suggestion
|
|
||||||
csharp_style_prefer_range_operator = true:suggestion
|
|
||||||
csharp_style_throw_expression = true:suggestion
|
|
||||||
csharp_style_unused_value_assignment_preference = discard_variable:suggestion
|
|
||||||
csharp_style_unused_value_expression_statement_preference = discard_variable:silent
|
|
||||||
|
|
||||||
# 'using' directive preferences
|
|
||||||
csharp_using_directive_placement = outside_namespace:silent
|
|
||||||
|
|
||||||
#### C# Formatting Rules ####
|
|
||||||
|
|
||||||
# New line preferences
|
|
||||||
csharp_new_line_before_catch = true
|
|
||||||
csharp_new_line_before_else = true
|
|
||||||
csharp_new_line_before_finally = true
|
|
||||||
csharp_new_line_before_members_in_anonymous_types = true
|
|
||||||
csharp_new_line_before_members_in_object_initializers = true
|
|
||||||
csharp_new_line_before_open_brace = all
|
|
||||||
csharp_new_line_between_query_expression_clauses = true
|
|
||||||
|
|
||||||
# Indentation preferences
|
|
||||||
csharp_indent_block_contents = true
|
|
||||||
csharp_indent_braces = false
|
|
||||||
csharp_indent_case_contents = true
|
|
||||||
csharp_indent_case_contents_when_block = true
|
|
||||||
csharp_indent_labels = one_less_than_current
|
|
||||||
csharp_indent_switch_labels = true
|
|
||||||
|
|
||||||
# Space preferences
|
|
||||||
csharp_space_after_cast = false
|
|
||||||
csharp_space_after_colon_in_inheritance_clause = true
|
|
||||||
csharp_space_after_comma = true
|
|
||||||
csharp_space_after_dot = false
|
|
||||||
csharp_space_after_keywords_in_control_flow_statements = true
|
|
||||||
csharp_space_after_semicolon_in_for_statement = true
|
|
||||||
csharp_space_around_binary_operators = before_and_after
|
|
||||||
csharp_space_around_declaration_statements = false
|
|
||||||
csharp_space_before_colon_in_inheritance_clause = true
|
|
||||||
csharp_space_before_comma = false
|
|
||||||
csharp_space_before_dot = false
|
|
||||||
csharp_space_before_open_square_brackets = false
|
|
||||||
csharp_space_before_semicolon_in_for_statement = false
|
|
||||||
csharp_space_between_empty_square_brackets = false
|
|
||||||
csharp_space_between_method_call_empty_parameter_list_parentheses = false
|
|
||||||
csharp_space_between_method_call_name_and_opening_parenthesis = false
|
|
||||||
csharp_space_between_method_call_parameter_list_parentheses = false
|
|
||||||
csharp_space_between_method_declaration_empty_parameter_list_parentheses = false
|
|
||||||
csharp_space_between_method_declaration_name_and_open_parenthesis = false
|
|
||||||
csharp_space_between_method_declaration_parameter_list_parentheses = false
|
|
||||||
csharp_space_between_parentheses = false
|
|
||||||
csharp_space_between_square_brackets = false
|
|
||||||
|
|
||||||
# Wrapping preferences
|
|
||||||
csharp_preserve_single_line_blocks = true
|
|
||||||
csharp_preserve_single_line_statements = true
|
|
||||||
|
|
||||||
#### Naming styles ####
|
|
||||||
[*.{cs,vb}]
|
|
||||||
|
|
||||||
# Naming rules
|
|
||||||
|
|
||||||
dotnet_naming_rule.types_and_namespaces_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.types_and_namespaces_should_be_pascalcase.symbols = types_and_namespaces
|
|
||||||
dotnet_naming_rule.types_and_namespaces_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.interfaces_should_be_ipascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.interfaces_should_be_ipascalcase.symbols = interfaces
|
|
||||||
dotnet_naming_rule.interfaces_should_be_ipascalcase.style = ipascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.type_parameters_should_be_tpascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.type_parameters_should_be_tpascalcase.symbols = type_parameters
|
|
||||||
dotnet_naming_rule.type_parameters_should_be_tpascalcase.style = tpascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.methods_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.methods_should_be_pascalcase.symbols = methods
|
|
||||||
dotnet_naming_rule.methods_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.properties_should_be_pascalcase.severity = none
|
|
||||||
dotnet_naming_rule.properties_should_be_pascalcase.symbols = properties
|
|
||||||
dotnet_naming_rule.properties_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.events_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.events_should_be_pascalcase.symbols = events
|
|
||||||
dotnet_naming_rule.events_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.local_variables_should_be_camelcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.local_variables_should_be_camelcase.symbols = local_variables
|
|
||||||
dotnet_naming_rule.local_variables_should_be_camelcase.style = camelcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.local_constants_should_be_camelcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.local_constants_should_be_camelcase.symbols = local_constants
|
|
||||||
dotnet_naming_rule.local_constants_should_be_camelcase.style = camelcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.parameters_should_be_camelcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.parameters_should_be_camelcase.symbols = parameters
|
|
||||||
dotnet_naming_rule.parameters_should_be_camelcase.style = camelcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.public_fields_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.public_fields_should_be_pascalcase.symbols = public_fields
|
|
||||||
dotnet_naming_rule.public_fields_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.private_fields_should_be__camelcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.private_fields_should_be__camelcase.symbols = private_fields
|
|
||||||
dotnet_naming_rule.private_fields_should_be__camelcase.style = _camelcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.private_static_fields_should_be_s_camelcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.private_static_fields_should_be_s_camelcase.symbols = private_static_fields
|
|
||||||
dotnet_naming_rule.private_static_fields_should_be_s_camelcase.style = s_camelcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.public_constant_fields_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.public_constant_fields_should_be_pascalcase.symbols = public_constant_fields
|
|
||||||
dotnet_naming_rule.public_constant_fields_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.private_constant_fields_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.private_constant_fields_should_be_pascalcase.symbols = private_constant_fields
|
|
||||||
dotnet_naming_rule.private_constant_fields_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.public_static_readonly_fields_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.public_static_readonly_fields_should_be_pascalcase.symbols = public_static_readonly_fields
|
|
||||||
dotnet_naming_rule.public_static_readonly_fields_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.private_static_readonly_fields_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.private_static_readonly_fields_should_be_pascalcase.symbols = private_static_readonly_fields
|
|
||||||
dotnet_naming_rule.private_static_readonly_fields_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.enums_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.enums_should_be_pascalcase.symbols = enums
|
|
||||||
dotnet_naming_rule.enums_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.local_functions_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.local_functions_should_be_pascalcase.symbols = local_functions
|
|
||||||
dotnet_naming_rule.local_functions_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
dotnet_naming_rule.non_field_members_should_be_pascalcase.severity = suggestion
|
|
||||||
dotnet_naming_rule.non_field_members_should_be_pascalcase.symbols = non_field_members
|
|
||||||
dotnet_naming_rule.non_field_members_should_be_pascalcase.style = pascalcase
|
|
||||||
|
|
||||||
# Symbol specifications
|
|
||||||
|
|
||||||
dotnet_naming_symbols.interfaces.applicable_kinds = interface
|
|
||||||
dotnet_naming_symbols.interfaces.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.interfaces.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.enums.applicable_kinds = enum
|
|
||||||
dotnet_naming_symbols.enums.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.enums.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.events.applicable_kinds = event
|
|
||||||
dotnet_naming_symbols.events.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.events.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.methods.applicable_kinds = method
|
|
||||||
dotnet_naming_symbols.methods.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.methods.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.properties.applicable_kinds = property
|
|
||||||
dotnet_naming_symbols.properties.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.properties.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.public_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.public_fields.applicable_accessibilities = public, internal
|
|
||||||
dotnet_naming_symbols.public_fields.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.private_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.private_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.private_fields.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.private_static_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.private_static_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.private_static_fields.required_modifiers = static
|
|
||||||
|
|
||||||
dotnet_naming_symbols.types_and_namespaces.applicable_kinds = namespace, class, struct, interface, enum
|
|
||||||
dotnet_naming_symbols.types_and_namespaces.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.types_and_namespaces.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.non_field_members.applicable_kinds = property, event, method
|
|
||||||
dotnet_naming_symbols.non_field_members.applicable_accessibilities = public, internal, private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.non_field_members.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.type_parameters.applicable_kinds = namespace
|
|
||||||
dotnet_naming_symbols.type_parameters.applicable_accessibilities = *
|
|
||||||
dotnet_naming_symbols.type_parameters.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.private_constant_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.private_constant_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.private_constant_fields.required_modifiers = const
|
|
||||||
|
|
||||||
dotnet_naming_symbols.local_variables.applicable_kinds = local
|
|
||||||
dotnet_naming_symbols.local_variables.applicable_accessibilities = local
|
|
||||||
dotnet_naming_symbols.local_variables.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.local_constants.applicable_kinds = local
|
|
||||||
dotnet_naming_symbols.local_constants.applicable_accessibilities = local
|
|
||||||
dotnet_naming_symbols.local_constants.required_modifiers = const
|
|
||||||
|
|
||||||
dotnet_naming_symbols.parameters.applicable_kinds = parameter
|
|
||||||
dotnet_naming_symbols.parameters.applicable_accessibilities = *
|
|
||||||
dotnet_naming_symbols.parameters.required_modifiers =
|
|
||||||
|
|
||||||
dotnet_naming_symbols.public_constant_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.public_constant_fields.applicable_accessibilities = public, internal
|
|
||||||
dotnet_naming_symbols.public_constant_fields.required_modifiers = const
|
|
||||||
|
|
||||||
dotnet_naming_symbols.public_static_readonly_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.public_static_readonly_fields.applicable_accessibilities = public, internal
|
|
||||||
dotnet_naming_symbols.public_static_readonly_fields.required_modifiers = readonly, static
|
|
||||||
|
|
||||||
dotnet_naming_symbols.private_static_readonly_fields.applicable_kinds = field
|
|
||||||
dotnet_naming_symbols.private_static_readonly_fields.applicable_accessibilities = private, protected, protected_internal, private_protected
|
|
||||||
dotnet_naming_symbols.private_static_readonly_fields.required_modifiers = readonly, static
|
|
||||||
|
|
||||||
dotnet_naming_symbols.local_functions.applicable_kinds = local_function
|
|
||||||
dotnet_naming_symbols.local_functions.applicable_accessibilities = *
|
|
||||||
dotnet_naming_symbols.local_functions.required_modifiers =
|
|
||||||
|
|
||||||
# Naming styles
|
|
||||||
|
|
||||||
dotnet_naming_style.pascalcase.required_prefix =
|
|
||||||
dotnet_naming_style.pascalcase.required_suffix =
|
|
||||||
dotnet_naming_style.pascalcase.word_separator =
|
|
||||||
dotnet_naming_style.pascalcase.capitalization = pascal_case
|
|
||||||
|
|
||||||
dotnet_naming_style.ipascalcase.required_prefix = I
|
|
||||||
dotnet_naming_style.ipascalcase.required_suffix =
|
|
||||||
dotnet_naming_style.ipascalcase.word_separator =
|
|
||||||
dotnet_naming_style.ipascalcase.capitalization = pascal_case
|
|
||||||
|
|
||||||
dotnet_naming_style.tpascalcase.required_prefix = T
|
|
||||||
dotnet_naming_style.tpascalcase.required_suffix =
|
|
||||||
dotnet_naming_style.tpascalcase.word_separator =
|
|
||||||
dotnet_naming_style.tpascalcase.capitalization = pascal_case
|
|
||||||
|
|
||||||
dotnet_naming_style._camelcase.required_prefix = _
|
|
||||||
dotnet_naming_style._camelcase.required_suffix =
|
|
||||||
dotnet_naming_style._camelcase.word_separator =
|
|
||||||
dotnet_naming_style._camelcase.capitalization = camel_case
|
|
||||||
|
|
||||||
dotnet_naming_style.camelcase.required_prefix =
|
|
||||||
dotnet_naming_style.camelcase.required_suffix =
|
|
||||||
dotnet_naming_style.camelcase.word_separator =
|
|
||||||
dotnet_naming_style.camelcase.capitalization = camel_case
|
|
||||||
|
|
||||||
dotnet_naming_style.s_camelcase.required_prefix = s_
|
|
||||||
dotnet_naming_style.s_camelcase.required_suffix =
|
|
||||||
dotnet_naming_style.s_camelcase.word_separator =
|
|
||||||
dotnet_naming_style.s_camelcase.capitalization = camel_case
|
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
use flake "$FLAKE#netdaemon"
|
|
||||||
cp -rf ./HomeAssistantGenerated ./HomeAssistantGenerated.cs
|
|
|
@ -1,4 +0,0 @@
|
||||||
obj
|
|
||||||
bin
|
|
||||||
NetDaemonCodegen
|
|
||||||
HomeAssistantGenerated.cs
|
|
|
@ -1 +0,0 @@
|
||||||
24.52.0
|
|
Binary file not shown.
|
@ -1,50 +0,0 @@
|
||||||
using System;
|
|
||||||
|
|
||||||
using HomeAssistantGenerated;
|
|
||||||
|
|
||||||
using NetDaemon.AppModel;
|
|
||||||
using NetDaemon.HassModel;
|
|
||||||
using NetDaemon.HassModel.Integration;
|
|
||||||
|
|
||||||
using NetDaemonConfig.Apps.Spotify.Types;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.PauseUnpause
|
|
||||||
{
|
|
||||||
public record PauseUnpauseData(bool pause);
|
|
||||||
|
|
||||||
[NetDaemonApp]
|
|
||||||
public class PauseUnpause
|
|
||||||
{
|
|
||||||
public PauseUnpause(IHaContext ha, Services services)
|
|
||||||
{
|
|
||||||
ha.RegisterServiceCallBack<PauseUnpauseData>(
|
|
||||||
"spotify_pause_unpause",
|
|
||||||
(e) =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (e.pause)
|
|
||||||
{
|
|
||||||
services.Spotifyplus.PlayerMediaPause(
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
deviceId: Globals.DefaultDevId);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
services.Spotifyplus.PlayerMediaResume(
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
deviceId: Globals.DefaultDevId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Exception error)
|
|
||||||
{
|
|
||||||
services.Notify.PersistentNotification(
|
|
||||||
message: error.Message,
|
|
||||||
title: "Erreur Spotify");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,120 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Globalization;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text.Json;
|
|
||||||
|
|
||||||
using FuzzySharp;
|
|
||||||
using FuzzySharp.Extractor;
|
|
||||||
|
|
||||||
using HomeAssistantGenerated;
|
|
||||||
|
|
||||||
using NetDaemon.AppModel;
|
|
||||||
using NetDaemon.HassModel;
|
|
||||||
using NetDaemon.HassModel.Integration;
|
|
||||||
|
|
||||||
using NetDaemonConfig.Apps.Spotify.Types;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.PlayAlbum
|
|
||||||
{
|
|
||||||
public record PlayAlbumData(string? artist, string? album);
|
|
||||||
|
|
||||||
[NetDaemonApp]
|
|
||||||
public class PlayAlbum
|
|
||||||
{
|
|
||||||
private readonly CultureInfo _cultureInfo = new("fr-CA", false);
|
|
||||||
|
|
||||||
// Snake-case json options
|
|
||||||
private readonly JsonSerializerOptions _jsonOptions = new()
|
|
||||||
{
|
|
||||||
PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower
|
|
||||||
};
|
|
||||||
|
|
||||||
public PlayAlbum(IHaContext ha, Services services)
|
|
||||||
{
|
|
||||||
ha.RegisterServiceCallBack<PlayAlbumData>(
|
|
||||||
"spotify_play_album",
|
|
||||||
async (e) =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
string uri;
|
|
||||||
|
|
||||||
if (e.artist is not null)
|
|
||||||
{
|
|
||||||
SpotifyplusSearchArtistsResponse? artistResult = (
|
|
||||||
await services.Spotifyplus.SearchArtistsAsync(
|
|
||||||
criteria: e?.artist ??
|
|
||||||
throw new TargetException($"The artist {e?.artist} could not be found."),
|
|
||||||
limitTotal: 1,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
// My Defaults
|
|
||||||
market: "CA",
|
|
||||||
includeExternal: "audio"
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyplusSearchArtistsResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
string artistId = artistResult?.Result?.Items?[0]?.Id ??
|
|
||||||
throw new TargetException($"The artist {e?.artist} could not be found.");
|
|
||||||
|
|
||||||
SpotifyPlusGetArtistAlbumsResponse? result = (
|
|
||||||
await services.Spotifyplus.GetArtistAlbumsAsync(
|
|
||||||
artistId: artistId,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
market: "CA"
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyPlusGetArtistAlbumsResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
List<ArtistAlbumItem> albums = result?.Result?.Items ??
|
|
||||||
throw new TargetException($"No albums found for artist {e.artist}");
|
|
||||||
|
|
||||||
ExtractedResult<ArtistAlbumItem> match = Process.ExtractOne(
|
|
||||||
new ArtistAlbumItem { Name = e.album?.ToLower(_cultureInfo) },
|
|
||||||
albums,
|
|
||||||
new Func<ArtistAlbumItem, string>((item) =>
|
|
||||||
(item.Name ?? "").ToLower(_cultureInfo))
|
|
||||||
);
|
|
||||||
|
|
||||||
uri = match.Value?.Uri ??
|
|
||||||
throw new TargetException($"No matches found for album {e.album}");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
SpotifyplusSearchAlbumsResponse? result = (
|
|
||||||
await services.Spotifyplus.SearchAlbumsAsync(
|
|
||||||
criteria: $"{e?.album}",
|
|
||||||
limitTotal: 1,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
// My Defaults
|
|
||||||
market: "CA",
|
|
||||||
includeExternal: "audio"
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyplusSearchAlbumsResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
uri = result?.Result?.Items?[0]?.Uri ??
|
|
||||||
throw new TargetException(
|
|
||||||
$"The album {e?.album}{(e?.artist is null ? "" : $" by {e?.artist}")} could not be found."
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
services.Spotifyplus.PlayerMediaPlayContext(
|
|
||||||
contextUri: uri,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
deviceId: Globals.DefaultDevId,
|
|
||||||
// My Defaults
|
|
||||||
positionMs: 0,
|
|
||||||
delay: 0.50
|
|
||||||
);
|
|
||||||
}
|
|
||||||
catch (Exception error)
|
|
||||||
{
|
|
||||||
services.Notify.PersistentNotification(
|
|
||||||
message: error.Message,
|
|
||||||
title: "Erreur Spotify");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,68 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text.Json;
|
|
||||||
|
|
||||||
using HomeAssistantGenerated;
|
|
||||||
|
|
||||||
using NetDaemon.AppModel;
|
|
||||||
using NetDaemon.HassModel;
|
|
||||||
using NetDaemon.HassModel.Integration;
|
|
||||||
|
|
||||||
using NetDaemonConfig.Apps.Spotify.Types;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.PlayArtist
|
|
||||||
{
|
|
||||||
public record PlayArtistData(string? artist);
|
|
||||||
|
|
||||||
[NetDaemonApp]
|
|
||||||
public class PlayArtist
|
|
||||||
{
|
|
||||||
// Snake-case json options
|
|
||||||
private readonly JsonSerializerOptions _jsonOptions = new()
|
|
||||||
{
|
|
||||||
PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower
|
|
||||||
};
|
|
||||||
|
|
||||||
public PlayArtist(IHaContext ha, Services services)
|
|
||||||
{
|
|
||||||
ha.RegisterServiceCallBack<PlayArtistData>(
|
|
||||||
"spotify_play_artist",
|
|
||||||
async (e) =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
SpotifyplusSearchArtistsResponse? result = (
|
|
||||||
await services.Spotifyplus.SearchArtistsAsync(
|
|
||||||
criteria: e?.artist ?? throw new TargetException($"The artist {e?.artist} could not be found."),
|
|
||||||
limitTotal: 1,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
// My Defaults
|
|
||||||
market: "CA",
|
|
||||||
includeExternal: "audio"
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyplusSearchArtistsResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
string uri = result?.Result?.Items?[0]?.Uri ??
|
|
||||||
throw new TargetException($"The artist {e?.artist} could not be found.");
|
|
||||||
|
|
||||||
services.Spotifyplus.PlayerMediaPlayContext(
|
|
||||||
contextUri: uri,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
deviceId: Globals.DefaultDevId,
|
|
||||||
// My Defaults
|
|
||||||
positionMs: 0,
|
|
||||||
delay: 0.50
|
|
||||||
);
|
|
||||||
}
|
|
||||||
catch (Exception error)
|
|
||||||
{
|
|
||||||
services.Notify.PersistentNotification(
|
|
||||||
message: error.Message,
|
|
||||||
title: "Erreur Spotify");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,102 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Globalization;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text.Json;
|
|
||||||
|
|
||||||
using FuzzySharp;
|
|
||||||
using FuzzySharp.Extractor;
|
|
||||||
|
|
||||||
using HomeAssistantGenerated;
|
|
||||||
|
|
||||||
using NetDaemon.AppModel;
|
|
||||||
using NetDaemon.HassModel;
|
|
||||||
using NetDaemon.HassModel.Integration;
|
|
||||||
|
|
||||||
using NetDaemonConfig.Apps.Spotify.Types;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.PlayPlaylist
|
|
||||||
{
|
|
||||||
public record PlayPlaylistData(string? playlist);
|
|
||||||
|
|
||||||
[NetDaemonApp]
|
|
||||||
public class PlayPlaylist
|
|
||||||
{
|
|
||||||
private readonly CultureInfo _cultureInfo = new("fr-CA", false);
|
|
||||||
|
|
||||||
// Snake-case json options
|
|
||||||
private readonly JsonSerializerOptions _jsonOptions = new()
|
|
||||||
{
|
|
||||||
PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower
|
|
||||||
};
|
|
||||||
|
|
||||||
public PlayPlaylist(IHaContext ha, Services services)
|
|
||||||
{
|
|
||||||
ha.RegisterServiceCallBack<PlayPlaylistData>(
|
|
||||||
"spotify_play_playlist",
|
|
||||||
async (e) =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
string query = e?.playlist ?? throw new TargetException("Query not found.");
|
|
||||||
|
|
||||||
SpotifyplusPlaylistResponse? result = (
|
|
||||||
await services.Spotifyplus.GetPlaylistFavoritesAsync(
|
|
||||||
limitTotal: 200,
|
|
||||||
sortResult: true,
|
|
||||||
entityId: Globals.DefaultEntityId
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyplusPlaylistResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
List<PlaylistsItem> myPlaylists = result?.Result?.Items ??
|
|
||||||
throw new TargetException($"No playlists found for query {query}");
|
|
||||||
|
|
||||||
ExtractedResult<PlaylistsItem> match = Process.ExtractOne(
|
|
||||||
new PlaylistsItem { Name = query.ToLower(_cultureInfo) },
|
|
||||||
myPlaylists,
|
|
||||||
new Func<PlaylistsItem, string>((item) => (item.Name ?? "").ToLower(_cultureInfo))
|
|
||||||
);
|
|
||||||
|
|
||||||
string uri = match.Value?.Uri ?? throw new TargetException($"No matches found for query {query}");
|
|
||||||
|
|
||||||
// We search outside the user's playlists if the score is too low
|
|
||||||
if (match.Score < 85)
|
|
||||||
{
|
|
||||||
SpotifyplusPlaylistResponse? otherResult = (
|
|
||||||
await services.Spotifyplus.SearchPlaylistsAsync(
|
|
||||||
criteria: query,
|
|
||||||
limitTotal: 1,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
// My Defaults
|
|
||||||
market: "CA",
|
|
||||||
includeExternal: "audio"
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyplusPlaylistResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
string potentialUri = otherResult?.Result?.Items?[0]?.Uri ??
|
|
||||||
throw new TargetException($"No public matches found for query {query}");
|
|
||||||
|
|
||||||
uri = potentialUri;
|
|
||||||
}
|
|
||||||
|
|
||||||
services.Spotifyplus.PlayerMediaPlayContext(
|
|
||||||
contextUri: uri,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
deviceId: Globals.DefaultDevId,
|
|
||||||
// My Defaults
|
|
||||||
positionMs: 0,
|
|
||||||
delay: 0.50
|
|
||||||
);
|
|
||||||
}
|
|
||||||
catch (Exception error)
|
|
||||||
{
|
|
||||||
services.Notify.PersistentNotification(
|
|
||||||
message: error.Message,
|
|
||||||
title: "Erreur Spotify");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,69 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Text.Json;
|
|
||||||
|
|
||||||
using HomeAssistantGenerated;
|
|
||||||
|
|
||||||
using NetDaemon.AppModel;
|
|
||||||
using NetDaemon.HassModel;
|
|
||||||
using NetDaemon.HassModel.Integration;
|
|
||||||
|
|
||||||
using NetDaemonConfig.Apps.Spotify.Types;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.PlaySong
|
|
||||||
{
|
|
||||||
public record PlaySongData(string? artist, string? song);
|
|
||||||
|
|
||||||
[NetDaemonApp]
|
|
||||||
public class PlaySong
|
|
||||||
{
|
|
||||||
// Snake-case json options
|
|
||||||
private readonly JsonSerializerOptions _jsonOptions = new()
|
|
||||||
{
|
|
||||||
PropertyNamingPolicy = JsonNamingPolicy.SnakeCaseLower
|
|
||||||
};
|
|
||||||
|
|
||||||
public PlaySong(IHaContext ha, Services services)
|
|
||||||
{
|
|
||||||
ha.RegisterServiceCallBack<PlaySongData>(
|
|
||||||
"spotify_play_song",
|
|
||||||
async (e) =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
SpotifyplusSearchTracksResponse? result = (
|
|
||||||
await services.Spotifyplus.SearchTracksAsync(
|
|
||||||
criteria: $"{e?.artist} {e?.song}",
|
|
||||||
limitTotal: 1,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
// My Defaults
|
|
||||||
market: "CA",
|
|
||||||
includeExternal: "audio"
|
|
||||||
)
|
|
||||||
).Value.Deserialize<SpotifyplusSearchTracksResponse>(_jsonOptions);
|
|
||||||
|
|
||||||
string uri = result?.Result?.Items?[0]?.Uri ?? throw new TargetException(
|
|
||||||
$"The song {e?.song}{(e?.artist is null ? "" : $" by {e?.artist}")} could not be found."
|
|
||||||
);
|
|
||||||
|
|
||||||
services.Spotifyplus.PlayerMediaPlayTracks(
|
|
||||||
uris: uri,
|
|
||||||
entityId: Globals.DefaultEntityId,
|
|
||||||
deviceId: Globals.DefaultDevId,
|
|
||||||
// My Defaults
|
|
||||||
positionMs: 0,
|
|
||||||
delay: 0.50
|
|
||||||
);
|
|
||||||
}
|
|
||||||
catch (Exception error)
|
|
||||||
{
|
|
||||||
services.Notify.PersistentNotification(
|
|
||||||
message: error.Message,
|
|
||||||
title: "Erreur Spotify");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,42 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.Types
|
|
||||||
{
|
|
||||||
public record ArtistAlbumItem
|
|
||||||
{
|
|
||||||
public string? AlbumType { get; set; }
|
|
||||||
public List<Artist>? Artists { get; set; }
|
|
||||||
public List<object>? AvailableMarkets { get; set; }
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? ImageUrl { get; set; }
|
|
||||||
public List<Image>? Images { get; set; }
|
|
||||||
public string? Name { get; set; }
|
|
||||||
public string? ReleaseDate { get; set; }
|
|
||||||
public string? ReleaseDatePrecision { get; set; }
|
|
||||||
public Restrictions? Restrictions { get; set; }
|
|
||||||
public int? TotalTracks { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record ArtistAlbumResult
|
|
||||||
{
|
|
||||||
public double? DateLastRefreshed { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public int? Limit { get; set; }
|
|
||||||
public object? Next { get; set; }
|
|
||||||
public int? Offset { get; set; }
|
|
||||||
public object? Previous { get; set; }
|
|
||||||
public int? Total { get; set; }
|
|
||||||
public List<ArtistAlbumItem>? Items { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record SpotifyPlusGetArtistAlbumsResponse
|
|
||||||
{
|
|
||||||
public UserProfile? UserProfile { get; set; }
|
|
||||||
public ArtistAlbumResult? Result { get; set; }
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,102 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.Types
|
|
||||||
{
|
|
||||||
public static class Globals
|
|
||||||
{
|
|
||||||
public const string DefaultDevId = "homie";
|
|
||||||
public const string DefaultEntityId = "media_player.spotifyplus";
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://jsonformatter.org/yaml-to-json
|
|
||||||
// https://json2csharp.com
|
|
||||||
// https://github.com/thlucas1/homeassistantcomponent_spotifyplus/blob/master/custom_components/spotifyplus/services.yaml
|
|
||||||
public record Restrictions { }
|
|
||||||
|
|
||||||
public record UserProfile
|
|
||||||
{
|
|
||||||
public string? Country { get; set; }
|
|
||||||
public string? DisplayName { get; set; }
|
|
||||||
public string? Email { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? Product { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record ExternalUrls
|
|
||||||
{
|
|
||||||
public string? Spotify { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record Followers
|
|
||||||
{
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public int? Total { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record Image
|
|
||||||
{
|
|
||||||
public string? Url { get; set; }
|
|
||||||
public int? Height { get; set; }
|
|
||||||
public int? Width { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record Owner
|
|
||||||
{
|
|
||||||
public string? DisplayName { get; set; }
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public Followers? Followers { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record Tracks
|
|
||||||
{
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public int? Total { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record Artist
|
|
||||||
{
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? Name { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record ExternalIds
|
|
||||||
{
|
|
||||||
public object? Ean { get; set; }
|
|
||||||
public string? Isrc { get; set; }
|
|
||||||
public object? Upc { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record Album
|
|
||||||
{
|
|
||||||
public string? AlbumType { get; set; }
|
|
||||||
public List<Artist>? Artists { get; set; }
|
|
||||||
public List<string>? AvailableMarkets { get; set; }
|
|
||||||
public List<object>? Copyrights { get; set; }
|
|
||||||
public ExternalIds? ExternalIds { get; set; }
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public List<object>? Genres { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? ImageUrl { get; set; }
|
|
||||||
public List<Image>? Images { get; set; }
|
|
||||||
public object? Label { get; set; }
|
|
||||||
public string? Name { get; set; }
|
|
||||||
public object? Popularity { get; set; }
|
|
||||||
public string? ReleaseDate { get; set; }
|
|
||||||
public string? ReleaseDatePrecision { get; set; }
|
|
||||||
public Restrictions? Restrictions { get; set; }
|
|
||||||
public int? TotalTracks { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,40 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.Types
|
|
||||||
{
|
|
||||||
public record PlaylistsItem
|
|
||||||
{
|
|
||||||
public bool? Collaborative { get; set; }
|
|
||||||
public string? Description { get; set; }
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? ImageUrl { get; set; }
|
|
||||||
public List<Image>? Images { get; set; }
|
|
||||||
public string? Name { get; set; }
|
|
||||||
public Owner? Owner { get; set; }
|
|
||||||
public bool? Public { get; set; }
|
|
||||||
public string? SnapshotId { get; set; }
|
|
||||||
public Tracks? Tracks { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record PlaylistsResult
|
|
||||||
{
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public int? Limit { get; set; }
|
|
||||||
public object? Next { get; set; }
|
|
||||||
public int? Offset { get; set; }
|
|
||||||
public object? Previous { get; set; }
|
|
||||||
public int? Total { get; set; }
|
|
||||||
public List<PlaylistsItem>? Items { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record SpotifyplusPlaylistResponse
|
|
||||||
{
|
|
||||||
public UserProfile? UserProfile { get; set; }
|
|
||||||
public PlaylistsResult? Result { get; set; }
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,41 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.Types
|
|
||||||
{
|
|
||||||
public record AlbumItem
|
|
||||||
{
|
|
||||||
public string? AlbumType { get; set; }
|
|
||||||
public List<Artist>? Artists { get; set; }
|
|
||||||
public List<string>? AvailableMarkets { get; set; }
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? ImageUrl { get; set; }
|
|
||||||
public List<Image>? Images { get; set; }
|
|
||||||
public string? Name { get; set; }
|
|
||||||
public string? ReleaseDate { get; set; }
|
|
||||||
public string? ReleaseDatePrecision { get; set; }
|
|
||||||
public Restrictions? Restrictions { get; set; }
|
|
||||||
public int? TotalTracks { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record AlbumResult
|
|
||||||
{
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public int? Limit { get; set; }
|
|
||||||
public string? Next { get; set; }
|
|
||||||
public int? Offset { get; set; }
|
|
||||||
public object? Previous { get; set; }
|
|
||||||
public int? Total { get; set; }
|
|
||||||
public List<AlbumItem>? Items { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record SpotifyplusSearchAlbumsResponse
|
|
||||||
{
|
|
||||||
public UserProfile? UserProfile { get; set; }
|
|
||||||
public AlbumResult? Result { get; set; }
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,37 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.Types
|
|
||||||
{
|
|
||||||
public record ArtistItem
|
|
||||||
{
|
|
||||||
public ExternalUrls? ExternalUrls { get; init; }
|
|
||||||
public Followers? Followers { get; init; }
|
|
||||||
public List<string>? Genres { get; init; }
|
|
||||||
public string? Href { get; init; }
|
|
||||||
public string? Id { get; init; }
|
|
||||||
public string? ImageUrl { get; init; }
|
|
||||||
public List<Image>? Images { get; init; }
|
|
||||||
public string? Name { get; init; }
|
|
||||||
public int? Popularity { get; init; }
|
|
||||||
public string? Type { get; init; }
|
|
||||||
public string? Uri { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record ArtistResult
|
|
||||||
{
|
|
||||||
public string? Href { get; init; }
|
|
||||||
public int? Limit { get; init; }
|
|
||||||
public string? Next { get; init; }
|
|
||||||
public int? Offset { get; set; }
|
|
||||||
public object? Previous { get; init; }
|
|
||||||
public int? Total { get; init; }
|
|
||||||
public List<ArtistItem>? Items { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record SpotifyplusSearchArtistsResponse
|
|
||||||
{
|
|
||||||
public UserProfile? UserProfile { get; init; }
|
|
||||||
public ArtistResult? Result { get; init; }
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,46 +0,0 @@
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Spotify.Types
|
|
||||||
{
|
|
||||||
public record SongItem
|
|
||||||
{
|
|
||||||
public Album? Album { get; set; }
|
|
||||||
public List<Artist>? Artists { get; set; }
|
|
||||||
public List<string>? AvailableMarkets { get; set; }
|
|
||||||
public int? DiscNumber { get; set; }
|
|
||||||
public int? DurationMs { get; set; }
|
|
||||||
public bool? Explicit { get; set; }
|
|
||||||
public ExternalIds? ExternalIds { get; set; }
|
|
||||||
public ExternalUrls? ExternalUrls { get; set; }
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public string? Id { get; set; }
|
|
||||||
public string? ImageUrl { get; set; }
|
|
||||||
public bool? IsLocal { get; set; }
|
|
||||||
public object? IsPlayable { get; set; }
|
|
||||||
public string? Name { get; set; }
|
|
||||||
public int? Popularity { get; set; }
|
|
||||||
public object? PreviewUrl { get; set; }
|
|
||||||
public Restrictions? Restrictions { get; set; }
|
|
||||||
public int? TrackNumber { get; set; }
|
|
||||||
public string? Type { get; set; }
|
|
||||||
public string? Uri { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record SongResult
|
|
||||||
{
|
|
||||||
public string? Href { get; set; }
|
|
||||||
public int? Limit { get; set; }
|
|
||||||
public string? Next { get; set; }
|
|
||||||
public int? Offset { get; set; }
|
|
||||||
public object? Previous { get; set; }
|
|
||||||
public int? Total { get; set; }
|
|
||||||
public List<SongItem>? Items { get; set; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public record SpotifyplusSearchTracksResponse
|
|
||||||
{
|
|
||||||
public UserProfile? UserProfile { get; set; }
|
|
||||||
public SongResult? Result { get; set; }
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,73 +0,0 @@
|
||||||
using System;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Reactive.Linq;
|
|
||||||
using System.Text.Json;
|
|
||||||
using System.Text.Json.Serialization;
|
|
||||||
|
|
||||||
using HomeAssistantGenerated;
|
|
||||||
|
|
||||||
using NetDaemon.AppModel;
|
|
||||||
using NetDaemon.HassModel;
|
|
||||||
using NetDaemon.HassModel.Entities;
|
|
||||||
|
|
||||||
|
|
||||||
namespace NetDaemonConfig.Apps.Timer.Setup
|
|
||||||
{
|
|
||||||
public record TimerFinishedEventData
|
|
||||||
{
|
|
||||||
[JsonPropertyName("entity_id")]
|
|
||||||
public string? entity_id { get; init; }
|
|
||||||
}
|
|
||||||
|
|
||||||
[NetDaemonApp]
|
|
||||||
public class Setup
|
|
||||||
{
|
|
||||||
private readonly string _timerTarget = "media_player.music_player_daemon";
|
|
||||||
|
|
||||||
private readonly string _timerTtsTarget = "tts.piper";
|
|
||||||
private readonly string _timerTtsMessage = "A set timer has finished.";
|
|
||||||
|
|
||||||
// TODO: private readonly string timerMediaLocation = "/path/to/file.mp3";
|
|
||||||
|
|
||||||
public Setup(IHaContext ha, Services services, Entities entities)
|
|
||||||
{
|
|
||||||
ha.Events
|
|
||||||
.Where(x => x.EventType == "timer.finished")
|
|
||||||
.Subscribe(x =>
|
|
||||||
{
|
|
||||||
if (x.DataElement.HasValue)
|
|
||||||
{
|
|
||||||
var data = JsonSerializer.Deserialize<TimerFinishedEventData>(x.DataElement.Value);
|
|
||||||
|
|
||||||
if (data?.entity_id?.StartsWith("timer.assist_timer") ?? false)
|
|
||||||
{
|
|
||||||
var timer = entities.Timer
|
|
||||||
.EnumerateAll()
|
|
||||||
.ToLookup((timer) => timer.EntityId,
|
|
||||||
(timer) => timer)[data.entity_id].First();
|
|
||||||
|
|
||||||
if (timer is not null)
|
|
||||||
{
|
|
||||||
services.Tts.Speak(
|
|
||||||
cache: true,
|
|
||||||
mediaPlayerEntityId: _timerTarget,
|
|
||||||
target: new ServiceTarget { EntityIds = [_timerTtsTarget] },
|
|
||||||
message: _timerTtsMessage);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
/*
|
|
||||||
entities.Timer.EnumerateAll()
|
|
||||||
.Where((timer) => timer.EntityId.StartsWith("timer.assist_timer"))
|
|
||||||
.StateChanges()
|
|
||||||
.Subscribe((timer) =>
|
|
||||||
{
|
|
||||||
if (timer.Old?.State != "idle" && timer.New?.State == "idle")
|
|
||||||
{}
|
|
||||||
});
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,22 +0,0 @@
|
||||||
{
|
|
||||||
"Logging": {
|
|
||||||
"LogLevel": {
|
|
||||||
"Default": "Debug",
|
|
||||||
"Microsoft": "Warning"
|
|
||||||
},
|
|
||||||
"ConsoleThemeType": "Ansi"
|
|
||||||
},
|
|
||||||
"HomeAssistant": {
|
|
||||||
"Host": "homie.nelim.org",
|
|
||||||
"Port": 443,
|
|
||||||
"Ssl": true
|
|
||||||
},
|
|
||||||
"NetDaemon": {
|
|
||||||
"ApplicationConfigurationFolder": "./apps"
|
|
||||||
},
|
|
||||||
"CodeGeneration": {
|
|
||||||
"Namespace": "HomeAssistantGenerated",
|
|
||||||
"OutputFile": "HomeAssistantGenerated.cs",
|
|
||||||
"UseAttributeBaseClasses" : "false"
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,103 +0,0 @@
|
||||||
{
|
|
||||||
config,
|
|
||||||
self,
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}: let
|
|
||||||
inherit (builtins) attrValues replaceStrings;
|
|
||||||
inherit (config.sops) secrets;
|
|
||||||
|
|
||||||
inherit (pkgs.callPackage ./package.nix {}) netdaemonConfig;
|
|
||||||
in {
|
|
||||||
khepri.compositions."netdaemon" = {
|
|
||||||
networks.netdaemon = {external = true;};
|
|
||||||
|
|
||||||
services."netdaemon5" = {
|
|
||||||
image = import ./images/netdaemon.nix pkgs;
|
|
||||||
restart = "always";
|
|
||||||
|
|
||||||
environmentFiles = [secrets.netdaemon.path];
|
|
||||||
environment = {
|
|
||||||
HomeAssistant__Host = "homie.nelim.org";
|
|
||||||
HomeAssistant__Port = "443";
|
|
||||||
HomeAssistant__Ssl = "true";
|
|
||||||
NetDaemon__ApplicationAssembly = "netdaemon.dll";
|
|
||||||
Logging__LogLevel__Default = "Information"; # use Information/Debug/Trace/Warning/Error
|
|
||||||
TZ = "America/New_York";
|
|
||||||
};
|
|
||||||
|
|
||||||
volumes = ["${netdaemonConfig}:/data"];
|
|
||||||
networks = ["netdaemon"];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
services.home-assistant = {
|
|
||||||
customComponents = attrValues {
|
|
||||||
inherit
|
|
||||||
(self.scopedPackages.${pkgs.system}.hass-components)
|
|
||||||
netdaemon
|
|
||||||
;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
environment.systemPackages = let
|
|
||||||
nixFetchDeps =
|
|
||||||
replaceStrings [(toString self)] ["$FLAKE"]
|
|
||||||
#nix
|
|
||||||
''
|
|
||||||
let
|
|
||||||
config = (builtins.getFlake ("$FLAKE")).nixosConfigurations.homie;
|
|
||||||
inherit (config) pkgs;
|
|
||||||
|
|
||||||
netdaemonConfig = pkgs.callPackage ${toString ./package.nix} {};
|
|
||||||
in
|
|
||||||
netdaemonConfig.fetch-deps
|
|
||||||
'';
|
|
||||||
in [
|
|
||||||
(pkgs.writeShellApplication {
|
|
||||||
name = "bumpNetdaemonDeps";
|
|
||||||
|
|
||||||
runtimeInputs = with pkgs; [
|
|
||||||
dos2unix
|
|
||||||
dotnet-sdk_9
|
|
||||||
];
|
|
||||||
|
|
||||||
text = ''
|
|
||||||
# Install codegen
|
|
||||||
dotnet tool install --create-manifest-if-needed NetDaemon.HassModel.CodeGen --version "$(cat ./.version)"
|
|
||||||
|
|
||||||
# Run it
|
|
||||||
dotnet tool run nd-codegen -token "$(sed 's/HomeAssistant__Token=//' ${secrets.netdaemon.path})"
|
|
||||||
dos2unix ./HomeAssistantGenerated.cs
|
|
||||||
|
|
||||||
# This is to not have it count towards CSharp in the repo
|
|
||||||
mv ./HomeAssistantGenerated.cs ./HomeAssistantGenerated
|
|
||||||
|
|
||||||
# Update all nugets to latest versions
|
|
||||||
regex='PackageReference Include="([^"]*)" Version="([^"]*)"'
|
|
||||||
|
|
||||||
find . -type f -name '*.csproj' | while read -r file; do
|
|
||||||
# Extract unique package names from the .csproj file
|
|
||||||
packages=$(grep -oP "$regex" "$file" | sed -E 's/.*Include="([^"]*)".*/\1/' | sort -u)
|
|
||||||
|
|
||||||
# Loop through each package and update
|
|
||||||
for package in $packages; do
|
|
||||||
echo -e "\033[35mUpdate $file package: $package\033[0m"
|
|
||||||
dotnet add "$file" package "$package"
|
|
||||||
done
|
|
||||||
done
|
|
||||||
|
|
||||||
$(nix build --no-link --print-out-paths --impure --expr "$(cat <<EOF
|
|
||||||
${nixFetchDeps}
|
|
||||||
EOF
|
|
||||||
)") .
|
|
||||||
|
|
||||||
alejandra -q .
|
|
||||||
rm -r "$FLAKE/.config"
|
|
||||||
|
|
||||||
sed -i "s/finalImageTag = .*/finalImageTag = \"$(cat ./.version)\";/" ./images/netdaemon.nix
|
|
||||||
updateImages .
|
|
||||||
'';
|
|
||||||
})
|
|
||||||
];
|
|
||||||
}
|
|
|
@ -1,297 +0,0 @@
|
||||||
[
|
|
||||||
{
|
|
||||||
"pname": "Cronos",
|
|
||||||
"version": "0.9.0",
|
|
||||||
"hash": "sha256-yDYBfqSXqvT/VPUf6UT3XOgqqPmOMYqhjCBxpF5i15c="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "FuzzySharp",
|
|
||||||
"version": "2.0.2",
|
|
||||||
"hash": "sha256-GuWqVOo+AG8MSvIbusLPjKfJFQRJhSSJ9eGWljTBA/c="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.CodeAnalysis.Analyzers",
|
|
||||||
"version": "3.3.4",
|
|
||||||
"hash": "sha256-qDzTfZBSCvAUu9gzq2k+LOvh6/eRvJ9++VCNck/ZpnE="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.CodeAnalysis.Common",
|
|
||||||
"version": "4.12.0",
|
|
||||||
"hash": "sha256-mm/OKG3zPLAeTVGZtuLxSG+jpQDOchn1oyHqBBJW2Ho="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.CodeAnalysis.CSharp",
|
|
||||||
"version": "4.12.0",
|
|
||||||
"hash": "sha256-m1i1Q5pyEq4lAoYjNE9baEjTplH8+bXx5wSA+eMmehk="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-uBLeb4z60y8z7NelHs9uT3cLD6wODkdwyfJm6/YZLDM="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.Abstractions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-xtG2USC9Qm0f2Nn6jkcklpyEDT3hcEZOxOwTc0ep7uc="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.Binder",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-6ajYWcNOQX2WqftgnoUmVtyvC1kkPOtTCif4AiKEffU="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.CommandLine",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-RE6DotU1FM1sy5p3hukT+WOFsDYJRsKX6jx5vhlPceM="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.EnvironmentVariables",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-tDJx2prYZpr0RKSwmJfsK9FlUGwaDmyuSz2kqQxsWoI="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.FileExtensions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-PsLo6mrLGYfbi96rfCG8YS1APXkUXBG4hLstpT60I4s="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.Json",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-qQn7Ol0CvPYuyecYWYBkPpTMdocO7I6n+jXQI2udzLI="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Configuration.UserSecrets",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-GoEk+Qq7lbiwWurHYx1LkDaUzIpOzaoTiVGDPfViGak="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.DependencyInjection",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-dAH52PPlTLn7X+1aI/7npdrDzMEFPMXRv4isV1a+14k="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.DependencyInjection.Abstractions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-CncVwkKZ5CsIG2O0+OM9qXuYXh3p6UGyueTHSLDVL+c="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.DependencyModel",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-xirwlMWM0hBqgTneQOGkZ8l45mHT08XuSSRIbprgq94="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Diagnostics",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-JMbhtjdcWRlrcrbgPlowfj26+pM+MYhnPIaYKnv9byU="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Diagnostics.Abstractions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-wG1LcET+MPRjUdz3HIOTHVEnbG/INFJUqzPErCM79eY="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.FileProviders.Abstractions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-mVfLjZ8VrnOQR/uQjv74P2uEG+rgW72jfiGdSZhIfDc="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.FileProviders.Physical",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-IzFpjKHmF1L3eVbFLUZa2N5aH3oJkJ7KE1duGIS7DP8="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.FileSystemGlobbing",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-eBLa8pW/y/hRj+JbEr340zbHRABIeFlcdqE0jf5/Uhc="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Hosting",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-apIN4Cz86ujsMp/ibxcvguA9uCFaFqOsZ4kAUPX5ASI="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Hosting.Abstractions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-NhEDqZGnwCDFyK/NKn1dwLQExYE82j1YVFcrhXVczqY="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Http",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-MsStH3oUfyBbcSEoxm+rfxFBKI/rtB5PZrSGvtDjVe0="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-kR16c+N8nQrWeYLajqnXPg7RiXjZMSFLnKLEs4VfjcM="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging.Abstractions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-iBTs9twjWXFeERt4CErkIIcoJZU1jrd1RWCI8V5j7KU="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging.Configuration",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-ysPjBq64p6JM4EmeVndryXnhLWHYYszzlVpPxRWkUkw="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging.Console",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-N2t9EUdlS6ippD4Z04qUUyBuQ4tKSR/8TpmKScb5zRw="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging.Debug",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-5W6fP9Eb98U3MTWKeLzSNl2cRFpE694OOPjpWp/qTAk="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging.EventLog",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-mIL1I85Ef5+/mXl24odoUpcXet+jCZTRtKCd5z6YUwI="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Logging.EventSource",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-pplZskMsR3gGbs3I0wycGsvIMPIpfWFJpOsR9GkiYRw="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Options",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-DT5euAQY/ItB5LPI8WIp6Dnd0lSvBRP35vFkOXC68ck="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Options.ConfigurationExtensions",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-r1Z3sEVSIjeH2UKj+KMj86har68g/zybSqoSjESBcoA="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Microsoft.Extensions.Primitives",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-ZNLusK1CRuq5BZYZMDqaz04PIKScE2Z7sS2tehU7EJs="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.AppModel",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-3q+FmCwPkWDm44EYZIwAnnwewcn7r0Z0Z3Gbgxq12bA="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.AppModel.SourceDeployedApps",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-fzCV53c4blMjEpdDMS5fI1KKvgNKfNlxQMqmmS11ng4="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.Client",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-GEf0VFJtWeiK6yUisY3U+LGDBWBgf4hWJ+dmDymB5LU="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.Extensions.Logging",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-9+UxeGN0xzOUEQ8cneCy/4YhY6qOQ6usDOnnVlDMV74="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.Extensions.Scheduling",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-HT0SNoxqujYjKcY4JfEVNKxdfaOFuyUQHKmUGKKTpao="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.Extensions.Tts",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-rgxPfSO/kMbq/a3kCCm+9nAAzVoHBUcPMRWhVVDrKlI="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.HassModel",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-oNL0JGAtZQVhYYsUv8rXMKmSNLvi2fI41gF9zT8mTB8="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.HassModel.Integration",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-YuhQZ///XjH0iNEF5wS1QcnFnAUD0ZA3ktPovF3P3dM="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "NetDaemon.Runtime",
|
|
||||||
"version": "24.52.0",
|
|
||||||
"hash": "sha256-FQqu3g/J/5AXcSrb7WXGSeFA9zjPnLPOCKwDNqYRVlI="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog",
|
|
||||||
"version": "4.0.0",
|
|
||||||
"hash": "sha256-j8hQ5TdL1TjfdGiBO9PyHJFMMPvATHWN1dtrrUZZlNw="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog",
|
|
||||||
"version": "4.2.0",
|
|
||||||
"hash": "sha256-7f3EpCsEbDxXgsuhE430KVI14p7oDUuCtwRpOCqtnbs="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.AspNetCore",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-h58CFtXBRvwhTCrhQPHQMKbp98YiK02o+cOyOmktVpQ="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Extensions.Hosting",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-bidr2foe7Dp4BJOlkc7ko0q6vt9ITG3IZ8b2BKRa0pw="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Extensions.Logging",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-aGkz1V4HVl0rWC1BkcnLhG1EC7WLBoT3tdLdUUTFXaw="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Formatting.Compact",
|
|
||||||
"version": "3.0.0",
|
|
||||||
"hash": "sha256-nejEYqJEMG9P2iFZvbsCUPr5LZRtxbdUTLCI9N71jHY="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Settings.Configuration",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-Q/q5UiSrcxoy5a/orod20E2RfiRtHDhxjjGMe1dW35I="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Sinks.Console",
|
|
||||||
"version": "6.0.0",
|
|
||||||
"hash": "sha256-QH8ykDkLssJ99Fgl+ZBFBr+RQRl0wRTkeccQuuGLyro="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Sinks.Debug",
|
|
||||||
"version": "3.0.0",
|
|
||||||
"hash": "sha256-7/LmoRF1rUDFhJ47bTRQQFRgSHnZDO8484r3sCGqYvE="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "Serilog.Sinks.File",
|
|
||||||
"version": "6.0.0",
|
|
||||||
"hash": "sha256-KQmlUpG9ovRpNqKhKe6rz3XMLUjkBqjyQhEm2hV5Sow="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "System.Collections.Immutable",
|
|
||||||
"version": "8.0.0",
|
|
||||||
"hash": "sha256-F7OVjKNwpqbUh8lTidbqJWYi476nsq9n+6k0+QVRo3w="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "System.Diagnostics.EventLog",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-tPvt6yoAp56sK/fe+/ei8M65eavY2UUhRnbrREj/Ems="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "System.IO.Pipelines",
|
|
||||||
"version": "9.0.0",
|
|
||||||
"hash": "sha256-vb0NrPjfEao3kfZ0tavp2J/29XnsQTJgXv3/qaAwwz0="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "System.Reactive",
|
|
||||||
"version": "6.0.1",
|
|
||||||
"hash": "sha256-Lo5UMqp8DsbVSUxa2UpClR1GoYzqQQcSxkfyFqB/d4Q="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "System.Reflection.Metadata",
|
|
||||||
"version": "8.0.0",
|
|
||||||
"hash": "sha256-dQGC30JauIDWNWXMrSNOJncVa1umR1sijazYwUDdSIE="
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pname": "YamlDotNet",
|
|
||||||
"version": "16.2.1",
|
|
||||||
"hash": "sha256-Nu/rD43sihE4PTHC5r2Ua2gafclqcd2U95RcNFvGFhc="
|
|
||||||
}
|
|
||||||
]
|
|
|
@ -1,8 +0,0 @@
|
||||||
pkgs:
|
|
||||||
pkgs.dockerTools.pullImage rec {
|
|
||||||
imageName = "netdaemon/netdaemon5";
|
|
||||||
imageDigest = "sha256:5acac9a6f9dca0d8eab178ee211a73e03b3cf9fa2c8deecf45afce33f57d2011";
|
|
||||||
hash = "sha256-IjRyb/6uAzPKu0pMFDxXBkZt2rTG43rXQx2SnmPXptI=";
|
|
||||||
finalImageName = imageName;
|
|
||||||
finalImageTag = "24.52.0";
|
|
||||||
}
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue