mirror of
				https://github.com/Smaug123/nix-dotfiles
				synced 2025-10-26 16:38:58 +00:00 
			
		
		
		
	Compare commits
	
		
			4 Commits
		
	
	
		
			main
			...
			5e3f257ce6
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 5e3f257ce6 | ||
|  | 134b0dc8a7 | ||
|  | a290279914 | ||
|  | bea4ca1220 | 
							
								
								
									
										12
									
								
								.github/workflows/lint.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/lint.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -11,18 +11,16 @@ jobs: | |||||||
|     runs-on: "ubuntu-latest" |     runs-on: "ubuntu-latest" | ||||||
|     steps: |     steps: | ||||||
|       - name: "Checkout" |       - name: "Checkout" | ||||||
|         uses: "actions/checkout@v5" |         uses: "actions/checkout@v4" | ||||||
|       - name: "Install Nix" |       - name: "Install Nix" | ||||||
|         uses: "cachix/install-nix-action@v31" |         uses: "cachix/install-nix-action@v25" | ||||||
|         with: { "extra_nix_config": "access-tokens = github.com=${{ secrets.GITHUB_TOKEN }}" } |         with: { "extra_nix_config": "access-tokens = github.com=${{ secrets.GITHUB_TOKEN }}" } | ||||||
|       - name: "Check flake" |       - name: "Check flake" | ||||||
|         run: "nix flake check --all-systems" |         run: "nix flake check --all-systems" | ||||||
|  |  | ||||||
|   all-required-checks-complete: |   all-required-checks-complete: | ||||||
|     if:  ${{ always() }} |  | ||||||
|     runs-on: "ubuntu-latest" |     runs-on: "ubuntu-latest" | ||||||
|     steps: |     steps: | ||||||
|       - uses: Smaug123/all-required-checks-complete-action@05b40a8c47ef0b175ea326e9abb09802cb67b44e |       - run: "echo \"All required checks complete.\"" | ||||||
|         with: |     needs: | ||||||
|           needs-context: ${{ toJSON(needs) }} |       - "flake-check" | ||||||
|     needs: [ "flake-check" ] |  | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,4 +3,3 @@ result | |||||||
| bin/ | bin/ | ||||||
| obj/ | obj/ | ||||||
| .DS_Store | .DS_Store | ||||||
| .direnv/ |  | ||||||
|   | |||||||
| @@ -1,6 +1,9 @@ | |||||||
|  | This repository currently has no licence applied to it, except for the NeoVim configuration. | ||||||
|  | That configuration is in large part derived from https://github.com/amix/vimrc and is therefore provided under the following licence. | ||||||
|  |  | ||||||
| The MIT License (MIT) | The MIT License (MIT) | ||||||
|  |  | ||||||
| Copyright (c) 2024 Patrick Stevens | Copyright (c) 2016 Amir Salihefendic | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
| of this software and associated documentation files (the "Software"), to deal | of this software and associated documentation files (the "Software"), to deal | ||||||
|   | |||||||
| @@ -1,11 +1,17 @@ | |||||||
| {pkgs, ...}: let | {pkgs, ...}: let | ||||||
|   mbsync = import ./mbsync.nix {inherit pkgs;}; |   python = import ./python.nix {inherit pkgs;}; | ||||||
| in { | in { | ||||||
|  |   nix.useDaemon = true; | ||||||
|  |  | ||||||
|   # List packages installed in system profile. To search by name, run: |   # List packages installed in system profile. To search by name, run: | ||||||
|   # $ nix-env -qaP | grep wget |   # $ nix-env -qaP | grep wget | ||||||
|  |  | ||||||
|   environment.systemPackages = [ |   environment.systemPackages = [ | ||||||
|     pkgs.python3 |     pkgs.alacritty | ||||||
|  |     pkgs.rustup | ||||||
|  |     pkgs.libiconv | ||||||
|  |     pkgs.clang | ||||||
|  |     python | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   users.users.patrick = { |   users.users.patrick = { | ||||||
| @@ -15,105 +21,22 @@ in { | |||||||
|  |  | ||||||
|   # This line is required; otherwise, on shell startup, you won't have Nix stuff in the PATH. |   # This line is required; otherwise, on shell startup, you won't have Nix stuff in the PATH. | ||||||
|   programs.zsh.enable = true; |   programs.zsh.enable = true; | ||||||
|   programs.gnupg.agent.enable = true; |  | ||||||
|  |  | ||||||
|   # Use a custom configuration.nix location. |   # Use a custom configuration.nix location. | ||||||
|   # $ darwin-rebuild switch -I darwin-config=$HOME/.config/nixpkgs/darwin/configuration.nix |   # $ darwin-rebuild switch -I darwin-config=$HOME/.config/nixpkgs/darwin/configuration.nix | ||||||
|   environment.darwinConfig = "$HOME/.nixpkgs/darwin-configuration.nix"; |   environment.darwinConfig = "$HOME/.nixpkgs/darwin-configuration.nix"; | ||||||
|  |  | ||||||
|   launchd.agents = { |  | ||||||
|     mbsync-btinternet = { |  | ||||||
|       command = "${mbsync}/bin/mbsync BTInternet > /tmp/mbsync.btinternet.log 2>/tmp/mbsync.btinternet.2.log"; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 60; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     mbsync-proton = { |  | ||||||
|       command = "${mbsync}/bin/mbsync Proton > /tmp/mbsync.proton.1.log 2>/tmp/mbsync.proton.2.log"; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 60; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     mbsync-gmail = { |  | ||||||
|       command = "${mbsync}/bin/mbsync Gmail > /tmp/mbsync.gmail.1.log 2>/tmp/mbsync.gmail.2.log"; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         # Refresh token is 60min long, so do this more often than that! |  | ||||||
|         StartInterval = 30; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     backup-calendar = { |  | ||||||
|       command = ''${pkgs.bash}/bin/bash -c "mkdir -p '/Users/patrick/Library/Application Support/RadicaleBackups' && if [ ! -d '/Users/patrick/Library/Application Support/RadicaleBackups/.git' ] ; then ${pkgs.git}/bin/git clone root@patrickstevens.co.uk:/preserve/radicale/data/.git '/Users/patrick/Library/Application Support/RadicaleBackups' >/tmp/radicale.out.log 2>/tmp/radicale.err.log; fi && ${pkgs.git}/bin/git --git-dir '/Users/patrick/Library/Application Support/RadicaleBackups/.git' --work-tree '/Users/patrick/Library/Application Support/RadicaleBackups/' pull 2>>/tmp/radicale.err.log"''; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 3600; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     sync-nixpkgs = { |  | ||||||
|       command = ''${pkgs.bash}/bin/bash -c "if [ -d /Users/patrick/Documents/GitHub/nixpkgs ] ; then ${pkgs.git}/bin/git --git-dir /Users/patrick/Documents/GitHub/nixpkgs/.git --work-tree '/Users/patrick/Documents/GitHub/nixpkgs/' fetch origin ; fi"''; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 36000; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     sync-dotnet-api-docs = { |  | ||||||
|       command = ''${pkgs.bash}/bin/bash -c "if [ -d /Users/patrick/Documents/GitHub/dotnet-api-docs ] ; then ${pkgs.git}/bin/git --git-dir /Users/patrick/Documents/GitHub/dotnet-api-docs/.git --work-tree '/Users/patrick/Documents/GitHub/dotnet-api-docs' fetch origin ; fi"''; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 36000; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     sync-dotnet-docs = { |  | ||||||
|       command = ''${pkgs.bash}/bin/bash -c "if [ -d /Users/patrick/Documents/GitHub/dotnet-docs ] ; then ${pkgs.git}/bin/git --git-dir /Users/patrick/Documents/GitHub/dotnet-docs/.git --work-tree '/Users/patrick/Documents/GitHub/dotnet-docs' fetch origin ; fi"''; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 36000; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     nix-store-optimise = { |  | ||||||
|       command = ''${pkgs.nix}/bin/nix store optimise''; |  | ||||||
|       serviceConfig = { |  | ||||||
|         KeepAlive = false; |  | ||||||
|         UserName = "patrick"; |  | ||||||
|         StartInterval = 72000; |  | ||||||
|         RunAtLoad = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   # Auto upgrade nix package and the daemon service. |   # Auto upgrade nix package and the daemon service. | ||||||
|  |   services.nix-daemon.enable = true; | ||||||
|   nix.package = pkgs.nixVersions.stable; |   nix.package = pkgs.nixVersions.stable; | ||||||
|   nix.gc.automatic = true; |   nix.gc.automatic = true; | ||||||
|  |   nix.nixPath = ["darwin=/nix/store/zq4v3pi2wsfsrjkpk71kcn8srhbwjabf-nix-darwin"]; | ||||||
|  |  | ||||||
|   # Sandbox causes failure: https://github.com/NixOS/nix/issues/4119 |   # Sandbox causes failure: https://github.com/NixOS/nix/issues/4119 | ||||||
|   nix.settings.sandbox = false; |   nix.settings.sandbox = false; | ||||||
|  |  | ||||||
|   # Optimising store leads to transient build failures https://github.com/NixOS/nix/issues/7273 |  | ||||||
|   nix.extraOptions = '' |   nix.extraOptions = '' | ||||||
|     auto-optimise-store = false |     auto-optimise-store = true | ||||||
|     experimental-features = nix-command flakes |     experimental-features = nix-command flakes | ||||||
|     extra-experimental-features = ca-derivations |     extra-experimental-features = ca-derivations | ||||||
|     max-jobs = auto  # Allow building multiple derivations in parallel |     max-jobs = auto  # Allow building multiple derivations in parallel | ||||||
| @@ -127,6 +50,4 @@ in { | |||||||
|   # Used for backwards compatibility, please read the changelog before changing. |   # Used for backwards compatibility, please read the changelog before changing. | ||||||
|   # $ darwin-rebuild changelog |   # $ darwin-rebuild changelog | ||||||
|   system.stateVersion = 4; |   system.stateVersion = 4; | ||||||
|  |  | ||||||
|   system.primaryUser = "patrick"; |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										189
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										189
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							| @@ -3,14 +3,15 @@ | |||||||
|     "apple-silicon": { |     "apple-silicon": { | ||||||
|       "inputs": { |       "inputs": { | ||||||
|         "flake-compat": "flake-compat", |         "flake-compat": "flake-compat", | ||||||
|         "nixpkgs": "nixpkgs" |         "nixpkgs": "nixpkgs", | ||||||
|  |         "rust-overlay": "rust-overlay" | ||||||
|       }, |       }, | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1759818599, |         "lastModified": 1705557527, | ||||||
|         "narHash": "sha256-4Go3gVl3E+geWMcFQ+06qlkO/lJlSvS9dyhYiXLWYq0=", |         "narHash": "sha256-DuxxHTQ/W5KToFLWG4FUF8hLldNo9eXlbt7JgvhrMnY=", | ||||||
|         "owner": "tpwrules", |         "owner": "tpwrules", | ||||||
|         "repo": "nixos-apple-silicon", |         "repo": "nixos-apple-silicon", | ||||||
|         "rev": "24ab28e47b586f741910b3a2f0428f3523a0fff3", |         "rev": "6e324ab06cb27a19409ebc1dc2664bf1e585490a", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -26,11 +27,11 @@ | |||||||
|         ] |         ] | ||||||
|       }, |       }, | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1760721282, |         "lastModified": 1705915768, | ||||||
|         "narHash": "sha256-aAHphQbU9t/b2RRy2Eb8oMv+I08isXv2KUGFAFn7nCo=", |         "narHash": "sha256-+Jlz8OAqkOwJlioac9wtpsCnjgGYUhvLpgJR/5tP9po=", | ||||||
|         "owner": "lnl7", |         "owner": "lnl7", | ||||||
|         "repo": "nix-darwin", |         "repo": "nix-darwin", | ||||||
|         "rev": "c3211fcd0c56c11ff110d346d4487b18f7365168", |         "rev": "1e706ef323de76236eb183d7784f3bd57255ec0b", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -40,13 +41,35 @@ | |||||||
|         "type": "github" |         "type": "github" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "emacs": { | ||||||
|  |       "inputs": { | ||||||
|  |         "flake-utils": "flake-utils", | ||||||
|  |         "nixpkgs": [ | ||||||
|  |           "nixpkgs" | ||||||
|  |         ], | ||||||
|  |         "nixpkgs-stable": "nixpkgs-stable" | ||||||
|  |       }, | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1706170797, | ||||||
|  |         "narHash": "sha256-oGuFylWYU9OY5DaEJEK+Z7EL81Ln27xz01LN9+8U0P0=", | ||||||
|  |         "owner": "nix-community", | ||||||
|  |         "repo": "emacs-overlay", | ||||||
|  |         "rev": "dd5d758f69dd1ae6d0399763aa73ca34974ce9e3", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "nix-community", | ||||||
|  |         "repo": "emacs-overlay", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "flake-compat": { |     "flake-compat": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1746162366, |         "lastModified": 1688025799, | ||||||
|         "narHash": "sha256-5SSSZ/oQkwfcAz/o/6TlejlVGqeK08wyREBQ5qFFPhM=", |         "narHash": "sha256-ktpB4dRtnksm9F5WawoIkEneh1nrEvuxb5lJFt1iOyw=", | ||||||
|         "owner": "nix-community", |         "owner": "nix-community", | ||||||
|         "repo": "flake-compat", |         "repo": "flake-compat", | ||||||
|         "rev": "0f158086a2ecdbb138cd0429410e44994f1b7e4b", |         "rev": "8bf105319d44f6b9f0d764efa4fdef9f1cc9ba1c", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -60,11 +83,29 @@ | |||||||
|         "systems": "systems" |         "systems": "systems" | ||||||
|       }, |       }, | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1731533236, |         "lastModified": 1705309234, | ||||||
|         "narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=", |         "narHash": "sha256-uNRRNRKmJyCRC/8y1RqBkqWBLM034y4qN7EprSdmgyA=", | ||||||
|         "owner": "numtide", |         "owner": "numtide", | ||||||
|         "repo": "flake-utils", |         "repo": "flake-utils", | ||||||
|         "rev": "11707dc2f618dd54ca8739b309ec4fc024de578b", |         "rev": "1ef2e671c3b0c19053962c07dbda38332dcebf26", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "numtide", | ||||||
|  |         "repo": "flake-utils", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "flake-utils_2": { | ||||||
|  |       "inputs": { | ||||||
|  |         "systems": "systems_2" | ||||||
|  |       }, | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1701680307, | ||||||
|  |         "narHash": "sha256-kAuep2h5ajznlPMD9rnQyffWG8EM/C73lejGofXvdM8=", | ||||||
|  |         "owner": "numtide", | ||||||
|  |         "repo": "flake-utils", | ||||||
|  |         "rev": "4022d587cbbfd70fe950c1e2083a02621806a725", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -80,11 +121,11 @@ | |||||||
|         ] |         ] | ||||||
|       }, |       }, | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1760887455, |         "lastModified": 1706134977, | ||||||
|         "narHash": "sha256-/xU8iYZjolWbMUNBQF6af5zgGs73Qw21WMgz1tLs3Yw=", |         "narHash": "sha256-KwNb1Li3K6vuVwZ77tFjZ89AWBo7AiCs9t0Cens4BsM=", | ||||||
|         "owner": "nix-community", |         "owner": "nix-community", | ||||||
|         "repo": "home-manager", |         "repo": "home-manager", | ||||||
|         "rev": "aeabc1ac63e6ebb8ba4714c4abdfe0556f2de765", |         "rev": "6359d40f6ec0b72a38e02b333f343c3d4929ec10", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -107,27 +148,59 @@ | |||||||
|     }, |     }, | ||||||
|     "nixpkgs": { |     "nixpkgs": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1759756860, |         "lastModified": 1705316053, | ||||||
|         "narHash": "sha256-7QLUQZein1TbpouPOAsTka8vQLujHRmmTqze2IkdrqY=", |         "narHash": "sha256-J2Ey5mPFT8gdfL2XC0JTZvKaBw/b2pnyudEXFvl+dQM=", | ||||||
|         "owner": "NixOS", |         "owner": "nixos", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "rev": "d3baaf296366efdea6737124f05b65d1cf25fa7c", |         "rev": "c3e128f3c0ecc1fb04aef9f72b3dcc2f6cecf370", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
|         "owner": "nixos", |         "owner": "nixos", | ||||||
|         "ref": "nixos-unstable", |         "repo": "nixpkgs", | ||||||
|  |         "rev": "c3e128f3c0ecc1fb04aef9f72b3dcc2f6cecf370", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "nixpkgs-stable": { | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1705916986, | ||||||
|  |         "narHash": "sha256-iBpfltu6QvN4xMpen6jGGEb6jOqmmVQKUrXdOJ32u8w=", | ||||||
|  |         "owner": "NixOS", | ||||||
|  |         "repo": "nixpkgs", | ||||||
|  |         "rev": "d7f206b723e42edb09d9d753020a84b3061a79d8", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "NixOS", | ||||||
|  |         "ref": "nixos-23.11", | ||||||
|  |         "repo": "nixpkgs", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "nixpkgs-stable_2": { | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1705033721, | ||||||
|  |         "narHash": "sha256-K5eJHmL1/kev6WuqyqqbS1cdNnSidIZ3jeqJ7GbrYnQ=", | ||||||
|  |         "owner": "NixOS", | ||||||
|  |         "repo": "nixpkgs", | ||||||
|  |         "rev": "a1982c92d8980a0114372973cbdfe0a307f1bdea", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "NixOS", | ||||||
|  |         "ref": "release-23.05", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "nixpkgs_2": { |     "nixpkgs_2": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1760872779, |         "lastModified": 1706006310, | ||||||
|         "narHash": "sha256-c5C907Raf9eY8f1NUXYeju9aUDlm227s/V0OptEbypA=", |         "narHash": "sha256-nDPz0fj0IFcDhSTlXBU2aixcnGs2Jm4Zcuoj0QtmiXQ=", | ||||||
|         "owner": "NixOS", |         "owner": "NixOS", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "rev": "63bdb5d90fa2fa11c42f9716ad1e23565613b07c", |         "rev": "b43bb235efeab5324c5e486882ef46749188eee2", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -139,11 +212,11 @@ | |||||||
|     }, |     }, | ||||||
|     "nixpkgs_3": { |     "nixpkgs_3": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1734254970, |         "lastModified": 1695033101, | ||||||
|         "narHash": "sha256-yZzYWWWeOqSFvIirHzY1SJiuSBPmXIYJFhyt+1zkv8A=", |         "narHash": "sha256-RQ4m+ycjdLdass7Hr4+Lzwnjw7wGhcUkKqWiJS3YxPM=", | ||||||
|         "owner": "nixos", |         "owner": "nixos", | ||||||
|         "repo": "nixpkgs", |         "repo": "nixpkgs", | ||||||
|         "rev": "d388ee0ec8c623389ab3a7caead258a94cec14de", |         "rev": "d941d9491804e0ca01e03468dbf6f8d3a7919a16", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
| @@ -156,11 +229,50 @@ | |||||||
|       "inputs": { |       "inputs": { | ||||||
|         "apple-silicon": "apple-silicon", |         "apple-silicon": "apple-silicon", | ||||||
|         "darwin": "darwin", |         "darwin": "darwin", | ||||||
|  |         "emacs": "emacs", | ||||||
|         "home-manager": "home-manager", |         "home-manager": "home-manager", | ||||||
|         "nixpkgs": "nixpkgs_2", |         "nixpkgs": "nixpkgs_2", | ||||||
|  |         "sops-nix": "sops-nix", | ||||||
|         "whisper": "whisper" |         "whisper": "whisper" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "rust-overlay": { | ||||||
|  |       "flake": false, | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1686795910, | ||||||
|  |         "narHash": "sha256-jDa40qRZ0GRQtP9EMZdf+uCbvzuLnJglTUI2JoHfWDc=", | ||||||
|  |         "owner": "oxalica", | ||||||
|  |         "repo": "rust-overlay", | ||||||
|  |         "rev": "5c2b97c0a9bc5217fc3dfb1555aae0fb756d99f9", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "oxalica", | ||||||
|  |         "repo": "rust-overlay", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "sops-nix": { | ||||||
|  |       "inputs": { | ||||||
|  |         "nixpkgs": [ | ||||||
|  |           "nixpkgs" | ||||||
|  |         ], | ||||||
|  |         "nixpkgs-stable": "nixpkgs-stable_2" | ||||||
|  |       }, | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1706130372, | ||||||
|  |         "narHash": "sha256-fHZxKH1DhsXPP36a2vJ91Zy6S+q6+QRIFlpLr9fZHU8=", | ||||||
|  |         "owner": "Mic92", | ||||||
|  |         "repo": "sops-nix", | ||||||
|  |         "rev": "4606d9b1595e42ffd9b75b9e69667708c70b1d68", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "Mic92", | ||||||
|  |         "repo": "sops-nix", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "systems": { |     "systems": { | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1681028828, |         "lastModified": 1681028828, | ||||||
| @@ -176,18 +288,33 @@ | |||||||
|         "type": "github" |         "type": "github" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "systems_2": { | ||||||
|  |       "locked": { | ||||||
|  |         "lastModified": 1681028828, | ||||||
|  |         "narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=", | ||||||
|  |         "owner": "nix-systems", | ||||||
|  |         "repo": "default", | ||||||
|  |         "rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e", | ||||||
|  |         "type": "github" | ||||||
|  |       }, | ||||||
|  |       "original": { | ||||||
|  |         "owner": "nix-systems", | ||||||
|  |         "repo": "default", | ||||||
|  |         "type": "github" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "whisper": { |     "whisper": { | ||||||
|       "inputs": { |       "inputs": { | ||||||
|         "flake-utils": "flake-utils", |         "flake-utils": "flake-utils_2", | ||||||
|         "model": "model", |         "model": "model", | ||||||
|         "nixpkgs": "nixpkgs_3" |         "nixpkgs": "nixpkgs_3" | ||||||
|       }, |       }, | ||||||
|       "locked": { |       "locked": { | ||||||
|         "lastModified": 1743962136, |         "lastModified": 1704121968, | ||||||
|         "narHash": "sha256-YsKxkEGqGE+c0L+k8Vczq9UHpzSktR9/tm3zrF7abzo=", |         "narHash": "sha256-N8FJb+ohJ4Qt/m5RoAbwm3RP4VRjl+hA6PUCfjPhZo8=", | ||||||
|         "owner": "Smaug123", |         "owner": "Smaug123", | ||||||
|         "repo": "whisper.cpp", |         "repo": "whisper.cpp", | ||||||
|         "rev": "e1faa8b19ead213f507dbabda45ac54b8765a6eb", |         "rev": "04f8e0cdc73abe7c593b2c9405f0f590c51de95a", | ||||||
|         "type": "github" |         "type": "github" | ||||||
|       }, |       }, | ||||||
|       "original": { |       "original": { | ||||||
|   | |||||||
							
								
								
									
										62
									
								
								flake.nix
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								flake.nix
									
									
									
									
									
								
							| @@ -14,6 +14,14 @@ | |||||||
|       # url = "github:Smaug123/nix-darwin/extract"; |       # url = "github:Smaug123/nix-darwin/extract"; | ||||||
|       inputs.nixpkgs.follows = "nixpkgs"; |       inputs.nixpkgs.follows = "nixpkgs"; | ||||||
|     }; |     }; | ||||||
|  |     emacs = { | ||||||
|  |       url = "github:nix-community/emacs-overlay"; | ||||||
|  |       inputs.nixpkgs.follows = "nixpkgs"; | ||||||
|  |     }; | ||||||
|  |     sops-nix = { | ||||||
|  |       url = "github:Mic92/sops-nix"; | ||||||
|  |       inputs.nixpkgs.follows = "nixpkgs"; | ||||||
|  |     }; | ||||||
|     apple-silicon = { |     apple-silicon = { | ||||||
|       url = "github:tpwrules/nixos-apple-silicon"; |       url = "github:tpwrules/nixos-apple-silicon"; | ||||||
|     }; |     }; | ||||||
| @@ -23,19 +31,23 @@ | |||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   outputs = { |   outputs = { | ||||||
|  |     self, | ||||||
|     darwin, |     darwin, | ||||||
|  |     emacs, | ||||||
|     nixpkgs, |     nixpkgs, | ||||||
|     home-manager, |     home-manager, | ||||||
|  |     sops-nix, | ||||||
|     apple-silicon, |     apple-silicon, | ||||||
|     whisper, |     whisper, | ||||||
|     ... |     ... | ||||||
|   }: let |   } @ inputs: let | ||||||
|     config = { |     config = { | ||||||
|       # contentAddressedByDefault = true; |       # contentAddressedByDefault = true; | ||||||
|       allowUnfree = true; |       allowUnfree = true; | ||||||
|     }; |     }; | ||||||
|     systems = ["aarch64-darwin" "aarch64-linux" "x86_64-linux"]; |     systems = ["aarch64-darwin" "aarch64-linux" "x86_64-linux"]; | ||||||
|   in let |   in let | ||||||
|  |     overlays = [emacs.overlay] ++ import ./overlays.nix; | ||||||
|     recursiveMerge = attrList: let |     recursiveMerge = attrList: let | ||||||
|       f = attrPath: |       f = attrPath: | ||||||
|         builtins.zipAttrsWith (n: values: |         builtins.zipAttrsWith (n: values: | ||||||
| @@ -50,40 +62,10 @@ | |||||||
|       f [] attrList; |       f [] attrList; | ||||||
|   in { |   in { | ||||||
|     nixosConfigurations = { |     nixosConfigurations = { | ||||||
|       capybara = let |  | ||||||
|         system = "x86_64-linux"; |  | ||||||
|       in let |  | ||||||
|         pkgs = import nixpkgs {inherit system config;}; |  | ||||||
|       in |  | ||||||
|         nixpkgs.lib.nixosSystem { |  | ||||||
|           inherit system; |  | ||||||
|           specialArgs = { |  | ||||||
|             username = "patrick"; |  | ||||||
|             dotnet = pkgs.dotnet-sdk_8; |  | ||||||
|           }; |  | ||||||
|           modules = let |  | ||||||
|             args = { |  | ||||||
|               nixpkgs = pkgs; |  | ||||||
|               username = "patrick"; |  | ||||||
|               dotnet = pkgs.dotnet-sdk_8; |  | ||||||
|               mbsync = import ./mbsync.nix {inherit pkgs;}; |  | ||||||
|               secretsPath = "/home/patrick/.secrets/"; |  | ||||||
|               machinename = "capybara"; |  | ||||||
|             }; |  | ||||||
|           in [ |  | ||||||
|             ./home-manager/capybara-config.nix |  | ||||||
|             home-manager.nixosModules.home-manager |  | ||||||
|             { |  | ||||||
|               home-manager.useGlobalPkgs = true; |  | ||||||
|               home-manager.useUserPackages = true; |  | ||||||
|               home-manager.users.patrick = recursiveMerge [(import ./home-manager/linux.nix args) (import ./home-manager/home.nix args)]; |  | ||||||
|             } |  | ||||||
|           ]; |  | ||||||
|         }; |  | ||||||
|       earthworm = let |       earthworm = let | ||||||
|         system = "aarch64-linux"; |         system = "aarch64-linux"; | ||||||
|       in let |       in let | ||||||
|         pkgs = import nixpkgs {inherit system config;}; |         pkgs = import nixpkgs {inherit system config overlays;}; | ||||||
|       in |       in | ||||||
|         nixpkgs.lib.nixosSystem { |         nixpkgs.lib.nixosSystem { | ||||||
|           inherit system; |           inherit system; | ||||||
| @@ -92,9 +74,6 @@ | |||||||
|               nixpkgs = pkgs; |               nixpkgs = pkgs; | ||||||
|               username = "patrick"; |               username = "patrick"; | ||||||
|               dotnet = pkgs.dotnet-sdk_8; |               dotnet = pkgs.dotnet-sdk_8; | ||||||
|               mbsync = import ./mbsync.nix {inherit pkgs;}; |  | ||||||
|               secretsPath = "/home/patrick/.secrets/"; |  | ||||||
|               machinename = "earthworm"; |  | ||||||
|             }; |             }; | ||||||
|           in [ |           in [ | ||||||
|             ./home-manager/earthworm-config.nix |             ./home-manager/earthworm-config.nix | ||||||
| @@ -103,7 +82,7 @@ | |||||||
|             { |             { | ||||||
|               home-manager.useGlobalPkgs = true; |               home-manager.useGlobalPkgs = true; | ||||||
|               home-manager.useUserPackages = true; |               home-manager.useUserPackages = true; | ||||||
|               home-manager.users.patrick = recursiveMerge [(import ./home-manager/linux.nix args) (import ./home-manager/home.nix args)]; |               home-manager.users.patrick = recursiveMerge [(import ./home-manager/earthworm.nix args) (import ./home-manager/home.nix args)]; | ||||||
|             } |             } | ||||||
|           ]; |           ]; | ||||||
|         }; |         }; | ||||||
| @@ -111,7 +90,7 @@ | |||||||
|     darwinConfigurations = let |     darwinConfigurations = let | ||||||
|       system = "aarch64-darwin"; |       system = "aarch64-darwin"; | ||||||
|     in let |     in let | ||||||
|       pkgs = import nixpkgs {inherit system config;}; |       pkgs = import nixpkgs {inherit system config overlays;}; | ||||||
|     in { |     in { | ||||||
|       nixpkgs = pkgs; |       nixpkgs = pkgs; | ||||||
|       patrick = darwin.lib.darwinSystem { |       patrick = darwin.lib.darwinSystem { | ||||||
| @@ -122,12 +101,10 @@ | |||||||
|             username = "patrick"; |             username = "patrick"; | ||||||
|             dotnet = pkgs.dotnet-sdk_8; |             dotnet = pkgs.dotnet-sdk_8; | ||||||
|             whisper = whisper.packages.${system}; |             whisper = whisper.packages.${system}; | ||||||
|             mbsync = import ./mbsync.nix {inherit pkgs;}; |  | ||||||
|             secretsPath = "/Users/patrick/.secrets/"; |  | ||||||
|             machinename = "darwin"; |  | ||||||
|           }; |           }; | ||||||
|         in [ |         in [ | ||||||
|           ./darwin-configuration.nix |           ./darwin-configuration.nix | ||||||
|  |           sops-nix.nixosModules.sops | ||||||
|           home-manager.darwinModules.home-manager |           home-manager.darwinModules.home-manager | ||||||
|           { |           { | ||||||
|             home-manager.useGlobalPkgs = true; |             home-manager.useGlobalPkgs = true; | ||||||
| @@ -144,12 +121,11 @@ | |||||||
|         pkgs.stdenvNoCC.mkDerivation { |         pkgs.stdenvNoCC.mkDerivation { | ||||||
|           name = "fmt-check"; |           name = "fmt-check"; | ||||||
|           src = ./.; |           src = ./.; | ||||||
|           nativeBuildInputs = [pkgs.alejandra pkgs.shellcheck pkgs.shfmt pkgs.stylua]; |           nativeBuildInputs = [pkgs.alejandra pkgs.shellcheck pkgs.shfmt]; | ||||||
|           checkPhase = '' |           checkPhase = '' | ||||||
|             find . -type f -name '*.sh' | xargs shfmt -d -s -i 2 -ci |             find . -type f -name '*.sh' | xargs shfmt -d -s -i 2 -ci | ||||||
|             alejandra -c . |             alejandra -c . | ||||||
|             find . -type f -name '*.sh' -exec shellcheck -x {} \; |             find . -type f -name '*.sh' -exec shellcheck -x {} \; | ||||||
|             find . -type f -name '*.lua' -exec stylua --check {} \; |  | ||||||
|           ''; |           ''; | ||||||
|           installPhase = "mkdir $out"; |           installPhase = "mkdir $out"; | ||||||
|           dontBuild = true; |           dontBuild = true; | ||||||
| @@ -167,7 +143,7 @@ | |||||||
|           pkgs = import nixpkgs {inherit config system;}; |           pkgs = import nixpkgs {inherit config system;}; | ||||||
|         in { |         in { | ||||||
|           default = pkgs.mkShell { |           default = pkgs.mkShell { | ||||||
|             buildInputs = [pkgs.alejandra pkgs.shellcheck pkgs.stylua]; |             buildInputs = [pkgs.alejandra pkgs.shellcheck]; | ||||||
|           }; |           }; | ||||||
|         } |         } | ||||||
|       ); |       ); | ||||||
|   | |||||||
| @@ -1,42 +0,0 @@ | |||||||
| # Do not modify this file!  It was generated by ‘nixos-generate-config’ |  | ||||||
| # and may be overwritten by future invocations.  Please make changes |  | ||||||
| # to /etc/nixos/configuration.nix instead. |  | ||||||
| { |  | ||||||
|   config, |  | ||||||
|   lib, |  | ||||||
|   modulesPath, |  | ||||||
|   ... |  | ||||||
| }: { |  | ||||||
|   imports = [ |  | ||||||
|     (modulesPath + "/installer/scan/not-detected.nix") |  | ||||||
|   ]; |  | ||||||
|  |  | ||||||
|   boot.initrd.availableKernelModules = ["nvme" "xhci_pci" "ahci" "usb_storage" "usbhid" "sd_mod"]; |  | ||||||
|   boot.initrd.kernelModules = []; |  | ||||||
|   boot.kernelModules = ["kvm-amd"]; |  | ||||||
|   boot.extraModulePackages = []; |  | ||||||
|  |  | ||||||
|   fileSystems."/" = { |  | ||||||
|     device = "/dev/disk/by-uuid/63c5394d-55ce-48a9-8d7c-2b68f3b5f834"; |  | ||||||
|     fsType = "ext4"; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   fileSystems."/boot" = { |  | ||||||
|     device = "/dev/disk/by-uuid/5BCD-7078"; |  | ||||||
|     fsType = "vfat"; |  | ||||||
|     options = ["fmask=0077" "dmask=0077"]; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   swapDevices = []; |  | ||||||
|  |  | ||||||
|   # Enables DHCP on each ethernet and wireless interface. In case of scripted networking |  | ||||||
|   # (the default) this is the recommended approach. When using systemd-networkd it's |  | ||||||
|   # still possible to use this option, but it's recommended to use it in conjunction |  | ||||||
|   # with explicit per-interface declarations with `networking.interfaces.<interface>.useDHCP`. |  | ||||||
|   networking.useDHCP = lib.mkDefault true; |  | ||||||
|   # networking.interfaces.eno1.useDHCP = lib.mkDefault true; |  | ||||||
|   # networking.interfaces.wlp9s0.useDHCP = lib.mkDefault true; |  | ||||||
|  |  | ||||||
|   nixpkgs.hostPlatform = lib.mkDefault "x86_64-linux"; |  | ||||||
|   hardware.cpu.amd.updateMicrocode = lib.mkDefault config.hardware.enableRedistributableFirmware; |  | ||||||
| } |  | ||||||
| @@ -1,90 +0,0 @@ | |||||||
| { |  | ||||||
|   pkgs, |  | ||||||
|   config, |  | ||||||
|   ... |  | ||||||
| }: { |  | ||||||
|   nixpkgs.config.allowUnfree = true; |  | ||||||
|   imports = [ |  | ||||||
|     ../hardware/capybara.nix |  | ||||||
|   ]; |  | ||||||
|  |  | ||||||
|   hardware.graphics = { |  | ||||||
|     enable = true; |  | ||||||
|     enable32Bit = true; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   hardware.bluetooth.enable = true; |  | ||||||
|  |  | ||||||
|   security.rtkit.enable = true; |  | ||||||
|   services.pipewire = { |  | ||||||
|     enable = true; |  | ||||||
|     alsa.enable = true; |  | ||||||
|     alsa.support32Bit = true; |  | ||||||
|     pulse.enable = true; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   boot.loader.systemd-boot.enable = true; |  | ||||||
|   boot.loader.efi.canTouchEfiVariables = true; |  | ||||||
|   boot.loader.grub.useOSProber = true; |  | ||||||
|  |  | ||||||
|   boot.kernelParams = [ |  | ||||||
|     "video=DP-1:2560x1440@144" |  | ||||||
|     "video=HDMI-A-1:1920x1080@144" |  | ||||||
|   ]; |  | ||||||
|  |  | ||||||
|   boot.extraModulePackages = [config.boot.kernelPackages.rtl8821au]; |  | ||||||
|  |  | ||||||
|   networking = { |  | ||||||
|     hostName = "capybara"; |  | ||||||
|     networkmanager.enable = true; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   time.timeZone = "Europe/London"; |  | ||||||
|  |  | ||||||
|   programs.sway.enable = true; |  | ||||||
|   programs.zsh.enable = true; |  | ||||||
|  |  | ||||||
|   # TODO: work out secrets management for password, then set mutableUsers to false |  | ||||||
|   users.mutableUsers = true; |  | ||||||
|   users.users.patrick = { |  | ||||||
|     isNormalUser = true; |  | ||||||
|     extraGroups = ["wheel" "networkManager"]; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   services.syncthing = { |  | ||||||
|     enable = true; |  | ||||||
|     user = "patrick"; |  | ||||||
|     dataDir = "/home/patrick/syncthing"; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   environment.systemPackages = [ |  | ||||||
|     pkgs.git |  | ||||||
|     pkgs.vim |  | ||||||
|     pkgs.wget |  | ||||||
|     pkgs.tmux |  | ||||||
|     pkgs.home-manager |  | ||||||
|     pkgs.firefox |  | ||||||
|   ]; |  | ||||||
|  |  | ||||||
|   environment.loginShellInit = '' |  | ||||||
|     [[ "$(tty)" == /dev/tty1 ]] && sway --unsupported-gpu |  | ||||||
|   ''; |  | ||||||
|  |  | ||||||
|   services.openssh.enable = true; |  | ||||||
|  |  | ||||||
|   system.stateVersion = "23.11"; |  | ||||||
|   nix.settings.experimental-features = ["nix-command" "flakes" "ca-derivations"]; |  | ||||||
|  |  | ||||||
|   nix.gc.automatic = true; |  | ||||||
|   nix.extraOptions = '' |  | ||||||
|     auto-optimise-store = true |  | ||||||
|     max-jobs = auto |  | ||||||
|     keep-outputs = true |  | ||||||
|     keep-derivations = true |  | ||||||
|   ''; |  | ||||||
|  |  | ||||||
|   programs.steam = { |  | ||||||
|     enable = true; |  | ||||||
|     remotePlay.openFirewall = true; |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
| @@ -23,15 +23,12 @@ | |||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   programs.vscode = { |   programs.vscode = { | ||||||
|     profiles.default = { |     userSettings = { | ||||||
|       userSettings = { |       "lean.leanpkgPath" = "/Users/${username}/.elan/toolchains/stable/bin/leanpkg"; | ||||||
|         "lean.leanpkgPath" = "/Users/${username}/.elan/toolchains/stable/bin/leanpkg"; |       "lean.executablePath" = "/Users/${username}/.elan/toolchains/stable/bin/lean"; | ||||||
|         "lean.executablePath" = "/Users/${username}/.elan/toolchains/stable/bin/lean"; |       "lean.memoryLimit" = 16384; | ||||||
|         "lean.memoryLimit" = 16384; |       "latex-workshop.view.pdf.viewer" = "tab"; | ||||||
|         "latex-workshop.view.pdf.viewer" = "tab"; |       "lean4.toolchainPath" = "/Users/${username}/.elan/toolchains/leanprover--lean4---nightly-2022-12-16"; | ||||||
|         "lean4.toolchainPath" = "/Users/${username}/.elan/toolchains/leanprover--lean4---nightly-2022-12-16"; |  | ||||||
|         "git.openRepositoryInParentFolders" = "always"; |  | ||||||
|       }; |  | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,28 +0,0 @@ | |||||||
| # Configure Rider to use the correct .NET paths from an ambient .NET |  | ||||||
| use_rider_dotnet() { |  | ||||||
|     # Get paths |  | ||||||
|     DOTNET_PATH=$(readlink "$(which dotnet)") |  | ||||||
|     SETTINGS_FILE=$(find . -maxdepth 1 -type f -name '*.sln.DotSettings.user') |  | ||||||
|     MSBUILD=$(realpath "$(find "$(dirname "$DOTNET_PATH")/../share/dotnet/sdk" -maxdepth 2 -type f -name MSBuild.dll)") |  | ||||||
|  |  | ||||||
|     # Update Rider settings if they exist |  | ||||||
|     if [ -f "$SETTINGS_FILE" ] ; then |  | ||||||
|         xmlstarlet ed --inplace \ |  | ||||||
|         -N wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation" \ |  | ||||||
|         -N x="http://schemas.microsoft.com/winfx/2006/xaml" \ |  | ||||||
|         -N s="clr-namespace:System;assembly=mscorlib" \ |  | ||||||
|         -N ss="urn:shemas-jetbrains-com:settings-storage-xaml" \ |  | ||||||
|         --update "//s:String[@x:Key='/Default/Environment/Hierarchy/Build/BuildTool/DotNetCliExePath/@EntryValue']" \ |  | ||||||
|         --value "$(realpath "$(dirname "$DOTNET_PATH")/../share/dotnet/dotnet")" \ |  | ||||||
|         "$SETTINGS_FILE" |  | ||||||
|          |  | ||||||
|         xmlstarlet ed --inplace \ |  | ||||||
|         -N wpf="http://schemas.microsoft.com/winfx/2006/xaml/presentation" \ |  | ||||||
|         -N x="http://schemas.microsoft.com/winfx/2006/xaml" \ |  | ||||||
|         -N s="clr-namespace:System;assembly=mscorlib" \ |  | ||||||
|         -N ss="urn:shemas-jetbrains-com:settings-storage-xaml" \ |  | ||||||
|         --update "//s:String[@x:Key='/Default/Environment/Hierarchy/Build/BuildTool/CustomBuildToolPath/@EntryValue']" \ |  | ||||||
|         --value "$MSBUILD" \ |  | ||||||
|         "$SETTINGS_FILE" |  | ||||||
|     fi |  | ||||||
| } |  | ||||||
| @@ -1,46 +1,13 @@ | |||||||
| {pkgs, ...}: { | { | ||||||
|  |   config, | ||||||
|  |   pkgs, | ||||||
|  |   ... | ||||||
|  | }: { | ||||||
|   imports = [ |   imports = [ | ||||||
|     ../hardware/earthworm.nix |     ../hardware/earthworm.nix | ||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   hardware.asahi.peripheralFirmwareDirectory = ./../firmware; |   hardware.asahi.peripheralFirmwareDirectory = ../firmware; | ||||||
|   hardware.asahi = { |  | ||||||
|     setupAsahiSound = true; |  | ||||||
|   }; |  | ||||||
|   hardware.graphics.enable = true; |  | ||||||
|   hardware.bluetooth.enable = true; |  | ||||||
|  |  | ||||||
|   programs.light.enable = true; |  | ||||||
|   services.actkbd = { |  | ||||||
|     enable = true; |  | ||||||
|     bindings = [ |  | ||||||
|       { |  | ||||||
|         keys = [225]; |  | ||||||
|         events = ["key"]; |  | ||||||
|         command = "${pkgs.light}/bin/light -A 10"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         keys = [224]; |  | ||||||
|         events = ["key"]; |  | ||||||
|         command = "${pkgs.light}/bin/light -U 10"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         keys = [113]; |  | ||||||
|         events = ["key"]; |  | ||||||
|         command = "${pkgs.alsa-utils}/bin/amixer -q set Master toggle"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         keys = [114]; |  | ||||||
|         events = ["key"]; |  | ||||||
|         command = "${pkgs.alsa-utils}/bin/amixer -q set Master 10- unmute"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         keys = [115]; |  | ||||||
|         events = ["key"]; |  | ||||||
|         command = "${pkgs.alsa-utils}/bin/amixer -q set Master 10+ unmute"; |  | ||||||
|       } |  | ||||||
|     ]; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   boot.loader.systemd-boot.enable = true; |   boot.loader.systemd-boot.enable = true; | ||||||
|   boot.loader.efi.canTouchEfiVariables = false; |   boot.loader.efi.canTouchEfiVariables = false; | ||||||
| @@ -51,10 +18,6 @@ | |||||||
|   networking = { |   networking = { | ||||||
|     hostName = "earthworm"; |     hostName = "earthworm"; | ||||||
|     networkmanager.enable = true; |     networkmanager.enable = true; | ||||||
|     wireless.iwd = { |  | ||||||
|       enable = true; |  | ||||||
|       settings.General.EnableNetworkConfiguration = true; |  | ||||||
|     }; |  | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   time.timeZone = "Europe/London"; |   time.timeZone = "Europe/London"; | ||||||
| @@ -75,7 +38,7 @@ | |||||||
|   ]; |   ]; | ||||||
|  |  | ||||||
|   environment.loginShellInit = '' |   environment.loginShellInit = '' | ||||||
|     [[ "$(tty)" == /dev/tty1 ]] && export WLR_RENDER_NO_EXPLICIT_SYNC=1 && sway |     [[ "$(tty)" == /dev/tty1 ]] && sway | ||||||
|   ''; |   ''; | ||||||
|  |  | ||||||
|   services.openssh.enable = true; |   services.openssh.enable = true; | ||||||
|   | |||||||
							
								
								
									
										112
									
								
								home-manager/earthworm.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								home-manager/earthworm.nix
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,112 @@ | |||||||
|  | { | ||||||
|  |   nixpkgs, | ||||||
|  |   username, | ||||||
|  |   dotnet, | ||||||
|  |   ... | ||||||
|  | }: { | ||||||
|  |   home.packages = [nixpkgs.firefox-wayland nixpkgs.wl-clipboard]; | ||||||
|  |   nixpkgs.config.firefox = { | ||||||
|  |     speechSynthesisSupport = true; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   # Sadly not implemented on Darwin | ||||||
|  |   programs.firefox = { | ||||||
|  |     enable = true; | ||||||
|  |     profiles = { | ||||||
|  |       patrick = { | ||||||
|  |         isDefault = true; | ||||||
|  |         name = "patrick"; | ||||||
|  |         search = {default = "Google";}; | ||||||
|  |         settings = { | ||||||
|  |           # see https://github.com/TLATER/dotfiles/blob/b39af91fbd13d338559a05d69f56c5a97f8c905d/home-config/config/graphical-applications/firefox.nix | ||||||
|  |           # see https://www.ghacks.net/2015/08/18/a-comprehensive-list-of-firefox-privacy-and-security-settings/ | ||||||
|  |           "browser.search.isUS" = false; | ||||||
|  |           "browser.search.region" = "GB"; | ||||||
|  |           "gfx.webrender.all" = true; # enable GPU acceleration | ||||||
|  |           "media.ffmpeg.vaapi.enabled" = true; | ||||||
|  |           "widget.dmabuf.force-enabled" = true; | ||||||
|  |           "privacy.webrtc.legacyGlobalIndicator" = false; | ||||||
|  |           "app.shield.optoutstudies.enabled" = false; | ||||||
|  |           "app.update.enabled" = false; | ||||||
|  |           "app.update.auto" = false; | ||||||
|  |           "app.update.silent" = false; | ||||||
|  |           "app.update.service.enabled" = false; | ||||||
|  |           "app.update.staging.enabled" = false; | ||||||
|  |           "browser.discovery.enabled" = false; | ||||||
|  |           "browser.laterrun.enabled" = false; | ||||||
|  |           "browser.shell.checkDefaultBrowser" = false; | ||||||
|  |           "browser.rights.3.shown" = true; | ||||||
|  |           "browser.search.update" = false; | ||||||
|  |           "extensions.update.enabled" = false; | ||||||
|  |           "extensions.update.autoUpdateDefault" = false; | ||||||
|  |           "extensions.getAddons.cache.enabled" = false; | ||||||
|  |           "dom.ipc.plugins.reportCrashURL" = false; | ||||||
|  |           "extensions.webservice.discoverURL" = "http://127.0.0.1"; | ||||||
|  |           "toolkit.telemetry.unified" = false; | ||||||
|  |           "toolkit.telemetry.unifiedIsOptIn" = true; | ||||||
|  |           "toolkit.telemetry.enabled" = false; | ||||||
|  |           "toolkit.telemetry.server" = ""; | ||||||
|  |           "toolkit.telemetry.archive.enabled" = false; | ||||||
|  |           "lightweightThemes.update.enabled" = false; | ||||||
|  |           "startup.homepage_welcome_url" = ""; | ||||||
|  |           "startup.homepage_welcome_url.additional" = ""; | ||||||
|  |           "startup.homepage_override_url" = ""; | ||||||
|  |           "datareporting.healthreport.uploadEnabled" = false; | ||||||
|  |           "datareporting.healthreport.documentServerURI" = ""; | ||||||
|  |           "datareporting.healthreport.service.enabled" = false; | ||||||
|  |           "datareporting.healthreport.about.reportUrl" = "data:text/plain,"; | ||||||
|  |           "toolkit.telemetry.cachedClientID" = ""; | ||||||
|  |           "browser.selfsupport.url" = ""; | ||||||
|  |           "browser.selfsupport.enabled" = false; | ||||||
|  |           "experiments.enabled" = false; | ||||||
|  |           "experiments.supported" = false; | ||||||
|  |           "experiments.activeExperiment" = false; | ||||||
|  |           "experiments.manifest.uri" = ""; | ||||||
|  |           "network.allow-experiments" = false; | ||||||
|  |           "breakpad.reportURL" = ""; | ||||||
|  |           "browser.tabs.crashReporting.sendReport" = false; | ||||||
|  |           "browser.newtab.preload" = false; | ||||||
|  |           "browser.newtabpage.directory.ping" = "data:text/plain,"; | ||||||
|  |           "browser.newtabpage.directory.source" = "data:text/plain,"; | ||||||
|  |           "browser.newtabpage.enabled" = false; | ||||||
|  |           "browser.newtabpage.enhanced" = false; | ||||||
|  |           "browser.newtabpage.introShown" = true; | ||||||
|  |           "browser.aboutHomeSnippets.updateUrl" = "https://127.0.0.1"; | ||||||
|  |           "extensions.pocket.enabled" = false; | ||||||
|  |           "extensions.pocket.api" = ""; | ||||||
|  |           "extensions.pocket.site" = ""; | ||||||
|  |           "extensions.pocket.oAuthConsumerKey" = ""; | ||||||
|  |           "social.whitelist" = ""; | ||||||
|  |           "social.toast-notifications.enabled" = false; | ||||||
|  |           "social.shareDirectory" = ""; | ||||||
|  |           "social.remote-install.enabled" = false; | ||||||
|  |           "social.directories" = ""; | ||||||
|  |           "social.share.activationPanelEnabled" = false; | ||||||
|  |           "social.enabled" = false; | ||||||
|  |           "dom.flyweb.enabled" = false; | ||||||
|  |           "services.sync.enabled" = false; | ||||||
|  |         }; | ||||||
|  |       }; | ||||||
|  |     }; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   wayland.windowManager.sway = { | ||||||
|  |     enable = true; | ||||||
|  |     config = { | ||||||
|  |       focus = {followMouse = false;}; | ||||||
|  |       modifier = "Mod4"; | ||||||
|  |       terminal = "alacritty"; | ||||||
|  |       window = {border = 5;}; | ||||||
|  |     }; | ||||||
|  |     extraConfig = '' | ||||||
|  |       output Unknown-1 scale 2 | ||||||
|  |     ''; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   services.swayidle = {enable = true;}; | ||||||
|  |   services.cbatticon = { | ||||||
|  |     lowLevelPercent = 20; | ||||||
|  |     iconType = "standard"; | ||||||
|  |     enable = true; | ||||||
|  |   }; | ||||||
|  | } | ||||||
| @@ -1,10 +1,7 @@ | |||||||
| { | { | ||||||
|   nixpkgs, |   nixpkgs, | ||||||
|   machinename, |  | ||||||
|   username, |   username, | ||||||
|   mbsync, |  | ||||||
|   dotnet, |   dotnet, | ||||||
|   secretsPath, |  | ||||||
|   ... |   ... | ||||||
| }: { | }: { | ||||||
|   # Let Home Manager install and manage itself. |   # Let Home Manager install and manage itself. | ||||||
| @@ -24,55 +21,64 @@ | |||||||
|   # changes in each release. |   # changes in each release. | ||||||
|   home.stateVersion = "22.05"; |   home.stateVersion = "22.05"; | ||||||
|  |  | ||||||
|   fonts.fontconfig.enable = true; |   programs.tmux = { | ||||||
|  |     shell = "${nixpkgs.zsh}/bin/zsh"; | ||||||
|  |     escapeTime = 50; | ||||||
|  |     mouse = false; | ||||||
|  |     prefix = "C-b"; | ||||||
|  |     enable = true; | ||||||
|  |     terminal = "screen-256color"; | ||||||
|  |     extraConfig = '' | ||||||
|  |       set-option -sa terminal-features ',xterm-256color:RGB' | ||||||
|  |     ''; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|   imports = [ |   programs.zsh = { | ||||||
|     # ./modules/agda.nix |     enable = true; | ||||||
|     # ./modules/emacs.nix |     autocd = true; | ||||||
|     ./modules/ghostty.nix |     enableAutosuggestions = true; | ||||||
|     ./modules/direnv.nix |     enableCompletion = true; | ||||||
|     ./modules/tmux.nix |     history = { | ||||||
|     ./modules/zsh.nix |       expireDuplicatesFirst = true; | ||||||
|     ./modules/ripgrep.nix |     }; | ||||||
|     ./modules/rust.nix |     oh-my-zsh = { | ||||||
|     ./modules/posix-sh.nix |       enable = true; | ||||||
|     (import ./modules/mail.nix |       plugins = ["git" "macos" "dircycle" "timer"]; | ||||||
|       { |       theme = "robbyrussell"; | ||||||
|         inherit mbsync secretsPath; |     }; | ||||||
|         pkgs = nixpkgs; |     sessionVariables = { | ||||||
|       }) |       EDITOR = "vim"; | ||||||
|   ]; |       LC_ALL = "en_US.UTF-8"; | ||||||
|  |       LC_CTYPE = "en_US.UTF-8"; | ||||||
|  |       RUSTFLAGS = "-L ${nixpkgs.libiconv}/lib -L ${nixpkgs.libcxxabi}/lib -L ${nixpkgs.libcxx}/lib"; | ||||||
|  |       RUST_BACKTRACE = "full"; | ||||||
|  |     }; | ||||||
|  |     shellAliases = { | ||||||
|  |       vim = "nvim"; | ||||||
|  |       view = "vim -R"; | ||||||
|  |       grep = "${nixpkgs.ripgrep}/bin/rg"; | ||||||
|  |     }; | ||||||
|  |     sessionVariables = { | ||||||
|  |       RIPGREP_CONFIG_PATH = "/Users/${username}/.config/ripgrep/config"; | ||||||
|  |     }; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|   programs.fzf = { |   programs.fzf = { | ||||||
|     enable = true; |     enable = true; | ||||||
|  |     enableZshIntegration = true; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   programs.difftastic = { |  | ||||||
|     enable = true; |  | ||||||
|     git.enable = true; |  | ||||||
|   }; |  | ||||||
|   programs.git = { |   programs.git = { | ||||||
|     package = nixpkgs.gitAndTools.gitFull; |     package = nixpkgs.gitAndTools.gitFull; | ||||||
|     enable = true; |     enable = true; | ||||||
|     settings = { |     userName = "Smaug123"; | ||||||
|       alias = { |     userEmail = "patrick+github@patrickstevens.co.uk"; | ||||||
|         co = "checkout"; |     aliases = { | ||||||
|         st = "status"; |       co = "checkout"; | ||||||
|       }; |       st = "status"; | ||||||
|       user = { |     }; | ||||||
|         email = "3138005+Smaug123@users.noreply.github.com"; |     delta = {enable = true;}; | ||||||
|         name = "Smaug123"; |     extraConfig = { | ||||||
|       }; |  | ||||||
|       commit.gpgsign = true; |  | ||||||
|       gpg.program = "${nixpkgs.gnupg}/bin/gpg"; |  | ||||||
|       user.signingkey = |  | ||||||
|         if machinename == "darwin" |  | ||||||
|         then "6D71064924BE1245" |  | ||||||
|         else if machinename == "earthworm" |  | ||||||
|         then "6E8B1BA1148AD7C9" |  | ||||||
|         else if machinename == "capybara" |  | ||||||
|         then "AE90453E879DBCFA" |  | ||||||
|         else throw "unrecognised machine name!"; |  | ||||||
|       core = { |       core = { | ||||||
|         autocrlf = "input"; |         autocrlf = "input"; | ||||||
|       }; |       }; | ||||||
| @@ -82,14 +88,6 @@ | |||||||
|       push = { |       push = { | ||||||
|         default = "current"; |         default = "current"; | ||||||
|         autoSetupRemote = true; |         autoSetupRemote = true; | ||||||
|         followTags = true; |  | ||||||
|       }; |  | ||||||
|       fetch = { |  | ||||||
|         prune = true; |  | ||||||
|         all = true; |  | ||||||
|       }; |  | ||||||
|       help = { |  | ||||||
|         autocorrect = "prompt"; |  | ||||||
|       }; |       }; | ||||||
|       pull = { |       pull = { | ||||||
|         rebase = false; |         rebase = false; | ||||||
| @@ -97,15 +95,6 @@ | |||||||
|       init = { |       init = { | ||||||
|         defaultBranch = "main"; |         defaultBranch = "main"; | ||||||
|       }; |       }; | ||||||
|       branch = { |  | ||||||
|         sort = "-committerdate"; |  | ||||||
|       }; |  | ||||||
|       column = { |  | ||||||
|         ui = "auto"; |  | ||||||
|       }; |  | ||||||
|       tag = { |  | ||||||
|         sort = "version:refname"; |  | ||||||
|       }; |  | ||||||
|       advice = { |       advice = { | ||||||
|         addIgnoredFile = false; |         addIgnoredFile = false; | ||||||
|       }; |       }; | ||||||
| @@ -123,227 +112,163 @@ | |||||||
|       }; |       }; | ||||||
|       diff = { |       diff = { | ||||||
|         colorMoved = "default"; |         colorMoved = "default"; | ||||||
|         algorithm = "histogram"; |  | ||||||
|         renames = true; |  | ||||||
|       }; |       }; | ||||||
|       "protocol.file" = { |       "protocol.file" = { | ||||||
|         allow = "always"; |         allow = "always"; | ||||||
|       }; |       }; | ||||||
|       url."git@github.com:" = { |  | ||||||
|         insteadOf = "https://github.com/"; |  | ||||||
|       }; |  | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   programs.vscode = { |   programs.vscode = { | ||||||
|     enable = true; |     enable = true; | ||||||
|  |     enableExtensionUpdateCheck = true; | ||||||
|  |     enableUpdateCheck = true; | ||||||
|     package = nixpkgs.vscode; |     package = nixpkgs.vscode; | ||||||
|     profiles.default = { |     extensions = import ./vscode-extensions.nix {pkgs = nixpkgs;}; | ||||||
|       extensions = import ./vscode-extensions.nix {pkgs = nixpkgs;}; |     userSettings = { | ||||||
|       enableExtensionUpdateCheck = true; |       workbench.colorTheme = "Default"; | ||||||
|       enableUpdateCheck = true; |       "files.Exclude" = { | ||||||
|       userSettings = { |         "**/.git" = true; | ||||||
|         workbench.colorTheme = "Default"; |         "**/.DS_Store" = true; | ||||||
|         "files.Exclude" = { |         "**/Thumbs.db" = true; | ||||||
|           "**/.git" = true; |         "**/*.olean" = true; | ||||||
|           "**/.DS_Store" = true; |         "**/result" = true; | ||||||
|           "**/Thumbs.db" = true; |  | ||||||
|           "**/*.olean" = true; |  | ||||||
|           "**/result" = true; |  | ||||||
|         }; |  | ||||||
|         "git.path" = "${nixpkgs.git}/bin/git"; |  | ||||||
|         "update.mode" = "none"; |  | ||||||
|         "explorer.confirmDelete" = false; |  | ||||||
|       }; |       }; | ||||||
|  |       "git.path" = "${nixpkgs.git}/bin/git"; | ||||||
|  |       "update.mode" = "none"; | ||||||
|  |       "explorer.confirmDelete" = false; | ||||||
|     }; |     }; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   services.syncthing = { |  | ||||||
|     enable = true; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   programs.neovim = let |   programs.neovim = let | ||||||
|     debugPyEnv = nixpkgs.python3.withPackages (ps: [ps.debugpy]); |     pythonEnv = nixpkgs.python3.withPackages (ps: [ | ||||||
|  |       ps.pynvim | ||||||
|  |       ps.pynvim-pp | ||||||
|  |       ps.pyyaml | ||||||
|  |       ps.std2 | ||||||
|  |     ]); | ||||||
|   in { |   in { | ||||||
|     enable = true; |     enable = true; | ||||||
|     plugins = [ |     plugins = [ | ||||||
|       { |       nixpkgs.vimPlugins.molokai | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-web-devicons; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.mini-nvim; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.satellite-nvim; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-lightbulb; |  | ||||||
|         type = "lua"; |  | ||||||
|         config = builtins.readFile ./nvim/nvim-lightbulb.lua; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.lean-nvim; |  | ||||||
|         type = "lua"; |  | ||||||
|         config = builtins.readFile ./nvim/lean.lua; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.which-key-nvim; |  | ||||||
|         type = "lua"; |  | ||||||
|         config = builtins.readFile ./nvim/which-key.lua; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.tokyonight-nvim; |  | ||||||
|         config = builtins.readFile ./nvim/tokyonight.lua; |  | ||||||
|         type = "lua"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-treesitter.withAllGrammars; |  | ||||||
|         config = builtins.readFile ./nvim/treesitter.lua; |  | ||||||
|         type = "lua"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-lspconfig; |  | ||||||
|         config = builtins.readFile ./nvim/lspconfig.lua; |  | ||||||
|         type = "lua"; |  | ||||||
|       } |  | ||||||
|       nixpkgs.vimPlugins.telescope-nvim |  | ||||||
|       nixpkgs.vimPlugins.tagbar |       nixpkgs.vimPlugins.tagbar | ||||||
|       nixpkgs.vimPlugins.fzf-vim |       nixpkgs.vimPlugins.fzf-vim | ||||||
|       { |  | ||||||
|         plugin = nixpkgs.vimPlugins.roslyn-nvim; |  | ||||||
|         config = builtins.readFile ./nvim/roslyn-nvim.lua; |  | ||||||
|         type = "lua"; |  | ||||||
|       } |  | ||||||
|       { |  | ||||||
|         plugin = let |  | ||||||
|           name = "venv-selector.nvim"; |  | ||||||
|           rev = "2ad34f36d498ff5193ea10f79c87688bd5284172"; |  | ||||||
|         in |  | ||||||
|           nixpkgs.vimUtils.buildVimPlugin { |  | ||||||
|             name = name; |  | ||||||
|             src = nixpkgs.fetchFromGitHub { |  | ||||||
|               owner = "linux-cultist"; |  | ||||||
|               repo = name; |  | ||||||
|               rev = rev; |  | ||||||
|               hash = "sha256-aOga7kJ1y3T2vDyYFl/XHOwk35ZqeUcfPUk+Pr1mIeo="; |  | ||||||
|             }; |  | ||||||
|           }; |  | ||||||
|         config = builtins.readFile ./nvim/venv-selector.lua; |  | ||||||
|         type = "lua"; |  | ||||||
|       } |  | ||||||
|       { |       { | ||||||
|         plugin = nixpkgs.vimPlugins.Ionide-vim; |         plugin = nixpkgs.vimPlugins.Ionide-vim; | ||||||
|         type = "lua"; |         config = '' | ||||||
|         config = builtins.readFile ./nvim/ionide-vim.lua; |           let g:fsharp#fsautocomplete_command = ['dotnet', 'fsautocomplete', '--background-service-enabled'] | ||||||
|  |           let g:fsharp#show_signature_on_cursor_move = 1 | ||||||
|  |           if has('nvim') && exists('*nvim_open_win') | ||||||
|  |             augroup FSharpGroup | ||||||
|  |               autocmd! | ||||||
|  |               autocmd FileType fsharp nnoremap <leader>t :call fsharp#showTooltip()<CR> | ||||||
|  |             augroup END | ||||||
|  |           endif | ||||||
|  |         ''; | ||||||
|       } |       } | ||||||
|       { |       { | ||||||
|         plugin = nixpkgs.vimPlugins.chadtree; |         plugin = nixpkgs.vimPlugins.chadtree; | ||||||
|         config = builtins.readFile ./nvim/chadtree.lua; |         config = "let g:chadtree_settings = {'xdg': v:true}"; | ||||||
|         type = "lua"; |  | ||||||
|       } |       } | ||||||
|       { |       { | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-cmp; |         plugin = nixpkgs.vimPlugins.coq_nvim; | ||||||
|         config = builtins.readFile ./nvim/nvim-cmp.lua; |         config = ''let g:coq_settings = { 'auto_start': v:true, 'xdg': v:true }''; | ||||||
|         type = "lua"; |  | ||||||
|       } |       } | ||||||
|       { |       { | ||||||
|         plugin = nixpkgs.vimPlugins.cmp-nvim-lsp; |         plugin = nixpkgs.vimPlugins.rust-vim; | ||||||
|  |         config = "let g:rustfmt_autosave = 1"; | ||||||
|       } |       } | ||||||
|       { |       { | ||||||
|         plugin = nixpkgs.vimPlugins.rustaceanvim; |         plugin = nixpkgs.vimPlugins.LanguageClient-neovim; | ||||||
|  |         config = "let g:LanguageClient_serverCommands = { 'nix': ['rnix-lsp'] }"; | ||||||
|       } |       } | ||||||
|       { |       { | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-dap; |         plugin = nixpkgs.vimPlugins.syntastic; | ||||||
|         config = builtins.readFile ./nvim/nvim-dap.lua; |         config = ''          let g:syntastic_rust_checkers = ['cargo'] | ||||||
|         type = "lua"; |           let g:syntastic_always_populate_loc_list = 1 | ||||||
|       } |           let g:syntastic_auto_loc_list = 1 | ||||||
|       { |           let g:syntastic_check_on_open = 1 | ||||||
|         plugin = nixpkgs.vimPlugins.nvim-dap-python; |           let g:syntastic_check_on_wq = 0''; | ||||||
|         config = builtins.replaceStrings ["%PYTHONENV%"] ["${debugPyEnv}"] (builtins.readFile ./nvim/nvim-dap-python.lua); |  | ||||||
|         type = "lua"; |  | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       nixpkgs.vimPlugins.tagbar | ||||||
|     ]; |     ]; | ||||||
|     viAlias = true; |     viAlias = true; | ||||||
|     vimAlias = true; |     vimAlias = true; | ||||||
|     vimdiffAlias = true; |     vimdiffAlias = true; | ||||||
|     withPython3 = true; |     withPython3 = true; | ||||||
|     extraPython3Packages = ps: [ |  | ||||||
|       ps.pip |  | ||||||
|       ps.pynvim |  | ||||||
|       ps.pynvim-pp |  | ||||||
|       ps.pyyaml |  | ||||||
|       ps.std2 |  | ||||||
|     ]; |  | ||||||
|     withRuby = true; |  | ||||||
|  |  | ||||||
|     extraLuaConfig = builtins.readFile ./nvim/build-utils.lua + "\n" + (builtins.replaceStrings ["_CURL_"] ["${nixpkgs.curl}/bin/curl"] (builtins.readFile ./nvim/dotnet.lua)) + "\n" + builtins.readFile ./nvim/init.lua + "\n" + builtins.readFile ./nvim/python.lua; |     extraLuaConfig = ''vim.g.python3_host_prog="${pythonEnv}/bin/python"''; | ||||||
|  |     extraConfig = builtins.readFile ./init.vim; | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|   home.packages = |   programs.direnv = { | ||||||
|     [ |     enable = true; | ||||||
|       nixpkgs.jq |     enableZshIntegration = true; | ||||||
|       nixpkgs.difftastic |     nix-direnv.enable = true; | ||||||
|       nixpkgs.syncthing |   }; | ||||||
|       nixpkgs.dockerfile-language-server |  | ||||||
|       nixpkgs.nodePackages_latest.vscode-langservers-extracted |  | ||||||
|       nixpkgs.hadolint |  | ||||||
|       nixpkgs.yaml-language-server |  | ||||||
|       nixpkgs.netcoredbg |  | ||||||
|       nixpkgs.nil |  | ||||||
|       nixpkgs.fsautocomplete |  | ||||||
|       nixpkgs.wget |  | ||||||
|       nixpkgs.yt-dlp |  | ||||||
|       nixpkgs.lldb |  | ||||||
|       nixpkgs.hledger |  | ||||||
|       nixpkgs.hledger-web |  | ||||||
|       dotnet |  | ||||||
|       nixpkgs.elan |  | ||||||
|       nixpkgs.coreutils-prefixed |  | ||||||
|       nixpkgs.asciinema |  | ||||||
|       nixpkgs.git-lfs |  | ||||||
|       nixpkgs.imagemagick |  | ||||||
|       nixpkgs.nixpkgs-fmt |  | ||||||
|       nixpkgs.age |  | ||||||
|       nixpkgs.pyright |  | ||||||
|       nixpkgs.woodpecker-agent |  | ||||||
|       nixpkgs.lynx |  | ||||||
|       nixpkgs.ffmpeg |  | ||||||
|       nixpkgs.bat |  | ||||||
|       nixpkgs.pandoc |  | ||||||
|       nixpkgs.fd |  | ||||||
|       nixpkgs.sumneko-lua-language-server |  | ||||||
|       nixpkgs.gnupg |  | ||||||
|       nixpkgs.gh |  | ||||||
|       nixpkgs.clang-tools |  | ||||||
|       nixpkgs.deno |  | ||||||
|       nixpkgs.yazi |  | ||||||
|       nixpkgs.font-awesome |  | ||||||
|       nixpkgs.gopls |  | ||||||
|       nixpkgs.go |  | ||||||
|       nixpkgs.libiconv |  | ||||||
|       nixpkgs.claude-code |  | ||||||
|     ] |  | ||||||
|     ++ ( |  | ||||||
|       if nixpkgs.stdenv.isLinux |  | ||||||
|       then [ |  | ||||||
|         nixpkgs.kdePackages.xwaylandvideobridge |  | ||||||
|         nixpkgs.protonmail-bridge |  | ||||||
|         nixpkgs.pinentry |  | ||||||
|         nixpkgs.signal-desktop |  | ||||||
|         nixpkgs.keepassxc |  | ||||||
|       ] |  | ||||||
|       else [] |  | ||||||
|     ) |  | ||||||
|     ++ ( |  | ||||||
|       if machinename == "capybara" |  | ||||||
|       then [ |  | ||||||
|         nixpkgs.steam-run |  | ||||||
|         nixpkgs.discord-canary |  | ||||||
|         nixpkgs.anki-bin |  | ||||||
|       ] |  | ||||||
|       else [] |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|  |   home.packages = [ | ||||||
|  |     nixpkgs.keepassxc | ||||||
|  |     nixpkgs.rust-analyzer | ||||||
|  |     nixpkgs.tmux | ||||||
|  |     nixpkgs.wget | ||||||
|  |     nixpkgs.yt-dlp | ||||||
|  |     nixpkgs.cmake | ||||||
|  |     nixpkgs.gnumake | ||||||
|  |     nixpkgs.gcc | ||||||
|  |     nixpkgs.lldb | ||||||
|  |     nixpkgs.hledger | ||||||
|  |     nixpkgs.hledger-web | ||||||
|  |     dotnet | ||||||
|  |     nixpkgs.jitsi-meet | ||||||
|  |     nixpkgs.ripgrep | ||||||
|  |     nixpkgs.elan | ||||||
|  |     nixpkgs.coreutils-prefixed | ||||||
|  |     nixpkgs.shellcheck | ||||||
|  |     nixpkgs.html-tidy | ||||||
|  |     nixpkgs.hugo | ||||||
|  |     nixpkgs.agda | ||||||
|  |     nixpkgs.pijul | ||||||
|  |     nixpkgs.universal-ctags | ||||||
|  |     nixpkgs.asciinema | ||||||
|  |     nixpkgs.git-lfs | ||||||
|  |     nixpkgs.imagemagick | ||||||
|  |     nixpkgs.nixpkgs-fmt | ||||||
|  |     nixpkgs.rnix-lsp | ||||||
|  |     nixpkgs.grpc-tools | ||||||
|  |     nixpkgs.element-desktop | ||||||
|  |     nixpkgs.ihp-new | ||||||
|  |     nixpkgs.direnv | ||||||
|  |     nixpkgs.lnav | ||||||
|  |     nixpkgs.age | ||||||
|  |     nixpkgs.nodejs | ||||||
|  |     nixpkgs.sqlitebrowser | ||||||
|  |     nixpkgs.typst | ||||||
|  |     nixpkgs.poetry | ||||||
|  |     nixpkgs.woodpecker-agent | ||||||
|  |     nixpkgs.alacritty | ||||||
|  |     nixpkgs.lynx | ||||||
|  |     nixpkgs.alejandra | ||||||
|  |     nixpkgs.ffmpeg | ||||||
|  |     nixpkgs.bat | ||||||
|  |     nixpkgs.pandoc | ||||||
|  |   ]; | ||||||
|  |  | ||||||
|  |   home.file.".mailcap".source = ./mailcap; | ||||||
|   home.file.".ideavimrc".source = ./ideavimrc; |   home.file.".ideavimrc".source = ./ideavimrc; | ||||||
|   home.file.".config/yt-dlp/config".source = ./youtube-dl.conf; |   home.file.".config/yt-dlp/config".source = ./youtube-dl.conf; | ||||||
|  |   home.file.".config/ripgrep/config".source = ./ripgrep.conf; | ||||||
|  |  | ||||||
|  |   programs.emacs = { | ||||||
|  |     enable = true; | ||||||
|  |     package = nixpkgs.emacs; | ||||||
|  |     extraPackages = epkgs: []; | ||||||
|  |     extraConfig = '' | ||||||
|  |       (load-file (let ((coding-system-for-read 'utf-8)) | ||||||
|  |                  (shell-command-to-string "agda-mode locate"))) | ||||||
|  |     ''; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   home.file.".cargo/config.toml".source = ./cargo-config.toml; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										357
									
								
								home-manager/init.vim
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										357
									
								
								home-manager/init.vim
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,357 @@ | |||||||
|  | set nu | ||||||
|  | colorscheme molokai | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Maintainer:  | ||||||
|  | "       Amir Salihefendic — @amix3k | ||||||
|  | " | ||||||
|  | " Awesome_version: | ||||||
|  | "       Get this config, nice color schemes and lots of plugins! | ||||||
|  | " | ||||||
|  | "       Install the awesome version from: | ||||||
|  | " | ||||||
|  | "           https://github.com/amix/vimrc | ||||||
|  | " | ||||||
|  | " Sections: | ||||||
|  | "    -> General | ||||||
|  | "    -> VIM user interface | ||||||
|  | "    -> Colors and Fonts | ||||||
|  | "    -> Files and backups | ||||||
|  | "    -> Text, tab and indent related | ||||||
|  | "    -> Visual mode related | ||||||
|  | "    -> Moving around, tabs and buffers | ||||||
|  | "    -> Status line | ||||||
|  | "    -> Editing mappings | ||||||
|  | "    -> vimgrep searching and cope displaying | ||||||
|  | "    -> Spell checking | ||||||
|  | "    -> Misc | ||||||
|  | "    -> Helper functions | ||||||
|  | " | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => General | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Sets how many lines of history VIM has to remember | ||||||
|  | set history=500 | ||||||
|  |  | ||||||
|  | " Enable filetype plugins | ||||||
|  | filetype plugin on | ||||||
|  | filetype indent on | ||||||
|  |  | ||||||
|  | " Set to auto read when a file is changed from the outside | ||||||
|  | set autoread | ||||||
|  |  | ||||||
|  | " With a map leader it's possible to do extra key combinations | ||||||
|  | " like <leader>w saves the current file | ||||||
|  | let mapleader = "`" | ||||||
|  |  | ||||||
|  | " :W sudo saves the file  | ||||||
|  | " (useful for handling the permission-denied error) | ||||||
|  | command W w !sudo tee % > /dev/null | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => VIM user interface | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Set 7 lines to the cursor - when moving vertically using j/k | ||||||
|  | set so=7 | ||||||
|  |  | ||||||
|  | " Avoid garbled characters in Chinese language windows OS | ||||||
|  | let $LANG='en'  | ||||||
|  | set langmenu=en | ||||||
|  | source $VIMRUNTIME/delmenu.vim | ||||||
|  | source $VIMRUNTIME/menu.vim | ||||||
|  |  | ||||||
|  | " Turn on the Wild menu | ||||||
|  | set wildmenu | ||||||
|  |  | ||||||
|  | " Ignore compiled files | ||||||
|  | set wildignore=*.o,*~,*.pyc | ||||||
|  | if has("win16") || has("win32") | ||||||
|  |     set wildignore+=.git\*,.hg\*,.svn\* | ||||||
|  | else | ||||||
|  |     set wildignore+=*/.git/*,*/.hg/*,*/.svn/*,*/.DS_Store | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | "Always show current position | ||||||
|  | set ruler | ||||||
|  |  | ||||||
|  | " Height of the command bar | ||||||
|  | set cmdheight=2 | ||||||
|  |  | ||||||
|  | " A buffer becomes hidden when it is abandoned | ||||||
|  | set hid | ||||||
|  |  | ||||||
|  | " Ignore case when searching | ||||||
|  | set ignorecase | ||||||
|  |  | ||||||
|  | " When searching try to be smart about cases  | ||||||
|  | set smartcase | ||||||
|  |  | ||||||
|  | " Highlight search results | ||||||
|  | set hlsearch | ||||||
|  |  | ||||||
|  | " Makes search act like search in modern browsers | ||||||
|  | set incsearch  | ||||||
|  |  | ||||||
|  | " Don't redraw while executing macros (good performance config) | ||||||
|  | set lazyredraw  | ||||||
|  |  | ||||||
|  | " For regular expressions turn magic on | ||||||
|  | set magic | ||||||
|  |  | ||||||
|  | " Show matching brackets when text indicator is over them | ||||||
|  | set showmatch  | ||||||
|  | " How many tenths of a second to blink when matching brackets | ||||||
|  | set mat=2 | ||||||
|  |  | ||||||
|  | " No annoying sound on errors | ||||||
|  | set noerrorbells | ||||||
|  | set novisualbell | ||||||
|  | set t_vb= | ||||||
|  | set tm=500 | ||||||
|  |  | ||||||
|  | " Properly disable sound on errors on MacVim | ||||||
|  | if has("gui_macvim") | ||||||
|  |     autocmd GUIEnter * set vb t_vb= | ||||||
|  | endif | ||||||
|  |  | ||||||
|  |  | ||||||
|  | " Add a bit extra margin to the left | ||||||
|  | set foldcolumn=1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Colors and Fonts | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Enable syntax highlighting | ||||||
|  | syntax enable  | ||||||
|  |  | ||||||
|  | " Enable 256 colors palette in Gnome Terminal | ||||||
|  | if $COLORTERM == 'gnome-terminal' | ||||||
|  |     set t_Co=256 | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | set background=dark | ||||||
|  |  | ||||||
|  | " Set extra options when running in GUI mode | ||||||
|  | if has("gui_running") | ||||||
|  |     set guioptions-=T | ||||||
|  |     set guioptions-=e | ||||||
|  |     set t_Co=256 | ||||||
|  |     set guitablabel=%M\ %t | ||||||
|  | endif | ||||||
|  |  | ||||||
|  | " Set utf8 as standard encoding and en_US as the standard language | ||||||
|  | set encoding=utf8 | ||||||
|  |  | ||||||
|  | " Use Unix as the standard file type | ||||||
|  | set ffs=unix,mac | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Files, backups and undo | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Turn backup off, since most stuff is in SVN, git et.c anyway... | ||||||
|  | set nobackup | ||||||
|  | set nowb | ||||||
|  | set noswapfile | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Text, tab and indent related | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Use spaces instead of tabs | ||||||
|  | set expandtab | ||||||
|  |  | ||||||
|  | " Be smart when using tabs ;) | ||||||
|  | set smarttab | ||||||
|  |  | ||||||
|  | " 1 tab == 4 spaces | ||||||
|  | set shiftwidth=4 | ||||||
|  | set tabstop=4 | ||||||
|  |  | ||||||
|  | " Linebreak on 500 characters | ||||||
|  | set lbr | ||||||
|  | set tw=500 | ||||||
|  |  | ||||||
|  | set ai "Auto indent | ||||||
|  | set si "Smart indent | ||||||
|  | set wrap "Wrap lines | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Moving around, tabs, windows and buffers | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Map <Space> to / (search) and Ctrl-<Space> to ? (backwards search) | ||||||
|  | map <space> / | ||||||
|  | map <c-space> ? | ||||||
|  |  | ||||||
|  | " Disable highlight when <leader><cr> is pressed | ||||||
|  | map <silent> <leader><cr> :noh<cr> | ||||||
|  |  | ||||||
|  | " Smart way to move between windows | ||||||
|  | map <C-j> <C-W>j | ||||||
|  | map <C-k> <C-W>k | ||||||
|  | map <C-h> <C-W>h | ||||||
|  | map <C-l> <C-W>l | ||||||
|  |  | ||||||
|  | " Close the current buffer | ||||||
|  | map <leader>bd :Bclose<cr>:tabclose<cr>gT | ||||||
|  |  | ||||||
|  | " Close all the buffers | ||||||
|  | map <leader>ba :bufdo bd<cr> | ||||||
|  |  | ||||||
|  | map <leader>l :bnext<cr> | ||||||
|  | map <leader>h :bprevious<cr> | ||||||
|  |  | ||||||
|  | " Useful mappings for managing tabs | ||||||
|  | map <leader>tn :tabnew<cr> | ||||||
|  | map <leader>to :tabonly<cr> | ||||||
|  | map <leader>tc :tabclose<cr> | ||||||
|  | map <leader>tm :tabmove  | ||||||
|  | map <leader>t<leader> :tabnext  | ||||||
|  |  | ||||||
|  | " Let 'tl' toggle between this and the last accessed tab | ||||||
|  | let g:lasttab = 1 | ||||||
|  | nmap <Leader>tl :exe "tabn ".g:lasttab<CR> | ||||||
|  | au TabLeave * let g:lasttab = tabpagenr() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | " Opens a new tab with the current buffer's path | ||||||
|  | " Super useful when editing files in the same directory | ||||||
|  | map <leader>te :tabedit <c-r>=expand("%:p:h")<cr>/ | ||||||
|  |  | ||||||
|  | " Switch CWD to the directory of the open buffer | ||||||
|  | map <leader>cd :cd %:p:h<cr>:pwd<cr> | ||||||
|  |  | ||||||
|  | " Specify the behavior when switching between buffers  | ||||||
|  | try | ||||||
|  |   set switchbuf=useopen,usetab,newtab | ||||||
|  |   set stal=2 | ||||||
|  | catch | ||||||
|  | endtry | ||||||
|  |  | ||||||
|  | " Return to last edit position when opening files (You want this!) | ||||||
|  | au BufReadPost * if line("'\"") > 1 && line("'\"") <= line("$") | exe "normal! g'\"" | endif | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """""""""""""""""""""""""""""" | ||||||
|  | " => Status line | ||||||
|  | """""""""""""""""""""""""""""" | ||||||
|  | " Always show the status line | ||||||
|  | set laststatus=2 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Editing mappings | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  |  | ||||||
|  | " Delete trailing white space on save, useful for some filetypes ;) | ||||||
|  | fun! CleanExtraSpaces() | ||||||
|  |     let save_cursor = getpos(".") | ||||||
|  |     let old_query = getreg('/') | ||||||
|  |     silent! %s/\s\+$//e | ||||||
|  |     call setpos('.', save_cursor) | ||||||
|  |     call setreg('/', old_query) | ||||||
|  | endfun | ||||||
|  |  | ||||||
|  | if has("autocmd") | ||||||
|  |     autocmd BufWritePre *.fs,*.fsi,*.txt,*.js,*.py,*.wiki,*.sh,*.coffee :call CleanExtraSpaces() | ||||||
|  | endif | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Spell checking | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Pressing ,ss will toggle and untoggle spell checking | ||||||
|  | map <leader>ss :setlocal spell!<cr> | ||||||
|  |  | ||||||
|  | " Shortcuts using <leader> | ||||||
|  | map <leader>sn ]s | ||||||
|  | map <leader>sp [s | ||||||
|  | map <leader>sa zg | ||||||
|  | map <leader>s? z= | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Misc | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Remove the Windows ^M - when the encodings gets messed up | ||||||
|  | noremap <Leader>m mmHmt:%s/<C-V><cr>//ge<cr>'tzt'm | ||||||
|  |  | ||||||
|  | " Quickly open a buffer for scribble | ||||||
|  | map <leader>q :e ~/buffer<cr> | ||||||
|  |  | ||||||
|  | " Quickly open a markdown buffer for scribble | ||||||
|  | map <leader>x :e ~/buffer.md<cr> | ||||||
|  |  | ||||||
|  | " Toggle paste mode on and off | ||||||
|  | map <leader>pp :setlocal paste!<cr> | ||||||
|  |  | ||||||
|  |  | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " => Helper functions | ||||||
|  | """"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""" | ||||||
|  | " Returns true if paste mode is enabled | ||||||
|  | function! HasPaste() | ||||||
|  |     if &paste | ||||||
|  |         return 'PASTE MODE  ' | ||||||
|  |     endif | ||||||
|  |     return '' | ||||||
|  | endfunction | ||||||
|  |  | ||||||
|  | " Don't close window, when deleting a buffer | ||||||
|  | command! Bclose call <SID>BufcloseCloseIt() | ||||||
|  | function! <SID>BufcloseCloseIt() | ||||||
|  |     let l:currentBufNum = bufnr("%") | ||||||
|  |     let l:alternateBufNum = bufnr("#") | ||||||
|  |  | ||||||
|  |     if buflisted(l:alternateBufNum) | ||||||
|  |         buffer # | ||||||
|  |     else | ||||||
|  |         bnext | ||||||
|  |     endif | ||||||
|  |  | ||||||
|  |     if bufnr("%") == l:currentBufNum | ||||||
|  |         new | ||||||
|  |     endif | ||||||
|  |  | ||||||
|  |     if buflisted(l:currentBufNum) | ||||||
|  |         execute("bdelete! ".l:currentBufNum) | ||||||
|  |     endif | ||||||
|  | endfunction | ||||||
|  |  | ||||||
|  | function! CmdLine(str) | ||||||
|  |     call feedkeys(":" . a:str) | ||||||
|  | endfunction  | ||||||
|  |  | ||||||
|  | function! VisualSelection(direction, extra_filter) range | ||||||
|  |     let l:saved_reg = @" | ||||||
|  |     execute "normal! vgvy" | ||||||
|  |  | ||||||
|  |     let l:pattern = escape(@", "\\/.*'$^~[]") | ||||||
|  |     let l:pattern = substitute(l:pattern, "\n$", "", "") | ||||||
|  |  | ||||||
|  |     if a:direction == 'gv' | ||||||
|  |         call CmdLine("Ack '" . l:pattern . "' " ) | ||||||
|  |     elseif a:direction == 'replace' | ||||||
|  |         call CmdLine("%s" . '/'. l:pattern . '/') | ||||||
|  |     endif | ||||||
|  |  | ||||||
|  |     let @/ = l:pattern | ||||||
|  |     let @" = l:saved_reg | ||||||
|  | endfunction | ||||||
|  |  | ||||||
|  | nnoremap <leader>c :!cargo clippy | ||||||
|  | nnoremap <leader>j :%!python -m json.tool | ||||||
|  |  | ||||||
|  | set statusline+=%#warningmsg# | ||||||
|  | set statusline+=%{SyntasticStatuslineFlag()} | ||||||
|  | set statusline+=%* | ||||||
|  | " Format the status line | ||||||
|  | set statusline=\ %{HasPaste()}%F%m%r%h\ %w\ \ CWD:\ %r%{getcwd()}%h\ \ \ Line:\ %l\ \ Column:\ %c | ||||||
|  |  | ||||||
|  | set fileformat=unix | ||||||
| @@ -1,204 +0,0 @@ | |||||||
| {nixpkgs, ...}: { |  | ||||||
|   home.packages = [nixpkgs.firefox-wayland nixpkgs.jetbrains.rider]; |  | ||||||
|   nixpkgs.config.firefox.speechSynthesisSupport = true; |  | ||||||
|  |  | ||||||
|   wayland.windowManager.sway = { |  | ||||||
|     enable = true; |  | ||||||
|     config = { |  | ||||||
|       focus = {followMouse = false;}; |  | ||||||
|       modifier = "Mod4"; |  | ||||||
|       terminal = "alacritty"; |  | ||||||
|       window = {border = 5;}; |  | ||||||
|       bars = [ |  | ||||||
|         {command = "${nixpkgs.waybar}/bin/waybar";} |  | ||||||
|       ]; |  | ||||||
|     }; |  | ||||||
|     extraConfig = builtins.replaceStrings ["@@WL-COPY@@" "@@GRIM@@" "@@SLURP@@"] ["${nixpkgs.wl-clipboard}/bin/wl-copy" "${nixpkgs.grim}/bin/grim" "${nixpkgs.slurp}/bin/slurp"] (builtins.readFile ./sway.conf); |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   programs.waybar = { |  | ||||||
|     enable = true; |  | ||||||
|     settings = { |  | ||||||
|       "bar-0" = { |  | ||||||
|         position = "bottom"; |  | ||||||
|         layer = "top"; |  | ||||||
|         height = 34; |  | ||||||
|         spacing = 8; |  | ||||||
|         modules-left = ["sway/workspaces" "sway/mode" "sway/scratchpad" "custom/media"]; |  | ||||||
|         modules-center = ["sway/window"]; |  | ||||||
|         modules-right = ["mpd" "idle_inhibitor" "pulseaudio" "network" "power-profiles-daemon" "cpu" "memory" "temperature" "backlight" "keyboard-state" "sway/language" "battery" "battery#bat2" "clock" "tray" "custom/power"]; |  | ||||||
|  |  | ||||||
|         "keyboard-state" = { |  | ||||||
|           "numlock" = true; |  | ||||||
|           "capslock" = true; |  | ||||||
|           "format" = "{name} {icon}"; |  | ||||||
|           "format-icons" = { |  | ||||||
|             "locked" = ""; |  | ||||||
|             "unlocked" = ""; |  | ||||||
|           }; |  | ||||||
|         }; |  | ||||||
|         "sway/mode" = { |  | ||||||
|           "format" = "<span style=\"italic\">{}</span>"; |  | ||||||
|         }; |  | ||||||
|         "sway/scratchpad" = { |  | ||||||
|           "format" = "{icon} {count}"; |  | ||||||
|           "show-empty" = false; |  | ||||||
|           "format-icons" = ["" ""]; |  | ||||||
|           "tooltip" = true; |  | ||||||
|           "tooltip-format" = "{app}: {title}"; |  | ||||||
|         }; |  | ||||||
|         "mpd" = { |  | ||||||
|           "format" = "{stateIcon} {consumeIcon}{randomIcon}{repeatIcon}{singleIcon}{artist} - {album} - {title} ({elapsedTime:%M:%S}/{totalTime:%M:%S}) ⸨{songPosition}|{queueLength}⸩ {volume}% "; |  | ||||||
|           "format-disconnected" = "Disconnected "; |  | ||||||
|           "format-stopped" = "{consumeIcon}{randomIcon}{repeatIcon}{singleIcon}Stopped "; |  | ||||||
|           "unknown-tag" = "N/A"; |  | ||||||
|           "interval" = 5; |  | ||||||
|           "consume-icons" = { |  | ||||||
|             "on" = " "; |  | ||||||
|           }; |  | ||||||
|           "random-icons" = { |  | ||||||
|             "off" = "<span color=\"#f53c3c\"></span> "; |  | ||||||
|             "on" = " "; |  | ||||||
|           }; |  | ||||||
|           "repeat-icons" = { |  | ||||||
|             "on" = " "; |  | ||||||
|           }; |  | ||||||
|           "single-icons" = { |  | ||||||
|             "on" = "1 "; |  | ||||||
|           }; |  | ||||||
|           "state-icons" = { |  | ||||||
|             "paused" = ""; |  | ||||||
|             "playing" = ""; |  | ||||||
|           }; |  | ||||||
|           "tooltip-format" = "MPD (connected)"; |  | ||||||
|           "tooltip-format-disconnected" = "MPD (disconnected)"; |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         "idle_inhibitor" = { |  | ||||||
|           "format" = "{icon}"; |  | ||||||
|           "format-icons" = { |  | ||||||
|             "activated" = ""; |  | ||||||
|             "deactivated" = ""; |  | ||||||
|           }; |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         "tray" = { |  | ||||||
|           "spacing" = 20; |  | ||||||
|         }; |  | ||||||
|         "clock" = { |  | ||||||
|           "tooltip-format" = "<big>{:%Y %B}</big>\n<tt><small>{calendar}</small></tt>"; |  | ||||||
|           "format" = "{:%Y-%m-%d %H:%M:%S}"; |  | ||||||
|           "interval" = 1; |  | ||||||
|         }; |  | ||||||
|         "cpu" = { |  | ||||||
|           "format" = "{usage}% "; |  | ||||||
|           "tooltip" = false; |  | ||||||
|         }; |  | ||||||
|         "memory" = { |  | ||||||
|           "format" = "{}% "; |  | ||||||
|         }; |  | ||||||
|         "temperature" = { |  | ||||||
|           "critical-threshold" = 80; |  | ||||||
|           "format" = "{temperatureC}°C {icon}"; |  | ||||||
|           "format-icons" = ["" "" ""]; |  | ||||||
|         }; |  | ||||||
|         "backlight" = { |  | ||||||
|           "format" = "{percent}% {icon}"; |  | ||||||
|           "format-icons" = ["" "" "" "" "" "" "" "" ""]; |  | ||||||
|         }; |  | ||||||
|         "battery" = { |  | ||||||
|           "states" = { |  | ||||||
|             "warning" = 30; |  | ||||||
|             "critical" = 15; |  | ||||||
|           }; |  | ||||||
|           "format" = "{capacity}% {icon}"; |  | ||||||
|           "format-full" = "{capacity}% {icon}"; |  | ||||||
|           "format-charging" = "{capacity}% "; |  | ||||||
|           "format-plugged" = "{capacity}% "; |  | ||||||
|           "format-alt" = "{time} {icon}"; |  | ||||||
|           "format-icons" = ["" "" "" "" ""]; |  | ||||||
|         }; |  | ||||||
|         "battery#bat2" = { |  | ||||||
|           "bat" = "BAT2"; |  | ||||||
|         }; |  | ||||||
|         "power-profiles-daemon" = { |  | ||||||
|           "format" = "{icon}"; |  | ||||||
|           "tooltip-format" = "Power profile: {profile}\nDriver: {driver}"; |  | ||||||
|           "tooltip" = true; |  | ||||||
|           "format-icons" = { |  | ||||||
|             "default" = ""; |  | ||||||
|             "performance" = ""; |  | ||||||
|             "balanced" = ""; |  | ||||||
|             "power-saver" = ""; |  | ||||||
|           }; |  | ||||||
|         }; |  | ||||||
|         "network" = { |  | ||||||
|           "format-wifi" = "{essid} ({signalStrength}%) "; |  | ||||||
|           "format-ethernet" = "{bandwidthDownBytes}/{bandwidthUpBytes} "; |  | ||||||
|           "interval" = 5; |  | ||||||
|           "tooltip-format" = "{ifname} via {gwaddr} "; |  | ||||||
|           "format-linked" = "{ifname} (No IP) "; |  | ||||||
|           "format-disconnected" = "Disconnected ⚠"; |  | ||||||
|           "format-alt" = "{ifname}: {ipaddr}/{cidr}"; |  | ||||||
|         }; |  | ||||||
|         "pulseaudio" = { |  | ||||||
|           "format" = "{volume}% {icon} {format_source}"; |  | ||||||
|           "format-bluetooth" = "{volume}% {icon} {format_source}"; |  | ||||||
|           "format-bluetooth-muted" = " {icon} {format_source}"; |  | ||||||
|           "format-muted" = " {format_source}"; |  | ||||||
|           "format-source" = "{volume}% "; |  | ||||||
|           "format-source-muted" = ""; |  | ||||||
|           "format-icons" = { |  | ||||||
|             "headphone" = ""; |  | ||||||
|             "hands-free" = ""; |  | ||||||
|             "headset" = ""; |  | ||||||
|             "phone" = ""; |  | ||||||
|             "portable" = ""; |  | ||||||
|             "car" = ""; |  | ||||||
|             "default" = ["" "" ""]; |  | ||||||
|           }; |  | ||||||
|           "on-click" = "${nixpkgs.pavucontrol}/bin/pavucontrol"; |  | ||||||
|         }; |  | ||||||
|         "custom/media" = { |  | ||||||
|           "format" = "{icon} {text}"; |  | ||||||
|           "return-type" = "json"; |  | ||||||
|           "max-length" = 40; |  | ||||||
|           "format-icons" = { |  | ||||||
|             "spotify" = ""; |  | ||||||
|             "default" = "🎜"; |  | ||||||
|           }; |  | ||||||
|           "escape" = true; |  | ||||||
|           "exec" = let |  | ||||||
|             python = nixpkgs.python312.withPackages (ppkgs: [ppkgs.pygobject3]); |  | ||||||
|           in "${python}/bin/python ${./modules/waybar/mediaplayer.py} 2> /dev/null"; |  | ||||||
|         }; |  | ||||||
|         "custom/power" = { |  | ||||||
|           "format" = "⏻ "; |  | ||||||
|           "tooltip" = false; |  | ||||||
|           "menu" = "on-click"; |  | ||||||
|           "menu-file" = ./modules/waybar/power_menu.xml; |  | ||||||
|           "menu-actions" = { |  | ||||||
|             "shutdown" = "shutdown now"; |  | ||||||
|             "reboot" = "reboot"; |  | ||||||
|             "suspend" = "systemctl suspend"; |  | ||||||
|             "hibernate" = "systemctl hibernate"; |  | ||||||
|           }; |  | ||||||
|         }; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|  |  | ||||||
|     style = '' |  | ||||||
|       * { |  | ||||||
|         font-family: FontAwesome, Roboto, Helvetica, Arial, sans-serif; |  | ||||||
|         font-size: 13px; |  | ||||||
|       } |  | ||||||
|     ''; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   services.gpg-agent = { |  | ||||||
|     enable = nixpkgs.stdenv.isLinux; |  | ||||||
|     pinentry.package = nixpkgs.pinentry-curses; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   services.swayidle = {enable = true;}; |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   imports = [./emacs.nix]; |  | ||||||
|  |  | ||||||
|   home.packages = [ |  | ||||||
|     pkgs.agda |  | ||||||
|   ]; |  | ||||||
| } |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   home.packages = [ |  | ||||||
|     pkgs.direnv |  | ||||||
|   ]; |  | ||||||
|   programs.direnv = { |  | ||||||
|     enable = true; |  | ||||||
|     enableZshIntegration = true; |  | ||||||
|     nix-direnv.enable = true; |  | ||||||
|     # stdlib = builtins.readFile ../direnv/envrc; |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   programs.emacs = { |  | ||||||
|     enable = true; |  | ||||||
|     package = pkgs.emacs; |  | ||||||
|     extraPackages = epkgs: [epkgs.evil]; |  | ||||||
|     extraConfig = '' |  | ||||||
|       (load-file (let ((coding-system-for-read 'utf-8)) |  | ||||||
|                  (shell-command-to-string "agda-mode locate"))) |  | ||||||
|       (require 'evil) |  | ||||||
|       (evil-mode 1) |  | ||||||
|       (evil-set-undo-system 'undo-redo) |  | ||||||
|     ''; |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   programs.ghostty = { |  | ||||||
|     enable = pkgs.stdenv.isLinux; |  | ||||||
|     enableZshIntegration = true; |  | ||||||
|     settings = { |  | ||||||
|       keybind = [ |  | ||||||
|         "shift+enter=text:\\n" |  | ||||||
|       ]; |  | ||||||
|     }; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   home.packages = |  | ||||||
|     if pkgs.stdenv.isLinux |  | ||||||
|     then [ |  | ||||||
|       pkgs.ghostty |  | ||||||
|     ] |  | ||||||
|     else []; |  | ||||||
| } |  | ||||||
| @@ -1,183 +0,0 @@ | |||||||
| { |  | ||||||
|   pkgs, |  | ||||||
|   mbsync, |  | ||||||
|   secretsPath, |  | ||||||
|   ... |  | ||||||
| }: let |  | ||||||
|   deobfuscate = str: let |  | ||||||
|     lib = pkgs.lib; |  | ||||||
|     base64Table = |  | ||||||
|       builtins.listToAttrs |  | ||||||
|       (lib.imap0 (i: c: lib.nameValuePair c i) |  | ||||||
|         (lib.stringToCharacters "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/")); |  | ||||||
|  |  | ||||||
|     # Generated using python3: |  | ||||||
|     # print(''.join([ chr(n) for n in range(1, 256) ]), file=open('ascii', 'w')) |  | ||||||
|     ascii = builtins.readFile ./mail/ascii; |  | ||||||
|  |  | ||||||
|     # List of base-64 numbers |  | ||||||
|     numbers64 = map (c: base64Table.${c}) (lib.lists.reverseList (lib.stringToCharacters str)); |  | ||||||
|  |  | ||||||
|     # List of base-256 numbers |  | ||||||
|     numbers256 = lib.concatLists (lib.genList ( |  | ||||||
|       i: let |  | ||||||
|         v = |  | ||||||
|           lib.foldl' |  | ||||||
|           (acc: el: acc * 64 + el) |  | ||||||
|           0 |  | ||||||
|           (lib.sublist (i * 4) 4 numbers64); |  | ||||||
|       in [ |  | ||||||
|         (lib.mod (v / 256 / 256) 256) |  | ||||||
|         (lib.mod (v / 256) 256) |  | ||||||
|         (lib.mod v 256) |  | ||||||
|       ] |  | ||||||
|     ) (lib.length numbers64 / 4)); |  | ||||||
|   in |  | ||||||
|     # Converts base-256 numbers to ascii |  | ||||||
|     lib.concatMapStrings ( |  | ||||||
|       n: |  | ||||||
|       # Can't represent the null byte in Nix.. |  | ||||||
|       let |  | ||||||
|         result = lib.substring (n - 1) 1 ascii; |  | ||||||
|       in |  | ||||||
|         if result == " " |  | ||||||
|         then "" |  | ||||||
|         else result |  | ||||||
|     ) |  | ||||||
|     numbers256; |  | ||||||
| in { |  | ||||||
|   accounts.email.accounts."Gmail" = let |  | ||||||
|     address = (deobfuscate "AFTN0cWdh12c") + "gmail.com"; |  | ||||||
|   in { |  | ||||||
|     notmuch.enable = false; |  | ||||||
|     neomutt = { |  | ||||||
|       enable = true; |  | ||||||
|     }; |  | ||||||
|     address = address; |  | ||||||
|     flavor = "gmail.com"; |  | ||||||
|     mbsync = { |  | ||||||
|       enable = true; |  | ||||||
|       create = "maildir"; |  | ||||||
|       extraConfig.account = { |  | ||||||
|         AuthMechs = "XOAUTH2"; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|     userName = address; |  | ||||||
|     # This is accompanied by a developer application at Google: |  | ||||||
|     # https://console.cloud.google.com/apis/credentials |  | ||||||
|     # Create an OAuth 2.0 Client ID with type `Desktop`. |  | ||||||
|     # The Google application needs the https://mail.google.com scope; mine has |  | ||||||
|     # an authorized domain `google.com` but I don't know if that's required. |  | ||||||
|     # Enter the client ID and client secret into a two-line text file |  | ||||||
|     # named gmail-client-app.txt immediately next to the intended destination |  | ||||||
|     # secret file (the arg to mutt-oauth2.py in the invocation): |  | ||||||
|     # so here it would be /path/to/gmail-client-app.txt . |  | ||||||
|     # Run `./mail/mutt-oauth2.py /path/to/secret --authorize --verbose` once manually, |  | ||||||
|     # and that will populate /path/to/secret. |  | ||||||
|     # I've left it unencrypted here; the original uses GPG to store it encrypted at rest. |  | ||||||
|     passwordCommand = ''${pkgs.python3}/bin/python ${./mail/mutt-oauth2.py} ${secretsPath}/gmail.txt 2>/tmp/gmail-passcmd.2.txt''; |  | ||||||
|     realName = "Patrick Stevens"; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   accounts.email.accounts."BTInternet" = let |  | ||||||
|     address = (deobfuscate "z5WZ2VGdz5yajlmc0FGc") + "@btinternet.com"; |  | ||||||
|   in { |  | ||||||
|     notmuch.enable = false; |  | ||||||
|     neomutt = { |  | ||||||
|       enable = true; |  | ||||||
|     }; |  | ||||||
|     address = address; |  | ||||||
|     imap = { |  | ||||||
|       host = "mail.btinternet.com"; |  | ||||||
|       port = 993; |  | ||||||
|       tls = { |  | ||||||
|         enable = true; |  | ||||||
|         useStartTls = false; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|     mbsync = { |  | ||||||
|       enable = true; |  | ||||||
|       create = "maildir"; |  | ||||||
|     }; |  | ||||||
|     realName = "Patrick Stevens"; |  | ||||||
|     passwordCommand = "${pkgs.coreutils}/bin/cat ${secretsPath}/btinternet.txt"; |  | ||||||
|     smtp = { |  | ||||||
|       host = "mail.btinternet.com"; |  | ||||||
|       port = 465; |  | ||||||
|       tls = { |  | ||||||
|         enable = true; |  | ||||||
|         useStartTls = false; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|     userName = address; |  | ||||||
|     primary = true; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   accounts.email.accounts."Proton" = let |  | ||||||
|     address = deobfuscate "gAya15ybj5ycuVmdlR3crNWayRXYwB0ajlmc0FGc"; |  | ||||||
|   in { |  | ||||||
|     # notmuch.enable = true; |  | ||||||
|     neomutt = { |  | ||||||
|       enable = true; |  | ||||||
|     }; |  | ||||||
|     address = address; |  | ||||||
|     # I use the ProtonMail bridge, which sits at localhost. |  | ||||||
|     imap = { |  | ||||||
|       host = "127.0.0.1"; |  | ||||||
|       port = 1143; # 8125; if using hydroxide |  | ||||||
|       tls = { |  | ||||||
|         enable = false; |  | ||||||
|         useStartTls = true; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|     mbsync = { |  | ||||||
|       enable = true; |  | ||||||
|       create = "maildir"; |  | ||||||
|       extraConfig.account = { |  | ||||||
|         # Because ProtonMail Bridge is localhost, we don't |  | ||||||
|         # care that we can only auth to it in plain text. |  | ||||||
|         AuthMechs = "LOGIN"; |  | ||||||
|       }; |  | ||||||
|     }; |  | ||||||
|     realName = "Patrick Stevens"; |  | ||||||
|     passwordCommand = |  | ||||||
|       # I store the ProtonMail Bridge password here. |  | ||||||
|       # Extracting it from a keychain would be better. |  | ||||||
|       "${pkgs.coreutils}/bin/cat ${secretsPath}/proton.txt"; |  | ||||||
|     smtp = { |  | ||||||
|       host = "127.0.0.1"; |  | ||||||
|       port = 1025; # 8126; if using hydroxide |  | ||||||
|       tls = {enable = false;}; |  | ||||||
|     }; |  | ||||||
|     userName = |  | ||||||
|       if pkgs.stdenv.isLinux |  | ||||||
|       then builtins.head (pkgs.lib.strings.splitString "@" address) |  | ||||||
|       else address; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   programs.mbsync = { |  | ||||||
|     enable = true; |  | ||||||
|     extraConfig = '' |  | ||||||
|       CopyArrivalDate yes |  | ||||||
|     ''; |  | ||||||
|     package = mbsync; |  | ||||||
|   }; |  | ||||||
|   programs.neomutt = { |  | ||||||
|     enable = true; |  | ||||||
|     package = pkgs.neomutt.override {withNotmuch = false;}; |  | ||||||
|     extraConfig = '' |  | ||||||
|       set use_threads=threads sort=last-date sort_aux=date |  | ||||||
|     ''; |  | ||||||
|     sidebar.enable = true; |  | ||||||
|     vimKeys = true; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   programs.notmuch.enable = false; |  | ||||||
|  |  | ||||||
|   home.file.".mailcap".source = ./mail/mailcap; |  | ||||||
|  |  | ||||||
|   home.packages = [ |  | ||||||
|     # pkgs.notmuch |  | ||||||
|     pkgs.lynx |  | ||||||
|   ]; |  | ||||||
| } |  | ||||||
| @@ -1,2 +0,0 @@ | |||||||
| 	 |  | ||||||
|  |  | ||||||
| @@ -1,402 +0,0 @@ | |||||||
| #!/usr/bin/env python3 |  | ||||||
| # |  | ||||||
| # Mutt OAuth2 token management script, version 2020-08-07 |  | ||||||
| # Written against python 3.7.3, not tried with earlier python versions. |  | ||||||
| # |  | ||||||
| #   Copyright (C) 2020 Alexander Perlis |  | ||||||
| # |  | ||||||
| #   This program is free software; you can redistribute it and/or |  | ||||||
| #   modify it under the terms of the GNU General Public License as |  | ||||||
| #   published by the Free Software Foundation; either version 2 of the |  | ||||||
| #   License, or (at your option) any later version. |  | ||||||
| # |  | ||||||
| #   This program is distributed in the hope that it will be useful, |  | ||||||
| #   but WITHOUT ANY WARRANTY; without even the implied warranty of |  | ||||||
| #   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU |  | ||||||
| #   General Public License for more details. |  | ||||||
| # |  | ||||||
| #   You should have received a copy of the GNU General Public License |  | ||||||
| #   along with this program; if not, write to the Free Software |  | ||||||
| #   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA |  | ||||||
| #   02110-1301, USA. |  | ||||||
|  |  | ||||||
| # Subsequently adapted by Patrick Stevens, who hacked it up to read gmail |  | ||||||
| # client app configuration from a file called gmail-client-app.txt that |  | ||||||
| # lives next to the secret file. |  | ||||||
|  |  | ||||||
| '''Mutt OAuth2 token management''' |  | ||||||
|  |  | ||||||
| import sys |  | ||||||
| import json |  | ||||||
| import argparse |  | ||||||
| import urllib.error |  | ||||||
| import urllib.parse |  | ||||||
| import urllib.request |  | ||||||
| import imaplib |  | ||||||
| import poplib |  | ||||||
| import smtplib |  | ||||||
| import base64 |  | ||||||
| import secrets |  | ||||||
| import hashlib |  | ||||||
| import time |  | ||||||
| from datetime import timedelta, datetime |  | ||||||
| from pathlib import Path |  | ||||||
| import socket |  | ||||||
| import http.server |  | ||||||
| import subprocess |  | ||||||
|  |  | ||||||
| ap = argparse.ArgumentParser(epilog=''' |  | ||||||
| This script obtains and prints a valid OAuth2 access token.  State is maintained in an |  | ||||||
| encrypted TOKENFILE.  Run with "--verbose --authorize" to get started or whenever all |  | ||||||
| tokens have expired, optionally with "--authflow" to override the default authorization |  | ||||||
| flow.  To truly start over from scratch, first delete TOKENFILE.  Use "--verbose --test" |  | ||||||
| to test the IMAP/POP/SMTP endpoints. |  | ||||||
| ''') |  | ||||||
| ap.add_argument('-v', '--verbose', action='store_true', help='increase verbosity') |  | ||||||
| ap.add_argument('-d', '--debug', action='store_true', help='enable debug output') |  | ||||||
| ap.add_argument('tokenfile', help='persistent token storage') |  | ||||||
| ap.add_argument('-a', '--authorize', action='store_true', help='manually authorize new tokens') |  | ||||||
| ap.add_argument('--authflow', help='authcode | localhostauthcode | devicecode') |  | ||||||
| ap.add_argument('-t', '--test', action='store_true', help='test IMAP/POP/SMTP endpoints') |  | ||||||
| args = ap.parse_args() |  | ||||||
|  |  | ||||||
| token = {} |  | ||||||
| path = Path(args.tokenfile) |  | ||||||
| if path.exists(): |  | ||||||
|     if 0o777 & path.stat().st_mode != 0o600: |  | ||||||
|         sys.exit('Token file has unsafe mode. Suggest deleting and starting over.') |  | ||||||
|     try: |  | ||||||
|         token = json.loads(path.read_bytes()) |  | ||||||
|     except subprocess.CalledProcessError: |  | ||||||
|         sys.exit('Difficulty decrypting token file. Is your decryption agent primed for ' |  | ||||||
|                  'non-interactive usage, or an appropriate environment variable such as ' |  | ||||||
|                  'GPG_TTY set to allow interactive agent usage from inside a pipe?') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| client_id, client_secret = (path.parent / "gmail-client-app.txt").read_text().strip().split('\n') |  | ||||||
|  |  | ||||||
| registrations = { |  | ||||||
|     'google': { |  | ||||||
|         'authorize_endpoint': 'https://accounts.google.com/o/oauth2/auth', |  | ||||||
|         'devicecode_endpoint': 'https://oauth2.googleapis.com/device/code', |  | ||||||
|         'token_endpoint': 'https://accounts.google.com/o/oauth2/token', |  | ||||||
|         'redirect_uri': 'urn:ietf:wg:oauth:2.0:oob', |  | ||||||
|         'imap_endpoint': 'imap.gmail.com', |  | ||||||
|         'pop_endpoint': 'pop.gmail.com', |  | ||||||
|         'smtp_endpoint': 'smtp.gmail.com', |  | ||||||
|         'sasl_method': 'OAUTHBEARER', |  | ||||||
|         'scope': 'https://mail.google.com/', |  | ||||||
|         'client_id': client_id, |  | ||||||
|         'client_secret': client_secret, |  | ||||||
|     }, |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def writetokenfile(): |  | ||||||
|     '''Writes global token dictionary into token file.''' |  | ||||||
|     if not path.exists(): |  | ||||||
|         path.touch(mode=0o600) |  | ||||||
|     if 0o777 & path.stat().st_mode != 0o600: |  | ||||||
|         sys.exit('Token file has unsafe mode. Suggest deleting and starting over.') |  | ||||||
|     path.write_bytes(json.dumps(token).encode('utf-8')) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if args.debug: |  | ||||||
|     print('Obtained from token file:', json.dumps(token)) |  | ||||||
| if not token: |  | ||||||
|     if not args.authorize: |  | ||||||
|         sys.exit('You must run script with "--authorize" at least once.') |  | ||||||
|     print('Available app and endpoint registrations:', *registrations) |  | ||||||
|     token['registration'] = input('OAuth2 registration: ') |  | ||||||
|     token['authflow'] = input('Preferred OAuth2 flow ("authcode" or "localhostauthcode" ' |  | ||||||
|                               'or "devicecode"): ') |  | ||||||
|     token['email'] = input('Account e-mail address: ') |  | ||||||
|     token['access_token'] = '' |  | ||||||
|     token['access_token_expiration'] = '' |  | ||||||
|     token['refresh_token'] = '' |  | ||||||
|     writetokenfile() |  | ||||||
|  |  | ||||||
| if token['registration'] not in registrations: |  | ||||||
|     sys.exit(f'ERROR: Unknown registration "{token["registration"]}". Delete token file ' |  | ||||||
|              f'and start over.') |  | ||||||
| registration = registrations[token['registration']] |  | ||||||
|  |  | ||||||
| authflow = token['authflow'] |  | ||||||
| if args.authflow: |  | ||||||
|     authflow = args.authflow |  | ||||||
|  |  | ||||||
| baseparams = {'client_id': registration['client_id']} |  | ||||||
| # Microsoft uses 'tenant' but Google does not |  | ||||||
| if 'tenant' in registration: |  | ||||||
|     baseparams['tenant'] = registration['tenant'] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def access_token_valid(): |  | ||||||
|     '''Returns True when stored access token exists and is still valid at this time.''' |  | ||||||
|     token_exp = token['access_token_expiration'] |  | ||||||
|     return token_exp and datetime.now() < datetime.fromisoformat(token_exp) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def update_tokens(r): |  | ||||||
|     '''Takes a response dictionary, extracts tokens out of it, and updates token file.''' |  | ||||||
|     token['access_token'] = r['access_token'] |  | ||||||
|     token['access_token_expiration'] = (datetime.now() + |  | ||||||
|                                         timedelta(seconds=int(r['expires_in']))).isoformat() |  | ||||||
|     if 'refresh_token' in r: |  | ||||||
|         token['refresh_token'] = r['refresh_token'] |  | ||||||
|     writetokenfile() |  | ||||||
|     if args.verbose: |  | ||||||
|         print(f'NOTICE: Obtained new access token, expires {token["access_token_expiration"]}.') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if args.authorize: |  | ||||||
|     p = baseparams.copy() |  | ||||||
|     p['scope'] = registration['scope'] |  | ||||||
|  |  | ||||||
|     if authflow in ('authcode', 'localhostauthcode'): |  | ||||||
|         verifier = secrets.token_urlsafe(90) |  | ||||||
|         challenge = base64.urlsafe_b64encode(hashlib.sha256(verifier.encode()).digest())[:-1] |  | ||||||
|         redirect_uri = registration['redirect_uri'] |  | ||||||
|         listen_port = 0 |  | ||||||
|         if authflow == 'localhostauthcode': |  | ||||||
|             # Find an available port to listen on |  | ||||||
|             s = socket.socket() |  | ||||||
|             s.bind(('127.0.0.1', 0)) |  | ||||||
|             listen_port = s.getsockname()[1] |  | ||||||
|             s.close() |  | ||||||
|             redirect_uri = 'http://localhost:'+str(listen_port)+'/' |  | ||||||
|             # Probably should edit the port number into the actual redirect URL. |  | ||||||
|  |  | ||||||
|         p.update({'login_hint': token['email'], |  | ||||||
|                   'response_type': 'code', |  | ||||||
|                   'redirect_uri': redirect_uri, |  | ||||||
|                   'code_challenge': challenge, |  | ||||||
|                   'code_challenge_method': 'S256'}) |  | ||||||
|         print(registration["authorize_endpoint"] + '?' + |  | ||||||
|               urllib.parse.urlencode(p, quote_via=urllib.parse.quote)) |  | ||||||
|  |  | ||||||
|         authcode = '' |  | ||||||
|         if authflow == 'authcode': |  | ||||||
|             authcode = input('Visit displayed URL to retrieve authorization code. Enter ' |  | ||||||
|                              'code from server (might be in browser address bar): ') |  | ||||||
|         else: |  | ||||||
|             print('Visit displayed URL to authorize this application. Waiting...', |  | ||||||
|                   end='', flush=True) |  | ||||||
|  |  | ||||||
|             class MyHandler(http.server.BaseHTTPRequestHandler): |  | ||||||
|                 '''Handles the browser query resulting from redirect to redirect_uri.''' |  | ||||||
|  |  | ||||||
|                 # pylint: disable=C0103 |  | ||||||
|                 def do_HEAD(self): |  | ||||||
|                     '''Response to a HEAD requests.''' |  | ||||||
|                     self.send_response(200) |  | ||||||
|                     self.send_header('Content-type', 'text/html') |  | ||||||
|                     self.end_headers() |  | ||||||
|  |  | ||||||
|                 def do_GET(self): |  | ||||||
|                     '''For GET request, extract code parameter from URL.''' |  | ||||||
|                     # pylint: disable=W0603 |  | ||||||
|                     global authcode |  | ||||||
|                     querystring = urllib.parse.urlparse(self.path).query |  | ||||||
|                     querydict = urllib.parse.parse_qs(querystring) |  | ||||||
|                     if 'code' in querydict: |  | ||||||
|                         authcode = querydict['code'][0] |  | ||||||
|                     self.do_HEAD() |  | ||||||
|                     self.wfile.write(b'<html><head><title>Authorizaton result</title></head>') |  | ||||||
|                     self.wfile.write(b'<body><p>Authorization redirect completed. You may ' |  | ||||||
|                                      b'close this window.</p></body></html>') |  | ||||||
|             with http.server.HTTPServer(('127.0.0.1', listen_port), MyHandler) as httpd: |  | ||||||
|                 try: |  | ||||||
|                     httpd.handle_request() |  | ||||||
|                 except KeyboardInterrupt: |  | ||||||
|                     pass |  | ||||||
|  |  | ||||||
|         if not authcode: |  | ||||||
|             sys.exit('Did not obtain an authcode.') |  | ||||||
|  |  | ||||||
|         for k in 'response_type', 'login_hint', 'code_challenge', 'code_challenge_method': |  | ||||||
|             del p[k] |  | ||||||
|         p.update({'grant_type': 'authorization_code', |  | ||||||
|                   'code': authcode, |  | ||||||
|                   'client_secret': registration['client_secret'], |  | ||||||
|                   'code_verifier': verifier}) |  | ||||||
|         print('Exchanging the authorization code for an access token') |  | ||||||
|         try: |  | ||||||
|             response = urllib.request.urlopen(registration['token_endpoint'], |  | ||||||
|                                               urllib.parse.urlencode(p).encode()) |  | ||||||
|         except urllib.error.HTTPError as err: |  | ||||||
|             print(err.code, err.reason) |  | ||||||
|             response = err |  | ||||||
|         response = response.read() |  | ||||||
|         if args.debug: |  | ||||||
|             print(response) |  | ||||||
|         response = json.loads(response) |  | ||||||
|         if 'error' in response: |  | ||||||
|             print(response['error']) |  | ||||||
|             if 'error_description' in response: |  | ||||||
|                 print(response['error_description']) |  | ||||||
|             sys.exit(1) |  | ||||||
|  |  | ||||||
|     elif authflow == 'devicecode': |  | ||||||
|         try: |  | ||||||
|             response = urllib.request.urlopen(registration['devicecode_endpoint'], |  | ||||||
|                                               urllib.parse.urlencode(p).encode()) |  | ||||||
|         except urllib.error.HTTPError as err: |  | ||||||
|             print(err.code, err.reason) |  | ||||||
|             response = err |  | ||||||
|         response = response.read() |  | ||||||
|         if args.debug: |  | ||||||
|             print(response) |  | ||||||
|         response = json.loads(response) |  | ||||||
|         if 'error' in response: |  | ||||||
|             print(response['error']) |  | ||||||
|             if 'error_description' in response: |  | ||||||
|                 print(response['error_description']) |  | ||||||
|             sys.exit(1) |  | ||||||
|         print(response['message']) |  | ||||||
|         del p['scope'] |  | ||||||
|         p.update({'grant_type': 'urn:ietf:params:oauth:grant-type:device_code', |  | ||||||
|                   'client_secret': registration['client_secret'], |  | ||||||
|                   'device_code': response['device_code']}) |  | ||||||
|         interval = int(response['interval']) |  | ||||||
|         print('Polling...', end='', flush=True) |  | ||||||
|         while True: |  | ||||||
|             time.sleep(interval) |  | ||||||
|             print('.', end='', flush=True) |  | ||||||
|             try: |  | ||||||
|                 response = urllib.request.urlopen(registration['token_endpoint'], |  | ||||||
|                                                   urllib.parse.urlencode(p).encode()) |  | ||||||
|             except urllib.error.HTTPError as err: |  | ||||||
|                 # Not actually always an error, might just mean "keep trying..." |  | ||||||
|                 response = err |  | ||||||
|             response = response.read() |  | ||||||
|             if args.debug: |  | ||||||
|                 print(response) |  | ||||||
|             response = json.loads(response) |  | ||||||
|             if 'error' not in response: |  | ||||||
|                 break |  | ||||||
|             if response['error'] == 'authorization_declined': |  | ||||||
|                 print(' user declined authorization.') |  | ||||||
|                 sys.exit(1) |  | ||||||
|             if response['error'] == 'expired_token': |  | ||||||
|                 print(' too much time has elapsed.') |  | ||||||
|                 sys.exit(1) |  | ||||||
|             if response['error'] != 'authorization_pending': |  | ||||||
|                 print(response['error']) |  | ||||||
|                 if 'error_description' in response: |  | ||||||
|                     print(response['error_description']) |  | ||||||
|                 sys.exit(1) |  | ||||||
|         print() |  | ||||||
|  |  | ||||||
|     else: |  | ||||||
|         sys.exit(f'ERROR: Unknown OAuth2 flow "{token["authflow"]}. Delete token file and ' |  | ||||||
|                  f'start over.') |  | ||||||
|  |  | ||||||
|     update_tokens(response) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if not access_token_valid(): |  | ||||||
|     if args.verbose: |  | ||||||
|         print('NOTICE: Invalid or expired access token; using refresh token ' |  | ||||||
|               'to obtain new access token.') |  | ||||||
|     if not token['refresh_token']: |  | ||||||
|         sys.exit('ERROR: No refresh token. Run script with "--authorize".') |  | ||||||
|     p = baseparams.copy() |  | ||||||
|     p.update({'client_secret': registration['client_secret'], |  | ||||||
|               'refresh_token': token['refresh_token'], |  | ||||||
|               'grant_type': 'refresh_token'}) |  | ||||||
|     try: |  | ||||||
|         response = urllib.request.urlopen(registration['token_endpoint'], |  | ||||||
|                                           urllib.parse.urlencode(p).encode()) |  | ||||||
|     except urllib.error.HTTPError as err: |  | ||||||
|         print(err.code, err.reason) |  | ||||||
|         response = err |  | ||||||
|     response = response.read() |  | ||||||
|     if args.debug: |  | ||||||
|         print(response) |  | ||||||
|     response = json.loads(response) |  | ||||||
|     if 'error' in response: |  | ||||||
|         print(response['error']) |  | ||||||
|         if 'error_description' in response: |  | ||||||
|             print(response['error_description']) |  | ||||||
|         print('Perhaps refresh token invalid. Try running once with "--authorize"') |  | ||||||
|         sys.exit(1) |  | ||||||
|     update_tokens(response) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if not access_token_valid(): |  | ||||||
|     sys.exit('ERROR: No valid access token. This should not be able to happen.') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if args.verbose: |  | ||||||
|     print('Access Token: ', end='') |  | ||||||
| print(token['access_token']) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def build_sasl_string(user, host, port, bearer_token): |  | ||||||
|     '''Build appropriate SASL string, which depends on cloud server's supported SASL method.''' |  | ||||||
|     if registration['sasl_method'] == 'OAUTHBEARER': |  | ||||||
|         return f'n,a={user},\1host={host}\1port={port}\1auth=Bearer {bearer_token}\1\1' |  | ||||||
|     if registration['sasl_method'] == 'XOAUTH2': |  | ||||||
|         return f'user={user}\1auth=Bearer {bearer_token}\1\1' |  | ||||||
|     sys.exit(f'Unknown SASL method {registration["sasl_method"]}.') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if args.test: |  | ||||||
|     errors = False |  | ||||||
|  |  | ||||||
|     imap_conn = imaplib.IMAP4_SSL(registration['imap_endpoint']) |  | ||||||
|     sasl_string = build_sasl_string(token['email'], registration['imap_endpoint'], 993, |  | ||||||
|                                     token['access_token']) |  | ||||||
|     if args.debug: |  | ||||||
|         imap_conn.debug = 4 |  | ||||||
|     try: |  | ||||||
|         imap_conn.authenticate(registration['sasl_method'], lambda _: sasl_string.encode()) |  | ||||||
|         # Microsoft has a bug wherein a mismatch between username and token can still report a |  | ||||||
|         # successful login... (Try a consumer login with the token from a work/school account.) |  | ||||||
|         # Fortunately subsequent commands fail with an error. Thus we follow AUTH with another |  | ||||||
|         # IMAP command before reporting success. |  | ||||||
|         imap_conn.list() |  | ||||||
|         if args.verbose: |  | ||||||
|             print('IMAP authentication succeeded') |  | ||||||
|     except imaplib.IMAP4.error as e: |  | ||||||
|         print('IMAP authentication FAILED (does your account allow IMAP?):', e) |  | ||||||
|         errors = True |  | ||||||
|  |  | ||||||
|     pop_conn = poplib.POP3_SSL(registration['pop_endpoint']) |  | ||||||
|     sasl_string = build_sasl_string(token['email'], registration['pop_endpoint'], 995, |  | ||||||
|                                     token['access_token']) |  | ||||||
|     if args.debug: |  | ||||||
|         pop_conn.set_debuglevel(2) |  | ||||||
|     try: |  | ||||||
|         # poplib doesn't have an auth command taking an authenticator object |  | ||||||
|         # Microsoft requires a two-line SASL for POP |  | ||||||
|         # pylint: disable=W0212 |  | ||||||
|         pop_conn._shortcmd('AUTH ' + registration['sasl_method']) |  | ||||||
|         pop_conn._shortcmd(base64.standard_b64encode(sasl_string.encode()).decode()) |  | ||||||
|         if args.verbose: |  | ||||||
|             print('POP authentication succeeded') |  | ||||||
|     except poplib.error_proto as e: |  | ||||||
|         print('POP authentication FAILED (does your account allow POP?):', e.args[0].decode()) |  | ||||||
|         errors = True |  | ||||||
|  |  | ||||||
|     # SMTP_SSL would be simpler but Microsoft does not answer on port 465. |  | ||||||
|     smtp_conn = smtplib.SMTP(registration['smtp_endpoint'], 587) |  | ||||||
|     sasl_string = build_sasl_string(token['email'], registration['smtp_endpoint'], 587, |  | ||||||
|                                     token['access_token']) |  | ||||||
|     smtp_conn.ehlo('test') |  | ||||||
|     smtp_conn.starttls() |  | ||||||
|     smtp_conn.ehlo('test') |  | ||||||
|     if args.debug: |  | ||||||
|         smtp_conn.set_debuglevel(2) |  | ||||||
|     try: |  | ||||||
|         smtp_conn.auth(registration['sasl_method'], lambda _=None: sasl_string) |  | ||||||
|         if args.verbose: |  | ||||||
|             print('SMTP authentication succeeded') |  | ||||||
|     except smtplib.SMTPAuthenticationError as e: |  | ||||||
|         print('SMTP authentication FAILED:', e) |  | ||||||
|         errors = True |  | ||||||
|  |  | ||||||
|     if errors: |  | ||||||
|         sys.exit(1) |  | ||||||
|  |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   home.packages = [ |  | ||||||
|     pkgs.shellcheck |  | ||||||
|     pkgs.nodePackages_latest.bash-language-server |  | ||||||
|   ]; |  | ||||||
| } |  | ||||||
| @@ -1,7 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   home.packages = [ |  | ||||||
|     pkgs.ripgrep |  | ||||||
|   ]; |  | ||||||
|  |  | ||||||
|   home.file.".config/ripgrep/config".source = ./ripgrep/ripgrep.conf; |  | ||||||
| } |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   programs.zsh.sessionVariables = { |  | ||||||
|     RUSTFLAGS = "-L ${pkgs.libiconv}/lib -L ${pkgs.libcxx}/lib"; |  | ||||||
|     RUST_BACKTRACE = "full"; |  | ||||||
|   }; |  | ||||||
|   home.file.".cargo/config.toml".source = ./rust/cargo-config.toml; |  | ||||||
|   home.packages = [ |  | ||||||
|     pkgs.rust-analyzer |  | ||||||
|   ]; |  | ||||||
| } |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   imports = [./zsh.nix]; |  | ||||||
|   home.packages = [ |  | ||||||
|     pkgs.tmux |  | ||||||
|   ]; |  | ||||||
|  |  | ||||||
|   programs.tmux = { |  | ||||||
|     shell = "${pkgs.zsh}/bin/zsh"; |  | ||||||
|     escapeTime = 50; |  | ||||||
|     mouse = false; |  | ||||||
|     prefix = "C-b"; |  | ||||||
|     enable = true; |  | ||||||
|     terminal = "screen-256color"; |  | ||||||
|     extraConfig = '' |  | ||||||
|       set-option -sa terminal-features ',xterm-256color:RGB' |  | ||||||
|       set -g default-command "exec ${pkgs.zsh}/bin/zsh" |  | ||||||
|  |  | ||||||
|       # Vi mode |  | ||||||
|       set-window-option -g mode-keys vi |  | ||||||
|  |  | ||||||
|       # Use v to begin selection in copy mode |  | ||||||
|       bind-key -T copy-mode-vi v send-keys -X begin-selection |  | ||||||
|  |  | ||||||
|       # Use Shift+V to select line |  | ||||||
|       bind-key -T copy-mode-vi V send-keys -X select-line |  | ||||||
|  |  | ||||||
|       # Use y to yank to clipboard |  | ||||||
|       ${ |  | ||||||
|         if pkgs.stdenv.isDarwin |  | ||||||
|         then '' |  | ||||||
|           bind-key -T copy-mode-vi y send-keys -X copy-pipe-and-cancel "pbcopy" |  | ||||||
|         '' |  | ||||||
|         else '' |  | ||||||
|           bind-key -T copy-mode-vi y send-keys -X copy-pipe-and-cancel "${pkgs.wl-clipboard}/bin/wl-copy" |  | ||||||
|         '' |  | ||||||
|       } |  | ||||||
|     ''; |  | ||||||
|   }; |  | ||||||
| } |  | ||||||
| @@ -1,199 +0,0 @@ | |||||||
| #!/usr/bin/env python3 |  | ||||||
|  |  | ||||||
| # MIT licenced, https://github.com/Alexays/Waybar/blob/dacecb9b265c1c7c36ee43d17526fa95f4e6596f/resources/custom_modules/mediaplayer.py |  | ||||||
| # See licence in power_menu.xml |  | ||||||
|  |  | ||||||
| import gi |  | ||||||
| gi.require_version("Playerctl", "2.0") |  | ||||||
| from gi.repository import Playerctl, GLib |  | ||||||
| from gi.repository.Playerctl import Player |  | ||||||
| import argparse |  | ||||||
| import logging |  | ||||||
| import sys |  | ||||||
| import signal |  | ||||||
| import gi |  | ||||||
| import json |  | ||||||
| import os |  | ||||||
| from typing import List |  | ||||||
|  |  | ||||||
| logger = logging.getLogger(__name__) |  | ||||||
|  |  | ||||||
| def signal_handler(sig, frame): |  | ||||||
|     logger.info("Received signal to stop, exiting") |  | ||||||
|     sys.stdout.write("\n") |  | ||||||
|     sys.stdout.flush() |  | ||||||
|     # loop.quit() |  | ||||||
|     sys.exit(0) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class PlayerManager: |  | ||||||
|     def __init__(self, selected_player=None, excluded_player=[]): |  | ||||||
|         self.manager = Playerctl.PlayerManager() |  | ||||||
|         self.loop = GLib.MainLoop() |  | ||||||
|         self.manager.connect( |  | ||||||
|             "name-appeared", lambda *args: self.on_player_appeared(*args)) |  | ||||||
|         self.manager.connect( |  | ||||||
|             "player-vanished", lambda *args: self.on_player_vanished(*args)) |  | ||||||
|  |  | ||||||
|         signal.signal(signal.SIGINT, signal_handler) |  | ||||||
|         signal.signal(signal.SIGTERM, signal_handler) |  | ||||||
|         signal.signal(signal.SIGPIPE, signal.SIG_DFL) |  | ||||||
|         self.selected_player = selected_player |  | ||||||
|         self.excluded_player = excluded_player.split(',') if excluded_player else [] |  | ||||||
|  |  | ||||||
|         self.init_players() |  | ||||||
|  |  | ||||||
|     def init_players(self): |  | ||||||
|         for player in self.manager.props.player_names: |  | ||||||
|             if player.name in self.excluded_player: |  | ||||||
|                 continue |  | ||||||
|             if self.selected_player is not None and self.selected_player != player.name: |  | ||||||
|                 logger.debug(f"{player.name} is not the filtered player, skipping it") |  | ||||||
|                 continue |  | ||||||
|             self.init_player(player) |  | ||||||
|  |  | ||||||
|     def run(self): |  | ||||||
|         logger.info("Starting main loop") |  | ||||||
|         self.loop.run() |  | ||||||
|  |  | ||||||
|     def init_player(self, player): |  | ||||||
|         logger.info(f"Initialize new player: {player.name}") |  | ||||||
|         player = Playerctl.Player.new_from_name(player) |  | ||||||
|         player.connect("playback-status", |  | ||||||
|                        self.on_playback_status_changed, None) |  | ||||||
|         player.connect("metadata", self.on_metadata_changed, None) |  | ||||||
|         self.manager.manage_player(player) |  | ||||||
|         self.on_metadata_changed(player, player.props.metadata) |  | ||||||
|  |  | ||||||
|     def get_players(self) -> List[Player]: |  | ||||||
|         return self.manager.props.players |  | ||||||
|  |  | ||||||
|     def write_output(self, text, player): |  | ||||||
|         logger.debug(f"Writing output: {text}") |  | ||||||
|  |  | ||||||
|         output = {"text": text, |  | ||||||
|                   "class": "custom-" + player.props.player_name, |  | ||||||
|                   "alt": player.props.player_name} |  | ||||||
|  |  | ||||||
|         sys.stdout.write(json.dumps(output) + "\n") |  | ||||||
|         sys.stdout.flush() |  | ||||||
|  |  | ||||||
|     def clear_output(self): |  | ||||||
|         sys.stdout.write("\n") |  | ||||||
|         sys.stdout.flush() |  | ||||||
|  |  | ||||||
|     def on_playback_status_changed(self, player, status, _=None): |  | ||||||
|         logger.debug(f"Playback status changed for player {player.props.player_name}: {status}") |  | ||||||
|         self.on_metadata_changed(player, player.props.metadata) |  | ||||||
|  |  | ||||||
|     def get_first_playing_player(self): |  | ||||||
|         players = self.get_players() |  | ||||||
|         logger.debug(f"Getting first playing player from {len(players)} players") |  | ||||||
|         if len(players) > 0: |  | ||||||
|             # if any are playing, show the first one that is playing |  | ||||||
|             # reverse order, so that the most recently added ones are preferred |  | ||||||
|             for player in players[::-1]: |  | ||||||
|                 if player.props.status == "Playing": |  | ||||||
|                     return player |  | ||||||
|             # if none are playing, show the first one |  | ||||||
|             return players[0] |  | ||||||
|         else: |  | ||||||
|             logger.debug("No players found") |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|     def show_most_important_player(self): |  | ||||||
|         logger.debug("Showing most important player") |  | ||||||
|         # show the currently playing player |  | ||||||
|         # or else show the first paused player |  | ||||||
|         # or else show nothing |  | ||||||
|         current_player = self.get_first_playing_player() |  | ||||||
|         if current_player is not None: |  | ||||||
|             self.on_metadata_changed(current_player, current_player.props.metadata) |  | ||||||
|         else: |  | ||||||
|             self.clear_output() |  | ||||||
|  |  | ||||||
|     def on_metadata_changed(self, player, metadata, _=None): |  | ||||||
|         logger.debug(f"Metadata changed for player {player.props.player_name}") |  | ||||||
|         player_name = player.props.player_name |  | ||||||
|         artist = player.get_artist() |  | ||||||
|         title = player.get_title() |  | ||||||
|         title = title.replace("&", "&") |  | ||||||
|  |  | ||||||
|         track_info = "" |  | ||||||
|         if player_name == "spotify" and "mpris:trackid" in metadata.keys() and ":ad:" in player.props.metadata["mpris:trackid"]: |  | ||||||
|             track_info = "Advertisement" |  | ||||||
|         elif artist is not None and title is not None: |  | ||||||
|             track_info = f"{artist} - {title}" |  | ||||||
|         else: |  | ||||||
|             track_info = title |  | ||||||
|  |  | ||||||
|         if track_info: |  | ||||||
|             if player.props.status == "Playing": |  | ||||||
|                 track_info = " " + track_info |  | ||||||
|             else: |  | ||||||
|                 track_info = " " + track_info |  | ||||||
|         # only print output if no other player is playing |  | ||||||
|         current_playing = self.get_first_playing_player() |  | ||||||
|         if current_playing is None or current_playing.props.player_name == player.props.player_name: |  | ||||||
|             self.write_output(track_info, player) |  | ||||||
|         else: |  | ||||||
|             logger.debug(f"Other player {current_playing.props.player_name} is playing, skipping") |  | ||||||
|  |  | ||||||
|     def on_player_appeared(self, _, player): |  | ||||||
|         logger.info(f"Player has appeared: {player.name}") |  | ||||||
|         if player.name in self.excluded_player: |  | ||||||
|             logger.debug( |  | ||||||
|                 "New player appeared, but it's in exclude player list, skipping") |  | ||||||
|             return |  | ||||||
|         if player is not None and (self.selected_player is None or player.name == self.selected_player): |  | ||||||
|             self.init_player(player) |  | ||||||
|         else: |  | ||||||
|             logger.debug( |  | ||||||
|                 "New player appeared, but it's not the selected player, skipping") |  | ||||||
|  |  | ||||||
|     def on_player_vanished(self, _, player): |  | ||||||
|         logger.info(f"Player {player.props.player_name} has vanished") |  | ||||||
|         self.show_most_important_player() |  | ||||||
|  |  | ||||||
| def parse_arguments(): |  | ||||||
|     parser = argparse.ArgumentParser() |  | ||||||
|  |  | ||||||
|     # Increase verbosity with every occurrence of -v |  | ||||||
|     parser.add_argument("-v", "--verbose", action="count", default=0) |  | ||||||
|  |  | ||||||
|     parser.add_argument("-x", "--exclude", "- Comma-separated list of excluded player") |  | ||||||
|  |  | ||||||
|     # Define for which player we"re listening |  | ||||||
|     parser.add_argument("--player") |  | ||||||
|  |  | ||||||
|     parser.add_argument("--enable-logging", action="store_true") |  | ||||||
|  |  | ||||||
|     return parser.parse_args() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): |  | ||||||
|     arguments = parse_arguments() |  | ||||||
|  |  | ||||||
|     # Initialize logging |  | ||||||
|     if arguments.enable_logging: |  | ||||||
|         logfile = os.path.join(os.path.dirname( |  | ||||||
|             os.path.realpath(__file__)), "media-player.log") |  | ||||||
|         logging.basicConfig(filename=logfile, level=logging.DEBUG, |  | ||||||
|                             format="%(asctime)s %(name)s %(levelname)s:%(lineno)d %(message)s") |  | ||||||
|  |  | ||||||
|     # Logging is set by default to WARN and higher. |  | ||||||
|     # With every occurrence of -v it's lowered by one |  | ||||||
|     logger.setLevel(max((3 - arguments.verbose) * 10, 0)) |  | ||||||
|  |  | ||||||
|     logger.info("Creating player manager") |  | ||||||
|     if arguments.player: |  | ||||||
|         logger.info(f"Filtering for player: {arguments.player}") |  | ||||||
|     if arguments.exclude: |  | ||||||
|         logger.info(f"Exclude player {arguments.exclude}") |  | ||||||
|  |  | ||||||
|     player = PlayerManager(arguments.player, arguments.exclude) |  | ||||||
|     player.run() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == "__main__": |  | ||||||
|     main() |  | ||||||
| @@ -1,53 +0,0 @@ | |||||||
| <?xml version="1.0" encoding="UTF-8"?> |  | ||||||
| <!-- Source: https://github.com/Alexays/Waybar/blob/dacecb9b265c1c7c36ee43d17526fa95f4e6596f/resources/custom_modules/power_menu.xml --> |  | ||||||
| <!-- MIT licence available at https://github.com/Alexays/Waybar/blob/dacecb9b265c1c7c36ee43d17526fa95f4e6596f/LICENSE |  | ||||||
|  |  | ||||||
| MIT License |  | ||||||
|  |  | ||||||
| Copyright (c) 2018 Alex |  | ||||||
|  |  | ||||||
| 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, subject to the following conditions: |  | ||||||
|  |  | ||||||
| The above copyright notice and this permission notice shall be included in all |  | ||||||
| copies or substantial portions of the Software. |  | ||||||
|  |  | ||||||
| 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. |  | ||||||
| --> |  | ||||||
| <interface> |  | ||||||
|   <object class="GtkMenu" id="menu"> |  | ||||||
|     <child> |  | ||||||
| 		<object class="GtkMenuItem" id="suspend"> |  | ||||||
| 			<property name="label">Suspend</property> |  | ||||||
|         </object> |  | ||||||
| 	</child> |  | ||||||
| 	<child> |  | ||||||
|         <object class="GtkMenuItem" id="hibernate"> |  | ||||||
| 			<property name="label">Hibernate</property> |  | ||||||
|         </object> |  | ||||||
| 	</child> |  | ||||||
|     <child> |  | ||||||
|         <object class="GtkMenuItem" id="shutdown"> |  | ||||||
| 			<property name="label">Shutdown</property> |  | ||||||
|         </object> |  | ||||||
|     </child> |  | ||||||
|     <child> |  | ||||||
|       <object class="GtkSeparatorMenuItem" id="delimiter1"/> |  | ||||||
|     </child> |  | ||||||
|     <child> |  | ||||||
| 		<object class="GtkMenuItem" id="reboot"> |  | ||||||
| 			<property name="label">Reboot</property> |  | ||||||
| 		</object> |  | ||||||
|     </child> |  | ||||||
|   </object> |  | ||||||
| </interface> |  | ||||||
| @@ -1,23 +0,0 @@ | |||||||
| {pkgs, ...}: { |  | ||||||
|   programs.zsh = { |  | ||||||
|     enable = true; |  | ||||||
|     autocd = true; |  | ||||||
|     autosuggestion.enable = true; |  | ||||||
|     enableCompletion = true; |  | ||||||
|     history = { |  | ||||||
|       expireDuplicatesFirst = true; |  | ||||||
|     }; |  | ||||||
|     sessionVariables = { |  | ||||||
|       EDITOR = "vim"; |  | ||||||
|       LC_ALL = "en_US.UTF-8"; |  | ||||||
|       LC_CTYPE = "en_US.UTF-8"; |  | ||||||
|     }; |  | ||||||
|     shellAliases = { |  | ||||||
|       vim = "nvim"; |  | ||||||
|       view = "vim -R"; |  | ||||||
|     }; |  | ||||||
|     initContent = builtins.readFile ./zsh/zshrc; |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   programs.fzf.enableZshIntegration = true; |  | ||||||
| } |  | ||||||
| @@ -1,39 +0,0 @@ | |||||||
| prompt_custom() { |  | ||||||
|   local cyan='%F{cyan}' |  | ||||||
|   local red='%F{red}' |  | ||||||
|   local blue='%F{blue}' |  | ||||||
|   local reset_color='%f' |  | ||||||
|  |  | ||||||
|   local git_info=$(git symbolic-ref --short HEAD 2> /dev/null || git describe --tags --exact-match 2> /dev/null || git rev-parse --short HEAD 2> /dev/null) |  | ||||||
|   if [[ -n "$git_info" ]]; then |  | ||||||
|     # escape the percent character, which is the only zsh prompt metacharacter |  | ||||||
|     git_info=$git_info:s/%/%%/ |  | ||||||
|     git_info=" ${blue}git:${reset_color}${red}(${git_info})${reset_color}" |  | ||||||
|   else |  | ||||||
|     git_info="" |  | ||||||
|   fi |  | ||||||
|  |  | ||||||
|   # %1 is the name of cwd |  | ||||||
|   PROMPT="${cyan}%1~${reset_color}${git_info} > " |  | ||||||
| } |  | ||||||
|  |  | ||||||
| # Full path to cwd, with `~` for any initial home component, in light green, |  | ||||||
| RPROMPT='%F{155}%~%f' |  | ||||||
|  |  | ||||||
| precmd_functions+=(prompt_custom) |  | ||||||
|  |  | ||||||
| export WORDCHARS='' |  | ||||||
|  |  | ||||||
| export OPENAI_API_KEY=$(cat ~/.secrets/openai.txt) |  | ||||||
| export ANTHROPIC_API_KEY=$(cat ~/.secrets/anthropic.txt) |  | ||||||
|  |  | ||||||
| autoload edit-command-line |  | ||||||
| zle -N edit-command-line |  | ||||||
| bindkey -e |  | ||||||
| bindkey '^X^E' edit-command-line |  | ||||||
|  |  | ||||||
| if [[ -z "$TMUX" ]]; then |  | ||||||
|     tmux new-session -A -s default |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| ttyctl -f |  | ||||||
| @@ -1,110 +0,0 @@ | |||||||
| BuildUtils = {} |  | ||||||
|  |  | ||||||
| -- Create a new buffer and a new floating window to hold that buffer. |  | ||||||
| local function create_floating_window() |  | ||||||
| 	-- Create a new buffer for build output |  | ||||||
| 	local buf = vim.api.nvim_create_buf(false, true) -- No listed, scratch buffer |  | ||||||
|  |  | ||||||
| 	-- Calculate window size and position here (example: full width, 10 lines high at the bottom) |  | ||||||
| 	local width = vim.api.nvim_get_option_value("columns", {}) |  | ||||||
| 	local height = vim.api.nvim_get_option_value("lines", {}) |  | ||||||
| 	local win_height = math.min(10, math.floor(height * 0.2)) -- 20% of total height or 10 lines |  | ||||||
| 	local win_opts = { |  | ||||||
| 		relative = "editor", |  | ||||||
| 		width = width, |  | ||||||
| 		height = win_height, |  | ||||||
| 		col = 0, |  | ||||||
| 		row = height - win_height, |  | ||||||
| 		style = "minimal", |  | ||||||
| 		border = "single", |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	local win = vim.api.nvim_open_win(buf, false, win_opts) |  | ||||||
|  |  | ||||||
| 	return { window = win, buffer = buf } |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function _on_output(context, is_stdout, err, data, on_line) |  | ||||||
| 	local prefix |  | ||||||
| 	if is_stdout then |  | ||||||
| 		prefix = "OUT" |  | ||||||
| 	else |  | ||||||
| 		prefix = "ERR" |  | ||||||
| 	end |  | ||||||
| 	if err or data then |  | ||||||
| 		vim.schedule(function() |  | ||||||
| 			if err then |  | ||||||
| 				-- Append the error message to the buffer |  | ||||||
| 				local count = vim.api.nvim_buf_line_count(context.buffer) |  | ||||||
| 				vim.api.nvim_buf_set_lines(context.buffer, count, count, false, { "error " .. prefix .. ": " .. err }) |  | ||||||
| 			end |  | ||||||
| 			if data then |  | ||||||
| 				-- Append the data to the buffer |  | ||||||
| 				local count = vim.api.nvim_buf_line_count(context.buffer) |  | ||||||
| 				vim.api.nvim_buf_set_lines( |  | ||||||
| 					context.buffer, |  | ||||||
| 					count, |  | ||||||
| 					count, |  | ||||||
| 					false, |  | ||||||
| 					vim.tbl_map(function(line) |  | ||||||
| 						return prefix .. ": " .. line |  | ||||||
| 					end, vim.split(data, "\n")) |  | ||||||
| 				) |  | ||||||
| 			end |  | ||||||
| 			if vim.api.nvim_win_is_valid(context.window) then |  | ||||||
| 				local cur_win = vim.api.nvim_get_current_win() |  | ||||||
| 				local cur_buf = vim.api.nvim_win_get_buf(cur_win) |  | ||||||
| 				if cur_buf ~= context.buffer then |  | ||||||
| 					local new_line_count = vim.api.nvim_buf_line_count(context.buffer) |  | ||||||
| 					vim.api.nvim_win_set_cursor(context.window, { new_line_count, 0 }) |  | ||||||
| 				end |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			on_line(data, is_stdout, context) |  | ||||||
| 		end) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Arguments: |  | ||||||
| -- * exe, a string (no need to escape this) |  | ||||||
| -- * args, a table like { "-m", "venv", vim.fn.shellescape(some_path) } |  | ||||||
| -- * description of this process, visible to the user, e.g. "venv creation" |  | ||||||
| -- * context, the result of `create_floating_window` |  | ||||||
| -- * on_line, a function which takes "the string written", (true if stdout else false), and the context table; should return nothing. We'll call that on every line of stdout and stderr. |  | ||||||
| -- * on_complete, takes `context`, `code` (exit code) and `signal` ("documented" with neovim's uv.spawn, hah) |  | ||||||
| local function run_external(exe, args, description, context, on_line, on_complete) |  | ||||||
| 	local handle |  | ||||||
| 	local stdout = vim.uv.new_pipe(false) |  | ||||||
| 	local stderr = vim.uv.new_pipe(false) |  | ||||||
| 	handle, _ = vim.uv.spawn( |  | ||||||
| 		exe, |  | ||||||
| 		{ |  | ||||||
| 			args = args, |  | ||||||
| 			stdio = { nil, stdout, stderr }, |  | ||||||
| 		}, |  | ||||||
| 		vim.schedule_wrap(function(code, signal) |  | ||||||
| 			stdout:read_stop() |  | ||||||
| 			stderr:read_stop() |  | ||||||
| 			stdout:close() |  | ||||||
| 			stderr:close() |  | ||||||
| 			handle:close() |  | ||||||
| 			print("External process " .. description .. " completed, exit code " .. code .. " and signal " .. signal) |  | ||||||
| 			on_complete(context, code, signal) |  | ||||||
| 		end) |  | ||||||
| 	) |  | ||||||
|  |  | ||||||
| 	if not handle then |  | ||||||
| 		print("Failed to start " .. description .. " process.") |  | ||||||
| 		return |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	vim.uv.read_start(stdout, function(err, data) |  | ||||||
| 		_on_output(context, true, err, data, on_line) |  | ||||||
| 	end) |  | ||||||
| 	vim.uv.read_start(stderr, function(err, data) |  | ||||||
| 		_on_output(context, false, err, data, on_line) |  | ||||||
| 	end) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| BuildUtils.create_window = create_floating_window |  | ||||||
| BuildUtils.run = run_external |  | ||||||
| @@ -1,57 +0,0 @@ | |||||||
| vim.g.chadtree_settings = { xdg = true } |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("VimEnter", { |  | ||||||
| 	pattern = "*", |  | ||||||
| 	command = "CHADopen --nofocus", |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("BufEnter", { |  | ||||||
| 	pattern = "*", |  | ||||||
| 	callback = function() |  | ||||||
| 		if vim.fn.winnr("$") == 1 and vim.bo.filetype == "CHADTree" then |  | ||||||
| 			vim.cmd("quit") |  | ||||||
| 		end |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| -- Variable to store the CHADtree window ID |  | ||||||
| local chadtree_winid_and_buf = nil |  | ||||||
|  |  | ||||||
| -- Function to check if a window is displaying CHADtree |  | ||||||
| local function is_chadtree_window(winid) |  | ||||||
| 	local bufnr = vim.api.nvim_win_get_buf(winid) |  | ||||||
| 	local filetype = vim.api.nvim_get_option_value("filetype", { buf = bufnr }) |  | ||||||
| 	return filetype == "CHADTree" |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Function to find and store the CHADtree window ID |  | ||||||
| local function find_chadtree_window() |  | ||||||
| 	for _, winid in ipairs(vim.api.nvim_list_wins()) do |  | ||||||
| 		if is_chadtree_window(winid) then |  | ||||||
| 			chadtree_winid_and_buf = { winid, vim.api.nvim_win_get_buf(winid) } |  | ||||||
| 			break |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Function to switch to CHADtree buffer in the CHADtree window |  | ||||||
| local function switch_to_chadtree() |  | ||||||
| 	if chadtree_winid_and_buf and vim.api.nvim_win_is_valid(chadtree_winid_and_buf[1]) then |  | ||||||
| 		local current_winid = vim.api.nvim_get_current_win() |  | ||||||
| 		if current_winid == chadtree_winid_and_buf[1] and not is_chadtree_window(current_winid) then |  | ||||||
| 			print("CHADtree window may only point to CHADtree") |  | ||||||
| 			vim.api.nvim_win_set_buf(chadtree_winid_and_buf[1], chadtree_winid_and_buf[2]) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Autocommand to find the CHADtree window after startup |  | ||||||
| vim.api.nvim_create_autocmd("VimEnter", { |  | ||||||
| 	callback = function() |  | ||||||
| 		vim.defer_fn(find_chadtree_window, 500) |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("BufEnter", { |  | ||||||
| 	callback = switch_to_chadtree, |  | ||||||
| }) |  | ||||||
| @@ -1,765 +0,0 @@ | |||||||
| local dotnet_has_set_status_line |  | ||||||
|  |  | ||||||
| function DetachSolution() |  | ||||||
| 	vim.g.current_sln_path = nil |  | ||||||
| 	-- TODO: unregister key bindings again |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function on_line(data, _, context) |  | ||||||
| 	-- Keep the window alive if there were warnings |  | ||||||
| 	if string.match(data, "%s[1-9]%d* Warning%(s%)") then |  | ||||||
| 		context.warn = context.warn + 1 |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
| local function on_complete(context, code, _) |  | ||||||
| 	if code ~= 0 then |  | ||||||
| 		print("Exit code " .. code) |  | ||||||
| 		context.errs = context.errs + 1 |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	if context.errs == 0 and context.warn == 0 then |  | ||||||
| 		-- Close the temporary floating window (but keep it alive if the |  | ||||||
| 		-- cursor is in it) |  | ||||||
| 		local cur_win = vim.api.nvim_get_current_win() |  | ||||||
| 		local cur_buf = vim.api.nvim_win_get_buf(cur_win) |  | ||||||
| 		if cur_buf ~= context.buffer then |  | ||||||
| 			vim.api.nvim_win_close(context.window, true) |  | ||||||
| 		end |  | ||||||
| 		print("All builds successful") |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function GetCurrentSln() |  | ||||||
| 	if vim.g.current_sln_path then |  | ||||||
| 		return vim.g.current_sln_path |  | ||||||
| 	else |  | ||||||
| 		return nil |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function BuildDotNetSolution() |  | ||||||
| 	if vim.g.current_sln_path then |  | ||||||
| 		local context = BuildUtils.create_window() |  | ||||||
| 		context.errs = 0 |  | ||||||
| 		context.warn = 0 |  | ||||||
| 		BuildUtils.run("dotnet", { "build", vim.g.current_sln_path }, "dotnet build", context, on_line, on_complete) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function TestDotNetSolution() |  | ||||||
| 	if vim.g.current_sln_path then |  | ||||||
| 		local context = BuildUtils.create_window() |  | ||||||
| 		context.warn = 0 |  | ||||||
| 		context.errs = 0 |  | ||||||
| 		BuildUtils.run("dotnet", { "test", vim.g.current_sln_path }, "dotnet test", context, on_line, on_complete) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function CurrentSlnOrEmpty() |  | ||||||
| 	local sln = GetCurrentSln() |  | ||||||
| 	if sln then |  | ||||||
| 		return sln |  | ||||||
| 	else |  | ||||||
| 		return "" |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function RegisterSolution(sln_path) |  | ||||||
| 	vim.g.current_sln_path = sln_path |  | ||||||
|  |  | ||||||
| 	if not dotnet_has_set_status_line then |  | ||||||
| 		dotnet_has_set_status_line = true |  | ||||||
| 		vim.o.statusline = vim.o.statusline .. "  %{v:lua.CurrentSlnOrEmpty()}" |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local whichkey = require("which-key") |  | ||||||
| 	whichkey.add({ |  | ||||||
| 		{ |  | ||||||
| 			"<localleader>s", |  | ||||||
| 			desc = ".NET solution", |  | ||||||
| 		}, |  | ||||||
| 		{ "<localleader>sb", BuildDotNetSolution, desc = "Build .NET solution" }, |  | ||||||
| 		{ "<localleader>st", TestDotNetSolution, desc = "Test .NET solution" }, |  | ||||||
| 	}, { buffer = vim.api.nvim_get_current_buf() }) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function find_nearest_slns() |  | ||||||
| 	local path = vim.fn.expand("%:p:h") -- Get the full path of the current buffer's directory |  | ||||||
|  |  | ||||||
| 	while path and path ~= "/" do |  | ||||||
| 		local sln_paths = vim.fn.glob(path .. "/*.sln", nil, true) |  | ||||||
| 		if #sln_paths > 0 then |  | ||||||
| 			return sln_paths |  | ||||||
| 		end |  | ||||||
| 		path = vim.fn.fnamemodify(path, ":h") -- Move up one directory |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	return {} |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function FindAndRegisterSolution(should_override) |  | ||||||
| 	if not should_override and GetCurrentSln() ~= nil then |  | ||||||
| 		RegisterSolution(GetCurrentSln()) |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local solutions = find_nearest_slns() |  | ||||||
| 	if not solutions or #solutions == 0 then |  | ||||||
| 		print("No .sln file found in any parent directory.") |  | ||||||
| 		return |  | ||||||
| 	elseif #solutions == 1 then |  | ||||||
| 		-- Exactly one solution found; register it directly |  | ||||||
| 		RegisterSolution(solutions[1]) |  | ||||||
| 	elseif #solutions > 1 then |  | ||||||
| 		-- Multiple solutions found; use Telescope to pick one |  | ||||||
| 		local pickers = require("telescope.pickers") |  | ||||||
| 		local finders = require("telescope.finders") |  | ||||||
| 		local actions = require("telescope.actions") |  | ||||||
| 		local action_state = require("telescope.actions.state") |  | ||||||
| 		local conf = require("telescope.config").values |  | ||||||
|  |  | ||||||
| 		pickers |  | ||||||
| 			.new({}, { |  | ||||||
| 				prompt_title = "Select a Solution File", |  | ||||||
| 				finder = finders.new_table({ |  | ||||||
| 					results = solutions, |  | ||||||
| 					entry_maker = function(entry) |  | ||||||
| 						return { |  | ||||||
| 							value = entry, |  | ||||||
| 							display = entry, |  | ||||||
| 							ordinal = entry, |  | ||||||
| 						} |  | ||||||
| 					end, |  | ||||||
| 				}), |  | ||||||
| 				sorter = conf.generic_sorter({}), |  | ||||||
| 				attach_mappings = function(prompt_bufnr, _) |  | ||||||
| 					actions.select_default:replace(function() |  | ||||||
| 						local selection = action_state.get_selected_entry() |  | ||||||
| 						actions.close(prompt_bufnr) |  | ||||||
| 						RegisterSolution(selection.value) |  | ||||||
| 					end) |  | ||||||
| 					return true |  | ||||||
| 				end, |  | ||||||
| 			}) |  | ||||||
| 			:find() |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd({ "BufReadPost", "BufNewFile" }, { |  | ||||||
| 	pattern = "*.sln", |  | ||||||
| 	callback = function() |  | ||||||
| 		if GetCurrentSln() == nil then |  | ||||||
| 			RegisterSolution(vim.fn.expand("%:p")) |  | ||||||
| 		end |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("FileType", { |  | ||||||
| 	pattern = { "fsharp", "cs", "fsharp_project" }, |  | ||||||
| 	callback = function() |  | ||||||
| 		FindAndRegisterSolution(false) |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| -- For what I'm sure are reasons, Lua appears to have nothing in its standard library |  | ||||||
| ---@generic K |  | ||||||
| ---@generic V1 |  | ||||||
| ---@generic V2 |  | ||||||
| ---@param tbl table<K, V1> |  | ||||||
| ---@param f fun(V1): V2 |  | ||||||
| ---@return table<K, V2> |  | ||||||
| local function map(tbl, f) |  | ||||||
| 	local t = {} |  | ||||||
| 	for k, v in pairs(tbl) do |  | ||||||
| 		t[k] = f(v) |  | ||||||
| 	end |  | ||||||
| 	return t |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@generic K |  | ||||||
| ---@generic V |  | ||||||
| ---@param tbl table<K, V> |  | ||||||
| ---@param f fun(V1): nil |  | ||||||
| local function iter(tbl, f) |  | ||||||
| 	for _, v in pairs(tbl) do |  | ||||||
| 		f(v) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@generic K |  | ||||||
| ---@generic V |  | ||||||
| ---@param tbl table<K, V> |  | ||||||
| ---@param predicate fun(V): boolean |  | ||||||
| ---@return boolean, V |  | ||||||
| local function find(tbl, predicate) |  | ||||||
| 	for _, v in pairs(tbl) do |  | ||||||
| 		if predicate(v) then |  | ||||||
| 			return true, v |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| 	return false, nil |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@class (exact) NuGetVersion |  | ||||||
| ---@field major number |  | ||||||
| ---@field minor number |  | ||||||
| ---@field patch number |  | ||||||
| ---@field suffix? string |  | ||||||
| local NuGetVersion = {} |  | ||||||
|  |  | ||||||
| ---@param v NuGetVersion |  | ||||||
| ---@nodiscard |  | ||||||
| ---@return string |  | ||||||
| local function nuGetVersionToString(v) |  | ||||||
| 	local s = tostring(v.major) .. "." .. tostring(v.minor) .. "." .. tostring(v.patch) |  | ||||||
| 	if v.suffix then |  | ||||||
| 		return s .. v.suffix |  | ||||||
| 	else |  | ||||||
| 		return s |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function get_all_variables() |  | ||||||
| 	local variables = {} |  | ||||||
| 	local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false) |  | ||||||
| 	for _, line in ipairs(lines) do |  | ||||||
| 		local var_name, var_value = line:match("<(%w+)>([^<]+)</(%w+)>") |  | ||||||
| 		if var_name and var_value then |  | ||||||
| 			variables[var_name] = var_value |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| 	return variables |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function resolve_variable(version, variables) |  | ||||||
| 	if version:match("^%$%((.+)%)$") then |  | ||||||
| 		local var_name = version:match("^%$%((.+)%)$") |  | ||||||
| 		return variables[var_name] or nil |  | ||||||
| 	end |  | ||||||
| 	return nil |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param v string |  | ||||||
| ---@nodiscard |  | ||||||
| ---@return NuGetVersion |  | ||||||
| local function parse_version(v) |  | ||||||
| 	local variables = get_all_variables() |  | ||||||
| 	local major, minor, patch, pre = v:match("(%d+)%.(%d+)%.(%d+)(.*)$") |  | ||||||
| 	if major == nil then |  | ||||||
| 		local resolved = resolve_variable(v, variables) |  | ||||||
| 		if resolved ~= nil then |  | ||||||
| 			return parse_version(resolved) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| 	-- TODO: why does this type-check if you remove the field names? |  | ||||||
| 	return { |  | ||||||
| 		major = tonumber(major) or 0, |  | ||||||
| 		minor = tonumber(minor) or 0, |  | ||||||
| 		patch = tonumber(patch) or 0, |  | ||||||
| 		suffix = pre or nil, |  | ||||||
| 	} |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param a NuGetVersion |  | ||||||
| ---@param b NuGetVersion |  | ||||||
| ---@nodiscard |  | ||||||
| ---@return boolean |  | ||||||
| local function compare_versions(a, b) |  | ||||||
| 	if a.major ~= b.major then |  | ||||||
| 		return a.major < b.major |  | ||||||
| 	elseif a.minor ~= b.minor then |  | ||||||
| 		return a.minor < b.minor |  | ||||||
| 	elseif a.patch ~= b.patch then |  | ||||||
| 		return a.patch < b.patch |  | ||||||
| 	elseif a.suffix and not b.suffix then |  | ||||||
| 		return false |  | ||||||
| 	elseif not a.suffix and b.suffix then |  | ||||||
| 		return true |  | ||||||
| 	else |  | ||||||
| 		return a.suffix < b.suffix |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param url string |  | ||||||
| ---@nodiscard |  | ||||||
| local function curl_sync(url) |  | ||||||
| 	local command = string.format("_CURL_ --silent --compressed --fail '%s'", url) |  | ||||||
| 	local response = vim.fn.system(command) |  | ||||||
| 	if vim.v.shell_error ~= 0 then |  | ||||||
| 		print("Failed to fetch " .. url) |  | ||||||
| 		return nil |  | ||||||
| 	end |  | ||||||
| 	local success, decoded = pcall(vim.fn.json_decode, response) |  | ||||||
| 	if not success then |  | ||||||
| 		print("Failed to decode JSON from curl at " .. url) |  | ||||||
| 		return nil |  | ||||||
| 	end |  | ||||||
| 	return decoded |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param url string |  | ||||||
| ---@param callback fun(table): nil |  | ||||||
| ---@return nil |  | ||||||
| local function curl(url, callback) |  | ||||||
| 	local stdout = vim.uv.new_pipe(false) |  | ||||||
| 	local stdout_text = "" |  | ||||||
| 	local handle |  | ||||||
| 	handle, _ = vim.uv.spawn( |  | ||||||
| 		"_CURL_", |  | ||||||
| 		{ args = { "--silent", "--compressed", "--fail", url }, stdio = { nil, stdout, nil } }, |  | ||||||
| 		vim.schedule_wrap(function(code, _) |  | ||||||
| 			stdout:read_stop() |  | ||||||
| 			stdout:close() |  | ||||||
| 			if handle and not handle:is_closing() then |  | ||||||
| 				handle:close() |  | ||||||
| 			end |  | ||||||
| 			if code ~= 0 then |  | ||||||
| 				print("Failed to fetch " .. url) |  | ||||||
| 			end |  | ||||||
| 			local success, decoded = pcall(vim.fn.json_decode, stdout_text) |  | ||||||
| 			if not success then |  | ||||||
| 				print("Failed to decode JSON from curl at " .. url .. "\n" .. stdout_text) |  | ||||||
| 			end |  | ||||||
| 			callback(decoded) |  | ||||||
| 		end) |  | ||||||
| 	) |  | ||||||
| 	vim.uv.read_start(stdout, function(err, data) |  | ||||||
| 		assert(not err, err) |  | ||||||
| 		if data then |  | ||||||
| 			stdout_text = stdout_text .. data |  | ||||||
| 		end |  | ||||||
| 	end) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local _nugetIndex |  | ||||||
| local _packageBaseAddress |  | ||||||
|  |  | ||||||
| ---@param callback fun(): nil |  | ||||||
| local function populate_nuget_api(callback) |  | ||||||
| 	if _nugetIndex ~= nil then |  | ||||||
| 		callback() |  | ||||||
| 	end |  | ||||||
| 	local url = string.format("https://api.nuget.org/v3/index.json") |  | ||||||
| 	local function handle(decoded) |  | ||||||
| 		local default_nuget_reg = "https://api.nuget.org/v3/registration5-semver1/" |  | ||||||
| 		local default_base_address = "https://api.nuget.org/v3-flatcontainer/" |  | ||||||
|  |  | ||||||
| 		if not decoded then |  | ||||||
| 			print("Failed to fetch NuGet index; falling back to default") |  | ||||||
| 			_nugetIndex = default_nuget_reg |  | ||||||
| 			_packageBaseAddress = default_base_address |  | ||||||
| 		else |  | ||||||
| 			local resources = decoded["resources"] |  | ||||||
| 			if resources == nil then |  | ||||||
| 				print("Failed to parse: " .. decoded .. tostring(decoded)) |  | ||||||
| 				for k, v in pairs(decoded) do |  | ||||||
| 					print(k .. ": " .. tostring(v)) |  | ||||||
| 				end |  | ||||||
| 				callback() |  | ||||||
| 				return |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local resourceSuccess, regUrl = find(resources, function(o) |  | ||||||
| 				return o["@type"] == "RegistrationsBaseUrl/3.6.0" |  | ||||||
| 			end) |  | ||||||
| 			if not resourceSuccess then |  | ||||||
| 				print("Failed to find endpoint in NuGet index; falling back to default") |  | ||||||
| 				_nugetIndex = default_nuget_reg |  | ||||||
| 			else |  | ||||||
| 				_nugetIndex = regUrl["@id"] |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local baseAddrSuccess, baseAddrUrl = find(resources, function(o) |  | ||||||
| 				return o["@type"] == "PackageBaseAddress/3.0.0" |  | ||||||
| 			end) |  | ||||||
| 			if not baseAddrSuccess then |  | ||||||
| 				print("Failed to find endpoint in NuGet index; falling back to default") |  | ||||||
| 				_packageBaseAddress = default_base_address |  | ||||||
| 			else |  | ||||||
| 				_packageBaseAddress = baseAddrUrl["@id"] |  | ||||||
| 			end |  | ||||||
| 		end |  | ||||||
| 		callback() |  | ||||||
| 	end |  | ||||||
| 	curl(url, handle) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@return nil |  | ||||||
| local function populate_nuget_api_sync() |  | ||||||
| 	if _nugetIndex ~= nil then |  | ||||||
| 		return |  | ||||||
| 	end |  | ||||||
| 	local url = string.format("https://api.nuget.org/v3/index.json") |  | ||||||
| 	local decoded = curl_sync(url) |  | ||||||
| 	local default_nuget_reg = "https://api.nuget.org/v3/registration5-semver1/" |  | ||||||
| 	local default_base_address = "https://api.nuget.org/v3-flatcontainer/" |  | ||||||
|  |  | ||||||
| 	if not decoded then |  | ||||||
| 		print("Failed to fetch NuGet index; falling back to default") |  | ||||||
| 		_nugetIndex = default_nuget_reg |  | ||||||
| 		_packageBaseAddress = default_base_address |  | ||||||
| 	else |  | ||||||
| 		local resources = decoded["resources"] |  | ||||||
|  |  | ||||||
| 		local resourceSuccess, regUrl = find(resources, function(o) |  | ||||||
| 			return o["@type"] == "RegistrationsBaseUrl/3.6.0" |  | ||||||
| 		end) |  | ||||||
| 		if not resourceSuccess then |  | ||||||
| 			print("Failed to find endpoint in NuGet index; falling back to default") |  | ||||||
| 			_nugetIndex = default_nuget_reg |  | ||||||
| 		else |  | ||||||
| 			_nugetIndex = regUrl["@id"] |  | ||||||
| 		end |  | ||||||
|  |  | ||||||
| 		local baseAddrSuccess, baseAddrUrl = find(resources, function(o) |  | ||||||
| 			return o["@type"] == "PackageBaseAddress/3.0.0" |  | ||||||
| 		end) |  | ||||||
| 		if not baseAddrSuccess then |  | ||||||
| 			print("Failed to find endpoint in NuGet index; falling back to default") |  | ||||||
| 			_packageBaseAddress = default_base_address |  | ||||||
| 		else |  | ||||||
| 			_packageBaseAddress = baseAddrUrl["@id"] |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@return nil |  | ||||||
| ---@param callback fun(nugetIndex: string): nil |  | ||||||
| local function get_nuget_index(callback) |  | ||||||
| 	populate_nuget_api(function() |  | ||||||
| 		callback(_nugetIndex) |  | ||||||
| 	end) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@return nil |  | ||||||
| ---@param callback fun(packageBaseIndex: string): nil |  | ||||||
| local function get_package_base_addr(callback) |  | ||||||
| 	populate_nuget_api(function() |  | ||||||
| 		callback(_packageBaseAddress) |  | ||||||
| 	end) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@return string |  | ||||||
| local function get_package_base_addr_sync() |  | ||||||
| 	populate_nuget_api_sync() |  | ||||||
| 	return _packageBaseAddress |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local _package_versions_cache = {} |  | ||||||
|  |  | ||||||
| ---@param package_name string |  | ||||||
| ---@return NuGetVersion[] |  | ||||||
| local function get_package_versions_sync(package_name) |  | ||||||
| 	if _package_versions_cache[package_name] ~= nil then |  | ||||||
| 		return _package_versions_cache[package_name] |  | ||||||
| 	end |  | ||||||
| 	local base = get_package_base_addr_sync() |  | ||||||
|  |  | ||||||
| 	local url = base .. string.format("%s/index.json", package_name:lower()) |  | ||||||
| 	local decoded = curl_sync(url) |  | ||||||
| 	if not decoded then |  | ||||||
| 		print("Failed to fetch package versions") |  | ||||||
| 		return {} |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local versions = map(decoded.versions, parse_version) |  | ||||||
| 	table.sort(versions, function(a, b) |  | ||||||
| 		return compare_versions(b, a) |  | ||||||
| 	end) |  | ||||||
| 	_package_versions_cache[package_name] = versions |  | ||||||
| 	return versions |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param package_name string |  | ||||||
| ---@param callback fun(v: NuGetVersion[]): nil |  | ||||||
| ---@return nil |  | ||||||
| local function get_package_versions(package_name, callback) |  | ||||||
| 	if _package_versions_cache[package_name] ~= nil then |  | ||||||
| 		callback(_package_versions_cache[package_name]) |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local function handle(base) |  | ||||||
| 		local url = base .. string.format("%s/index.json", package_name:lower()) |  | ||||||
| 		local function handle2(decoded) |  | ||||||
| 			if not decoded then |  | ||||||
| 				print("Failed to fetch package versions") |  | ||||||
| 				callback({}) |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local versions = map(decoded.versions, parse_version) |  | ||||||
| 			table.sort(versions, function(a, b) |  | ||||||
| 				return compare_versions(b, a) |  | ||||||
| 			end) |  | ||||||
| 			_package_versions_cache[package_name] = versions |  | ||||||
| 			callback(versions) |  | ||||||
| 		end |  | ||||||
| 		curl(url, handle2) |  | ||||||
| 	end |  | ||||||
| 	get_package_base_addr(handle) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param version NuGetVersion |  | ||||||
| ---@return nil |  | ||||||
| local function update_package_version(version) |  | ||||||
| 	local line = vim.api.nvim_get_current_line() |  | ||||||
| 	local new_line = line:gsub('Version="[^"]+"', string.format('Version="%s"', nuGetVersionToString(version))) |  | ||||||
| 	vim.api.nvim_set_current_line(new_line) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- A map from package to { packageWeDependOn: version }. |  | ||||||
| --- @type table<string, table<string, string>> |  | ||||||
| local _package_dependency_cache = {} |  | ||||||
| ---@param package_name string |  | ||||||
| ---@param version NuGetVersion |  | ||||||
| ---@param callback fun(result: table<string, string>): nil |  | ||||||
| ---@return nil |  | ||||||
| local function get_package_dependencies(package_name, version, callback) |  | ||||||
| 	local key = package_name .. "@" .. nuGetVersionToString(version) |  | ||||||
| 	local cache_hit = _package_dependency_cache[key] |  | ||||||
| 	if cache_hit ~= nil then |  | ||||||
| 		callback(cache_hit) |  | ||||||
| 		return |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local function handle1(index) |  | ||||||
| 		local url = index .. string.format("%s/%s.json", package_name:lower(), nuGetVersionToString(version):lower()) |  | ||||||
|  |  | ||||||
| 		local function handle(response) |  | ||||||
| 			if not response then |  | ||||||
| 				print( |  | ||||||
| 					"Failed to get dependencies of " |  | ||||||
| 						.. package_name |  | ||||||
| 						.. " at version " |  | ||||||
| 						.. version |  | ||||||
| 						.. " : unsuccessful response to " |  | ||||||
| 						.. url |  | ||||||
| 				) |  | ||||||
| 				return |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local entry_url = response["catalogEntry"] |  | ||||||
| 			local function handle2(catalog_entry) |  | ||||||
| 				if not catalog_entry then |  | ||||||
| 					print( |  | ||||||
| 						"Failed to get dependencies of " |  | ||||||
| 							.. package_name |  | ||||||
| 							.. " at version " |  | ||||||
| 							.. version |  | ||||||
| 							.. " : unsuccessful response to " |  | ||||||
| 							.. entry_url |  | ||||||
| 					) |  | ||||||
| 					return |  | ||||||
| 				end |  | ||||||
|  |  | ||||||
| 				local result = {} |  | ||||||
| 				if catalog_entry["dependencyGroups"] then |  | ||||||
| 					iter(catalog_entry["dependencyGroups"], function(grp) |  | ||||||
| 						if grp["dependencies"] then |  | ||||||
| 							for _, dep in pairs(grp["dependencies"]) do |  | ||||||
| 								result[dep["id"]] = dep["range"] |  | ||||||
| 							end |  | ||||||
| 						end |  | ||||||
| 					end) |  | ||||||
| 				end |  | ||||||
|  |  | ||||||
| 				_package_dependency_cache[key] = result |  | ||||||
|  |  | ||||||
| 				callback(result) |  | ||||||
| 			end |  | ||||||
| 			curl(entry_url, handle2) |  | ||||||
| 		end |  | ||||||
|  |  | ||||||
| 		curl(url, handle) |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	get_nuget_index(handle1) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@return table<string, NuGetVersion> |  | ||||||
| ---@nodiscard |  | ||||||
| local function get_all_package_references() |  | ||||||
| 	local packages = {} |  | ||||||
| 	local lines = vim.api.nvim_buf_get_lines(0, 0, -1, false) |  | ||||||
|  |  | ||||||
| 	for _, line in ipairs(lines) do |  | ||||||
| 		local package_name = line:match('PackageReference Include="([^"]+)"') |  | ||||||
| 			or line:match('PackageReference Update="([^"]+)"') |  | ||||||
| 		local version = line:match('Version="([^"]+)"') |  | ||||||
|  |  | ||||||
| 		if package_name and version then |  | ||||||
| 			if not packages[package_name] then |  | ||||||
| 				packages[package_name] = {} |  | ||||||
| 			end |  | ||||||
| 			table.insert(packages[package_name], parse_version(version)) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	return packages |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function ClearNuGetDependencyCache() |  | ||||||
| 	for k, _ in pairs(_package_dependency_cache) do |  | ||||||
| 		_package_dependency_cache[k] = nil |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
| vim.api.nvim_create_user_command("ClearNuGetDependencyCache", ClearNuGetDependencyCache, {}) |  | ||||||
|  |  | ||||||
| function PrintNuGetDependencyCache() |  | ||||||
| 	for k, v in pairs(_package_dependency_cache) do |  | ||||||
| 		print(k .. ":") |  | ||||||
| 		for dep, ver in pairs(v) do |  | ||||||
| 			print("  " .. dep .. ": " .. ver) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
| vim.api.nvim_create_user_command("PrintNuGetDependencyCache", PrintNuGetDependencyCache, {}) |  | ||||||
|  |  | ||||||
| local function prefetch_dependencies() |  | ||||||
| 	local packages = get_all_package_references() |  | ||||||
|  |  | ||||||
| 	local function process_package(package_name, versions, callback) |  | ||||||
| 		local count = #versions |  | ||||||
| 		for _, version in ipairs(versions) do |  | ||||||
| 			vim.schedule(function() |  | ||||||
| 				get_package_dependencies(package_name, version, function(_) |  | ||||||
| 					count = count - 1 |  | ||||||
| 					if count == 0 then |  | ||||||
| 						callback() |  | ||||||
| 					end |  | ||||||
| 				end) |  | ||||||
| 			end) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local total_packages = 0 |  | ||||||
| 	for _ in pairs(packages) do |  | ||||||
| 		total_packages = total_packages + 1 |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local processed_packages = 0 |  | ||||||
| 	for package_name, versions in pairs(packages) do |  | ||||||
| 		process_package(package_name, versions, function() |  | ||||||
| 			local function handle(package_versions) |  | ||||||
| 				if package_versions then |  | ||||||
| 					process_package(package_name, package_versions, function() |  | ||||||
| 						processed_packages = processed_packages + 1 |  | ||||||
| 						if processed_packages == total_packages then |  | ||||||
| 							print("All dependencies prefetched") |  | ||||||
| 						end |  | ||||||
| 					end) |  | ||||||
| 				else |  | ||||||
| 					processed_packages = processed_packages + 1 |  | ||||||
| 					if processed_packages == total_packages then |  | ||||||
| 						print("All dependencies prefetched") |  | ||||||
| 					end |  | ||||||
| 				end |  | ||||||
| 			end |  | ||||||
| 			get_package_versions(package_name, handle) |  | ||||||
| 		end) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| ---@param v1 NuGetVersion |  | ||||||
| ---@param v2 NuGetVersion |  | ||||||
| ---@return boolean |  | ||||||
| ---@nodiscard |  | ||||||
| local function nuget_versions_equal(v1, v2) |  | ||||||
| 	return v1.major == v2.major and v1.minor == v2.minor and v1.patch == v2.patch and v1.suffix == v2.suffix |  | ||||||
| end |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("FileType", { |  | ||||||
| 	pattern = { "fsharp_project", "csharp_project", "xml" }, |  | ||||||
| 	callback = function() |  | ||||||
| 		function UpdateNuGetVersion() |  | ||||||
| 			local line = vim.api.nvim_get_current_line() |  | ||||||
| 			local package_name = line:match('PackageReference Include="([^"]+)"') |  | ||||||
| 				or line:match('PackageReference Update="([^"]+)"') |  | ||||||
|  |  | ||||||
| 			if not package_name then |  | ||||||
| 				print("No package reference found on the current line") |  | ||||||
| 				return |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local current_version = parse_version(line:match('Version="([^"]+)"')) |  | ||||||
| 			if not current_version then |  | ||||||
| 				print("oh no!") |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local package_versions = get_package_versions_sync(package_name) |  | ||||||
|  |  | ||||||
| 			if #package_versions == 0 then |  | ||||||
| 				print("No versions found for the package") |  | ||||||
| 				return |  | ||||||
| 			end |  | ||||||
|  |  | ||||||
| 			local pickers = require("telescope.pickers") |  | ||||||
| 			local finders = require("telescope.finders") |  | ||||||
| 			local previewers = require("telescope.previewers") |  | ||||||
|  |  | ||||||
| 			pickers |  | ||||||
| 				.new({}, { |  | ||||||
| 					prompt_title = string.format("Select version for %s", package_name), |  | ||||||
| 					finder = finders.new_table({ |  | ||||||
| 						results = package_versions, |  | ||||||
| 						entry_maker = function(entry) |  | ||||||
| 							local val = nuGetVersionToString(entry) |  | ||||||
| 							local display_value = val |  | ||||||
| 							if current_version and nuget_versions_equal(entry, current_version) then |  | ||||||
| 								display_value = "[CURRENT] " .. val |  | ||||||
| 							end |  | ||||||
| 							return { |  | ||||||
| 								value = entry, |  | ||||||
| 								display = display_value, |  | ||||||
| 								ordinal = entry, |  | ||||||
| 							} |  | ||||||
| 						end, |  | ||||||
| 					}), |  | ||||||
| 					previewer = previewers.new_buffer_previewer({ |  | ||||||
| 						define_preview = function(self, entry, _) |  | ||||||
| 							local bufnr = self.state.bufnr |  | ||||||
| 							vim.api.nvim_buf_set_lines(bufnr, 0, -1, false, { "Loading..." }) |  | ||||||
| 							get_package_dependencies(package_name, entry.value, function(package_dependencies) |  | ||||||
| 								if not package_dependencies then |  | ||||||
| 									vim.api.nvim_buf_set_lines(bufnr, 0, -1, false, { "No dependencies found" }) |  | ||||||
| 									return |  | ||||||
| 								end |  | ||||||
|  |  | ||||||
| 								local display = {} |  | ||||||
| 								table.insert( |  | ||||||
| 									display, |  | ||||||
| 									"Dependencies for " |  | ||||||
| 										.. package_name |  | ||||||
| 										.. " at version " |  | ||||||
| 										.. nuGetVersionToString(entry.value) |  | ||||||
| 										.. ":" |  | ||||||
| 								) |  | ||||||
| 								for dep, range in pairs(package_dependencies) do |  | ||||||
| 									table.insert(display, dep .. ": " .. range) |  | ||||||
| 								end |  | ||||||
| 								local ok, err = pcall(vim.api.nvim_buf_set_lines, bufnr, 0, -1, false, display) |  | ||||||
| 								if not ok then |  | ||||||
| 									-- If we can't set lines, the window's probably gone. Ignore. |  | ||||||
| 									return |  | ||||||
| 								end |  | ||||||
| 							end) |  | ||||||
| 						end, |  | ||||||
| 					}), |  | ||||||
| 					attach_mappings = function(_, mapping) |  | ||||||
| 						mapping("i", "<CR>", function(prompt_bufnr) |  | ||||||
| 							local selection = require("telescope.actions.state").get_selected_entry() |  | ||||||
| 							require("telescope.actions").close(prompt_bufnr) |  | ||||||
| 							update_package_version(selection.value) |  | ||||||
| 						end) |  | ||||||
| 						return true |  | ||||||
| 					end, |  | ||||||
| 				}) |  | ||||||
| 				:find() |  | ||||||
| 		end |  | ||||||
| 		local whichkey = require("which-key") |  | ||||||
| 		whichkey.add({ |  | ||||||
| 			{ "<localleader>n", desc = "NuGet" }, |  | ||||||
| 			{ "<localleader>nu", UpdateNuGetVersion, desc = "Upgrade NuGet versions" }, |  | ||||||
| 		}, { buffer = vim.api.nvim_get_current_buf() }) |  | ||||||
|  |  | ||||||
| 		vim.schedule(prefetch_dependencies) |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
| @@ -1,290 +0,0 @@ | |||||||
| vim.opt.mouse = "" |  | ||||||
| vim.opt.history = 500 |  | ||||||
| vim.opt.background = "dark" |  | ||||||
|  |  | ||||||
| vim.opt.signcolumn = "yes" |  | ||||||
|  |  | ||||||
| vim.opt.wildmenu = true |  | ||||||
| vim.opt.wildignore = vim.opt.wildignore + { "*/.git/*", "*/.hg/*", "*/.svn/*", "*/.DS_Store" } |  | ||||||
|  |  | ||||||
| vim.opt.ignorecase = true |  | ||||||
| vim.opt.smartcase = true |  | ||||||
| vim.opt.incsearch = true |  | ||||||
| vim.opt.magic = true |  | ||||||
| vim.opt.hlsearch = true |  | ||||||
|  |  | ||||||
| vim.opt.autoindent = true |  | ||||||
| vim.opt.smartindent = true |  | ||||||
|  |  | ||||||
| vim.opt.wrap = true |  | ||||||
| vim.opt.linebreak = true |  | ||||||
| vim.opt.textwidth = 500 |  | ||||||
|  |  | ||||||
| vim.opt.switchbuf = "useopen" |  | ||||||
|  |  | ||||||
| vim.opt.laststatus = 2 |  | ||||||
| -- I don't use tabs, but one day I might! |  | ||||||
| vim.opt.showtabline = 2 |  | ||||||
|  |  | ||||||
| vim.opt.langmenu = "en" |  | ||||||
|  |  | ||||||
| vim.opt.ffs = "unix" |  | ||||||
| vim.opt.encoding = "utf8" |  | ||||||
|  |  | ||||||
| -- Always show current position |  | ||||||
| vim.opt.ruler = true |  | ||||||
| vim.opt.number = true |  | ||||||
|  |  | ||||||
| -- A bit of extra margin to the left |  | ||||||
| vim.opt.foldcolumn = "1" |  | ||||||
|  |  | ||||||
| vim.opt.autoread = true |  | ||||||
| vim.opt.backup = false |  | ||||||
| vim.opt.writebackup = true |  | ||||||
| vim.opt.swapfile = false |  | ||||||
|  |  | ||||||
| vim.opt.cmdheight = 2 |  | ||||||
|  |  | ||||||
| -- Use spaces instead of tabs |  | ||||||
| vim.opt.expandtab = true |  | ||||||
| vim.opt.smarttab = true |  | ||||||
| vim.opt.shiftwidth = 4 |  | ||||||
| vim.opt.tabstop = 4 |  | ||||||
|  |  | ||||||
| vim.opt.lazyredraw = true |  | ||||||
|  |  | ||||||
| -- Show matching brackets when text indicator is on one of them |  | ||||||
| vim.opt.showmatch = true |  | ||||||
| vim.opt.mat = 2 |  | ||||||
|  |  | ||||||
| -- Turn off sound |  | ||||||
| vim.opt.errorbells = false |  | ||||||
| vim.opt.visualbell = false |  | ||||||
|  |  | ||||||
| vim.opt.timeoutlen = 500 |  | ||||||
|  |  | ||||||
| vim.opt.scrolloff = 2 |  | ||||||
|  |  | ||||||
| -- Return to last edit position when opening files |  | ||||||
| vim.api.nvim_create_autocmd("BufReadPost", { |  | ||||||
| 	pattern = "*", |  | ||||||
| 	callback = function() |  | ||||||
| 		local line = vim.fn.line |  | ||||||
| 		local last_pos = line("'\"") |  | ||||||
| 		if last_pos > 1 and last_pos <= line("$") then |  | ||||||
| 			vim.cmd("normal! g'\"") |  | ||||||
| 		end |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| -- Trim trailing whitespace on save |  | ||||||
| function CleanExtraSpaces() |  | ||||||
| 	local save_cursor = vim.api.nvim_win_get_cursor(0) |  | ||||||
| 	local old_query = vim.fn.getreg("/") |  | ||||||
| 	vim.cmd("%s/\\s\\+$//e") |  | ||||||
| 	vim.api.nvim_win_set_cursor(0, save_cursor) |  | ||||||
| 	vim.fn.setreg("/", old_query) |  | ||||||
| end |  | ||||||
| vim.api.nvim_create_autocmd("BufWritePre", { |  | ||||||
| 	pattern = { "*.fs", "*.fsi", "*.txt", "*.js", "*.py", "*.wiki", "*.sh", "*.coffee" }, |  | ||||||
| 	callback = CleanExtraSpaces, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| -- Status line |  | ||||||
|  |  | ||||||
| -- Returns true if paste mode is enabled |  | ||||||
| function HasPaste() |  | ||||||
| 	if vim.opt.paste:get() then |  | ||||||
| 		return "PASTE MODE  " |  | ||||||
| 	end |  | ||||||
| 	return "" |  | ||||||
| end |  | ||||||
|  |  | ||||||
| vim.o.statusline = vim.o.statusline .. "%{v:lua.HasPaste()}%F%m%r%h %w  Line: %l  Column: %c" |  | ||||||
|  |  | ||||||
| -------------------------------------------------------------- |  | ||||||
|  |  | ||||||
| vim.api.nvim_set_keymap("n", ";", "<Nop>", { noremap = true }) |  | ||||||
| vim.api.nvim_set_var("maplocalleader", ";") |  | ||||||
| vim.api.nvim_set_var("mapleader", " ") |  | ||||||
|  |  | ||||||
| function MarkdownPreview() |  | ||||||
| 	local temp_file = vim.fn.tempname() .. ".md" |  | ||||||
| 	local file_name = vim.fn.substitute(vim.fn.tolower(vim.fn.expand("%:t")), "\\W", "_", "g") |  | ||||||
| 	local temp_html = "/tmp/" .. file_name .. "_tmp.html" |  | ||||||
|  |  | ||||||
| 	-- Write the current buffer to the temp file |  | ||||||
| 	vim.cmd("write! " .. temp_file) |  | ||||||
|  |  | ||||||
| 	local pandoc_cmd = "pandoc " .. temp_file .. " -o " .. temp_html |  | ||||||
|  |  | ||||||
| 	-- Execute the pandoc command |  | ||||||
| 	vim.fn.system(pandoc_cmd) |  | ||||||
|  |  | ||||||
| 	-- Use tmux and lynx to preview the HTML file |  | ||||||
| 	local lynx_cmd = "tmux split-window -h lynx " .. temp_html |  | ||||||
| 	vim.fn.jobstart(vim.split(lynx_cmd, " "), { silent = true }) |  | ||||||
|  |  | ||||||
| 	-- Delete the temp markdown file |  | ||||||
| 	vim.fn.delete(temp_file, "rf") |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function RemoveCarriageReturn() |  | ||||||
| 	vim.cmd("mark m") |  | ||||||
| 	vim.cmd("normal! Hmt") |  | ||||||
| 	vim.cmd("%s/\r//ge") |  | ||||||
| 	vim.cmd("normal! 'tzt'm") |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function FormatJson() |  | ||||||
| 	vim.cmd("%!python -m json.tool") |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function ChangeToCurrentDirectory() |  | ||||||
| 	vim.cmd(":cd %:p:h") |  | ||||||
| 	vim.cmd(":pwd") |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function close_loclist_if_orphaned() |  | ||||||
| 	local win = vim.fn.expand("<afile>") |  | ||||||
| 	vim.fn.win_execute(win, "lclose") |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Set up an autocmd using the nvim_create_autocmd API |  | ||||||
| vim.api.nvim_create_autocmd("WinClosed", { |  | ||||||
| 	pattern = "*", |  | ||||||
| 	callback = close_loclist_if_orphaned, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| local whichkey = require("which-key") |  | ||||||
| local pickers = require("telescope.pickers") |  | ||||||
| local action_state = require("telescope.actions.state") |  | ||||||
| local actions = require("telescope.actions") |  | ||||||
| local finders = require("telescope.finders") |  | ||||||
| local conf = require("telescope.config").values |  | ||||||
|  |  | ||||||
| function DisplayAllMappingsWithTelescope() |  | ||||||
| 	local mappings = {} |  | ||||||
| 	local commands = {} -- Store commands keyed by the display string |  | ||||||
|  |  | ||||||
| 	local function accumulate(tree) |  | ||||||
| 		tree:walk(function(node) |  | ||||||
| 			if node.mapping then |  | ||||||
| 				local mapping = node.mapping |  | ||||||
| 				if not mapping.group then |  | ||||||
| 					local description = mapping.desc or mapping.label or mapping.cmd |  | ||||||
| 					-- Some actions are just there for which-key to hook into to display prefixes; they don't have a description. |  | ||||||
| 					if description then |  | ||||||
| 						local displayString = description .. " | " .. mapping.prefix |  | ||||||
| 						commands[displayString] = mapping.prefix |  | ||||||
| 						mappings[#mappings + 1] = displayString |  | ||||||
| 					else |  | ||||||
| 						for k, v in pairs(mapping) do |  | ||||||
| 							print("Nothing: " .. k .. " : " .. tostring(v) .. " (type: " .. type(v) .. ")") |  | ||||||
| 						end |  | ||||||
| 						print("-----") |  | ||||||
| 					end |  | ||||||
| 				end |  | ||||||
| 			end |  | ||||||
| 		end) |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local cur_buf = vim.api.nvim_win_get_buf(0) |  | ||||||
|  |  | ||||||
| 	accumulate(require("which-key.keys").get_tree("n").tree) |  | ||||||
| 	accumulate(require("which-key.keys").get_tree("n", cur_buf).tree) |  | ||||||
|  |  | ||||||
| 	pickers |  | ||||||
| 		.new({}, { |  | ||||||
| 			prompt_title = "Actions", |  | ||||||
| 			finder = finders.new_table({ |  | ||||||
| 				results = mappings, |  | ||||||
| 			}), |  | ||||||
| 			sorter = conf.generic_sorter({}), |  | ||||||
| 			attach_mappings = function(_, map) |  | ||||||
| 				map("i", "<CR>", function(bufnr) |  | ||||||
| 					local selection = action_state.get_selected_entry() |  | ||||||
| 					actions.close(bufnr) |  | ||||||
| 					local cmd = commands[selection.value] |  | ||||||
| 					if cmd then |  | ||||||
| 						vim.api.nvim_command(":normal " .. vim.api.nvim_replace_termcodes(cmd, true, true, true)) |  | ||||||
| 					else |  | ||||||
| 						print("no command found") |  | ||||||
| 					end |  | ||||||
| 				end) |  | ||||||
| 				return true |  | ||||||
| 			end, |  | ||||||
| 		}) |  | ||||||
| 		:find() |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function ToggleSpell() |  | ||||||
| 	vim.cmd("setlocal spell!") |  | ||||||
| end |  | ||||||
|  |  | ||||||
| whichkey.add({ |  | ||||||
| 	{ |  | ||||||
| 		"<localleader><localleader>", |  | ||||||
| 		function() |  | ||||||
| 			require("which-key").show({ global = false }) |  | ||||||
| 		end, |  | ||||||
| 		desc = "View all mappings", |  | ||||||
| 	}, |  | ||||||
| 	{ "<localleader>mp", MarkdownPreview, desc = "Preview Markdown in Lynx" }, |  | ||||||
| 	{ "<localleader>md", RemoveCarriageReturn, desc = "Delete carriage returns from file" }, |  | ||||||
| 	{ "<localleader>j", FormatJson, desc = "Auto-format JSON" }, |  | ||||||
| }) |  | ||||||
| whichkey.add({ |  | ||||||
| 	{ |  | ||||||
| 		"<leader>g", |  | ||||||
| 		function() |  | ||||||
| 			require("telescope.builtin").grep_string() |  | ||||||
| 		end, |  | ||||||
| 		desc = "Find instances of text under cursor", |  | ||||||
| 	}, |  | ||||||
| 	{ "<leader>h", desc = "Find historical..." }, |  | ||||||
| 	{ |  | ||||||
| 		"<leader>hf", |  | ||||||
| 		function() |  | ||||||
| 			require("telescope.builtin").oldfiles() |  | ||||||
| 		end, |  | ||||||
| 		desc = "List previously open files", |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"<leader>hc", |  | ||||||
| 		function() |  | ||||||
| 			require("telescope.builtin").command_history() |  | ||||||
| 		end, |  | ||||||
| 		desc = "List previously run commands", |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"<leader>hs", |  | ||||||
| 		function() |  | ||||||
| 			require("telescope.builtin").search_history() |  | ||||||
| 		end, |  | ||||||
| 		desc = "List previously run searches", |  | ||||||
| 	}, |  | ||||||
| 	{ |  | ||||||
| 		"<leader>m", |  | ||||||
| 		function() |  | ||||||
| 			require("telescope.builtin").marks() |  | ||||||
| 		end, |  | ||||||
| 		desc = "List marks", |  | ||||||
| 	}, |  | ||||||
| 	{ "<leader>cd", ChangeToCurrentDirectory, desc = "Switch CWD to the directory of the open buffer" }, |  | ||||||
| 	{ "<leader>ss", ToggleSpell, desc = "Toggle spell-checker on or off" }, |  | ||||||
| 	{ |  | ||||||
| 		"<leader><leader>", |  | ||||||
| 		function() |  | ||||||
| 			require("telescope.builtin").find_files() |  | ||||||
| 		end, |  | ||||||
| 		desc = "Find files by name", |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd({ "BufRead", "BufNewFile" }, { |  | ||||||
| 	pattern = { "Directory.Build.props", "*.fsproj", "*.csproj" }, |  | ||||||
| 	callback = function() |  | ||||||
| 		vim.bo.filetype = "xml" |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
| @@ -1,230 +0,0 @@ | |||||||
| vim.g["fsharp#fsautocomplete_command"] = { "fsautocomplete" } |  | ||||||
| vim.g["fsharp#show_signature_on_cursor_move"] = 1 |  | ||||||
| vim.g["fsharp#fsi_keymap"] = "none" |  | ||||||
|  |  | ||||||
| -- Supply nil to get all loaded F# projects and build them. |  | ||||||
| local function BuildFSharpProjects(projects) |  | ||||||
| 	local function on_line(data, _, context) |  | ||||||
| 		-- Keep the window alive if there were warnings |  | ||||||
| 		if string.match(data, "%s[1-9]%d* Warning%(s%)") then |  | ||||||
| 			context.warn = context.warn + 1 |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	local on_complete |  | ||||||
| 	local function spawn_next(context) |  | ||||||
| 		BuildUtils.run( |  | ||||||
| 			"dotnet", |  | ||||||
| 			{ "build", context.projects[context.completed + 1] }, |  | ||||||
| 			"dotnet build", |  | ||||||
| 			context, |  | ||||||
| 			on_line, |  | ||||||
| 			on_complete |  | ||||||
| 		) |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	on_complete = function(context, code, signal) |  | ||||||
| 		print("Build process exited with code " .. code .. " and signal " .. signal) |  | ||||||
| 		if code ~= 0 then |  | ||||||
| 			context.errs = context.errs + 1 |  | ||||||
| 		end |  | ||||||
| 		context.completed = context.completed + 1 |  | ||||||
|  |  | ||||||
| 		print( |  | ||||||
| 			"Completed: " |  | ||||||
| 				.. context.completed |  | ||||||
| 				.. " out of " |  | ||||||
| 				.. context.expected |  | ||||||
| 				.. " (errors: " |  | ||||||
| 				.. context.errs |  | ||||||
| 				.. ", warnings: " |  | ||||||
| 				.. context.warn |  | ||||||
| 				.. ")" |  | ||||||
| 		) |  | ||||||
|  |  | ||||||
| 		if context.completed == context.expected then |  | ||||||
| 			if context.errs == 0 and context.warn == 0 then |  | ||||||
| 				-- Close the temporary floating window (but keep it alive if the |  | ||||||
| 				-- cursor is in it) |  | ||||||
| 				local cur_win = vim.api.nvim_get_current_win() |  | ||||||
| 				local cur_buf = vim.api.nvim_win_get_buf(cur_win) |  | ||||||
| 				if cur_buf ~= context.buffer then |  | ||||||
| 					vim.api.nvim_win_close(context.window, true) |  | ||||||
| 				end |  | ||||||
| 				print("All builds successful") |  | ||||||
| 			end |  | ||||||
| 		else |  | ||||||
| 			spawn_next(context) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	if not projects then |  | ||||||
| 		projects = vim.fn["fsharp#getLoadedProjects"]() |  | ||||||
| 	end |  | ||||||
| 	if projects then |  | ||||||
| 		local total_projects = 0 |  | ||||||
| 		for _, _ in ipairs(projects) do |  | ||||||
| 			total_projects = total_projects + 1 |  | ||||||
| 		end |  | ||||||
| 		local context = BuildUtils.create_window() |  | ||||||
| 		context.warn = 0 |  | ||||||
| 		context.errs = 0 |  | ||||||
| 		context.completed = 0 |  | ||||||
| 		context.expected = total_projects |  | ||||||
| 		context.projects = projects |  | ||||||
|  |  | ||||||
| 		spawn_next(context) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_user_command("BuildFSharpProject", function(opts) |  | ||||||
| 	if opts.fargs and opts.fargs[1] then |  | ||||||
| 		BuildFSharpProjects(opts.fargs) |  | ||||||
| 	else |  | ||||||
| 		local pickers = require("telescope.pickers") |  | ||||||
| 		local finders = require("telescope.finders") |  | ||||||
| 		local conf = require("telescope.config").values |  | ||||||
| 		local action_state = require("telescope.actions.state") |  | ||||||
| 		local actions = require("telescope.actions") |  | ||||||
| 		pickers |  | ||||||
| 			.new({}, { |  | ||||||
| 				prompt_title = "Projects", |  | ||||||
| 				finder = finders.new_table({ |  | ||||||
| 					results = vim.fn["fsharp#getLoadedProjects"](), |  | ||||||
| 				}), |  | ||||||
| 				sorter = conf.generic_sorter({}), |  | ||||||
| 				attach_mappings = function(prompt_buf, _) |  | ||||||
| 					actions.select_default:replace(function() |  | ||||||
| 						actions.close(prompt_buf) |  | ||||||
| 						local selection = action_state.get_selected_entry() |  | ||||||
| 						BuildFSharpProjects({ selection.value }) |  | ||||||
| 					end) |  | ||||||
| 					return true |  | ||||||
| 				end, |  | ||||||
| 			}) |  | ||||||
| 			:find() |  | ||||||
| 	end |  | ||||||
| end, { nargs = "?", complete = "file" }) |  | ||||||
|  |  | ||||||
| local function TableConcat(tables) |  | ||||||
| 	local result = {} |  | ||||||
| 	for _, tab in ipairs(tables) do |  | ||||||
| 		for _, v in ipairs(tab) do |  | ||||||
| 			table.insert(result, v) |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
| 	return result |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- args is a table that will be splatted into the command line and will be immediately |  | ||||||
| -- followed by the project. |  | ||||||
| local function RunDotnet(command, args, project, configuration) |  | ||||||
| 	local function on_line(data, _, context) end |  | ||||||
|  |  | ||||||
| 	local function on_complete(context, code, signal) end |  | ||||||
|  |  | ||||||
| 	local context = BuildUtils.create_window() |  | ||||||
|  |  | ||||||
| 	BuildUtils.run( |  | ||||||
| 		"dotnet", |  | ||||||
| 		TableConcat({ { command }, args, { project, "--configuration", configuration } }), |  | ||||||
| 		"dotnet", |  | ||||||
| 		context, |  | ||||||
| 		on_line, |  | ||||||
| 		on_complete |  | ||||||
| 	) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Call this as: |  | ||||||
| -- RunFSharpProject path/to/fsproj |  | ||||||
| -- RunFSharpProject Debug path/to/fsproj |  | ||||||
| vim.api.nvim_create_user_command("RunFSharpProject", function(opts) |  | ||||||
| 	local configuration = "Release" |  | ||||||
| 	if opts.fargs and opts.fargs[1] and opts.fargs[1]:match("sproj$") then |  | ||||||
| 		RunDotnet("run", { "--project" }, opts.fargs[1], configuration) |  | ||||||
| 	elseif opts.fargs and opts.fargs[1] and opts.fargs[2] then |  | ||||||
| 		configuration = opts.fargs[1] |  | ||||||
| 		RunDotnet("run", { "--project" }, opts.fargs[2], configuration) |  | ||||||
| 	else |  | ||||||
| 		configuration = opts.fargs[1] |  | ||||||
| 		local pickers = require("telescope.pickers") |  | ||||||
| 		local finders = require("telescope.finders") |  | ||||||
| 		local conf = require("telescope.config").values |  | ||||||
| 		local action_state = require("telescope.actions.state") |  | ||||||
| 		local actions = require("telescope.actions") |  | ||||||
| 		pickers |  | ||||||
| 			.new({}, { |  | ||||||
| 				prompt_title = "Projects", |  | ||||||
| 				finder = finders.new_table({ |  | ||||||
| 					results = vim.fn["fsharp#getLoadedProjects"](), |  | ||||||
| 				}), |  | ||||||
| 				sorter = conf.generic_sorter({}), |  | ||||||
| 				attach_mappings = function(prompt_buf, _) |  | ||||||
| 					actions.select_default:replace(function() |  | ||||||
| 						actions.close(prompt_buf) |  | ||||||
| 						local selection = action_state.get_selected_entry() |  | ||||||
| 						RunDotnet("run", { "--project" }, selection.value, configuration) |  | ||||||
| 					end) |  | ||||||
| 					return true |  | ||||||
| 				end, |  | ||||||
| 			}) |  | ||||||
| 			:find() |  | ||||||
| 	end |  | ||||||
| end, { nargs = "*", complete = "file" }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_user_command("PublishFSharpProject", function(opts) |  | ||||||
| 	if opts.fargs and opts.fargs[1] then |  | ||||||
| 		RunDotnet("publish", {}, opts.fargs[1], "Release") |  | ||||||
| 	else |  | ||||||
| 		local pickers = require("telescope.pickers") |  | ||||||
| 		local finders = require("telescope.finders") |  | ||||||
| 		local conf = require("telescope.config").values |  | ||||||
| 		local action_state = require("telescope.actions.state") |  | ||||||
| 		local actions = require("telescope.actions") |  | ||||||
| 		pickers |  | ||||||
| 			.new({}, { |  | ||||||
| 				prompt_title = "Projects", |  | ||||||
| 				finder = finders.new_table({ |  | ||||||
| 					results = vim.fn["fsharp#getLoadedProjects"](), |  | ||||||
| 				}), |  | ||||||
| 				sorter = conf.generic_sorter({}), |  | ||||||
| 				attach_mappings = function(prompt_buf, _) |  | ||||||
| 					actions.select_default:replace(function() |  | ||||||
| 						actions.close(prompt_buf) |  | ||||||
| 						local selection = action_state.get_selected_entry() |  | ||||||
| 						RunDotnet("publish", {}, selection.value, "Release") |  | ||||||
| 					end) |  | ||||||
| 					return true |  | ||||||
| 				end, |  | ||||||
| 			}) |  | ||||||
| 			:find() |  | ||||||
| 	end |  | ||||||
| end, { nargs = "*", complete = "file" }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("FileType", { |  | ||||||
| 	pattern = "fsharp", |  | ||||||
| 	callback = function() |  | ||||||
| 		local status, whichkey = pcall(require, "which-key") |  | ||||||
| 		if status then |  | ||||||
| 			whichkey.add({ |  | ||||||
| 				{ "<localleader>f", desc = "F#" }, |  | ||||||
| 				{ "<localleader>ft", ":call fsharp#showTooltip()<CR>", desc = "Show F# Tooltip" }, |  | ||||||
| 				{ "<localleader>fsi", ":call fsharp#toggleFsi()<CR>", desc = "Toggle FSI (F# Interactive)" }, |  | ||||||
| 				{ "<localleader>fsl", ":call fsharp#sendLineToFsi()<cr>", desc = "Send line to FSI (F# Interactive)" }, |  | ||||||
| 				{ "<localleader>fr", desc = "Run F# project" }, |  | ||||||
| 				{ "<localleader>frd", ":RunFSharpProject Debug", desc = "Run F# project in debug configuration" }, |  | ||||||
| 				{ "<localleader>frr", ":RunFSharpProject Release", desc = "Run F# project in release configuration" }, |  | ||||||
| 				{ "<localleader>fp", ":PublishFSharpProject", desc = "Publish F# project" }, |  | ||||||
| 				{ "<localleader>fb", desc = "Build F# project" }, |  | ||||||
| 				{ "<localleader>fba", BuildFSharpProjects, desc = "Build all projects" }, |  | ||||||
| 				{ "<localleader>fbs", ":BuildFSharpProject", desc = "Build specified project" }, |  | ||||||
| 			}, { buffer = vim.api.nvim_get_current_buf() }) |  | ||||||
| 		else |  | ||||||
| 			vim.api.nvim_set_keymap("n", "<localleader>ft", ":call fsharp#showTooltip()<CR>", { noremap = true }) |  | ||||||
| 			vim.api.nvim_set_keymap("n", "<localleader>fsi", ":call fsharp#toggleFsi()<CR>", { noremap = true }) |  | ||||||
| 			vim.api.nvim_set_keymap("n", "<localleader>fsl", ":call fsharp#sendLineToFsi()<CR>", { noremap = true }) |  | ||||||
| 			vim.api.nvim_set_keymap("n", "<localleader>bpa", ":lua BuildFSharpProjects()", { noremap = true }) |  | ||||||
| 			vim.api.nvim_set_keymap("n", "<localleader>bps", ":BuildFSharpProject", { noremap = true }) |  | ||||||
| 		end |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| vim.lsp.config.leanls = { |  | ||||||
| 	cmd = { "lean-language-server", "--stdio" }, |  | ||||||
| 	root_markers = { "lean-toolchain", "lakefile.lean" }, |  | ||||||
| } |  | ||||||
| vim.lsp.enable("leanls") |  | ||||||
|  |  | ||||||
| require("lean").setup({}) |  | ||||||
|  |  | ||||||
| require("which-key").add({ |  | ||||||
| 	{ "<localleader>l", desc = "Lean" }, |  | ||||||
| 	{ "<localleader>li", "<Cmd>LeanInfoviewToggle<CR>", desc = "Toggle Lean info view" }, |  | ||||||
| 	{ "<localleader>lp", "<Cmd>LeanInfoviewPinTogglePause<CR>", desc = "Pause Lean info view" }, |  | ||||||
| 	{ "<localleader>ls", "<Cmd>LeanSorryFill<CR>", desc = "Fill open goals with sorry" }, |  | ||||||
| 	{ "<localleader>lw", "<Cmd>LeanInfoviewEnableWidgets<CR>", desc = "Enable Lean widgets" }, |  | ||||||
| 	{ "<localleader>lW", "<Cmd>LeanInfoviewDisableWidgets<CR>", desc = "Disable Lean widgets" }, |  | ||||||
| 	{ |  | ||||||
| 		"<localleader>l?", |  | ||||||
| 		"<Cmd>LeanAbbreviationsReverseLookup<CR>", |  | ||||||
| 		desc = "Show what Lean abbreviation produces the symbol under the cursor", |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
| @@ -1,196 +0,0 @@ | |||||||
| local nvim_cmp = require("cmp") |  | ||||||
|  |  | ||||||
| -- Using rustaceanvim means we shouldn't set up the LSP for Rust manually. |  | ||||||
| -- Similarly csharp_ls is unnecessary given roslyn.nvim |  | ||||||
| -- require("lspconfig")["csharp_ls"].setup({}) |  | ||||||
| local schemas = { |  | ||||||
| 	["https://raw.githubusercontent.com/docker/compose/master/compose/config/compose_spec.json"] = "docker-compose*.{yml,yaml}", |  | ||||||
| 	["https://json.schemastore.org/github-workflow.json"] = ".github/**/*.{yml,yaml}", |  | ||||||
| 	["https://json.schemastore.org/package.json"] = "package.json", |  | ||||||
| 	["https://json.schemastore.org/global.json"] = "global.json", |  | ||||||
| 	["https://raw.githubusercontent.com/dotnet/Nerdbank.GitVersioning/master/src/NerdBank.GitVersioning/version.schema.json"] = "version.json", |  | ||||||
| 	["https://json-schema.org/draft/2020-12/schema"] = "*.schema.json", |  | ||||||
| 	["https://json.schemastore.org/dotnet-tools.json"] = "dotnet-tools.json", |  | ||||||
| } |  | ||||||
|  |  | ||||||
| require("lspconfig")["clangd"].setup({}) |  | ||||||
|  |  | ||||||
| require("lspconfig")["gopls"].setup({}) |  | ||||||
|  |  | ||||||
| require("lspconfig")["yamlls"].setup({ |  | ||||||
| 	settings = { |  | ||||||
| 		yaml = { |  | ||||||
| 			validate = true, |  | ||||||
| 			-- disable the schema store |  | ||||||
| 			schemaStore = { |  | ||||||
| 				enable = false, |  | ||||||
| 				url = "", |  | ||||||
| 			}, |  | ||||||
| 			-- manually select schemas |  | ||||||
| 			schemas = schemas, |  | ||||||
| 		}, |  | ||||||
| 	}, |  | ||||||
| 	filetypes = { "yaml", "json", "jsonc" }, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| local capabilities = vim.lsp.protocol.make_client_capabilities() |  | ||||||
| capabilities = require("cmp_nvim_lsp").default_capabilities(capabilities) |  | ||||||
|  |  | ||||||
| capabilities.textDocument.completion.completionItem.snippetSupport = true |  | ||||||
| require("lspconfig")["jsonls"].setup({ |  | ||||||
| 	capabilities = capabilities, |  | ||||||
| 	cmd = { "vscode-json-language-server", "--stdio" }, |  | ||||||
| 	settings = { |  | ||||||
| 		json = { |  | ||||||
| 			validate = { enable = true }, |  | ||||||
| 		}, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| require("lspconfig")["denols"].setup({}) |  | ||||||
| require("lspconfig")["bashls"].setup({}) |  | ||||||
| require("lspconfig")["dockerls"].setup({}) |  | ||||||
| require("lspconfig")["html"].setup({ |  | ||||||
| 	capabilities = capabilities, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| require("lspconfig")["lua_ls"].setup({ |  | ||||||
| 	on_init = function(client) |  | ||||||
| 		if not client.workspace_folders then |  | ||||||
| 			return |  | ||||||
| 		end |  | ||||||
| 		local path = client.workspace_folders[1].name |  | ||||||
| 		if vim.uv.fs_stat(path .. "/.luarc.json") or vim.loop.fs_stat(path .. "/.luarc.jsonc") then |  | ||||||
| 			return |  | ||||||
| 		end |  | ||||||
|  |  | ||||||
| 		client.config.settings.Lua = vim.tbl_deep_extend("force", client.config.settings.Lua, { |  | ||||||
| 			runtime = { |  | ||||||
| 				-- Tell the language server which version of Lua you're using |  | ||||||
| 				-- (most likely LuaJIT in the case of Neovim) |  | ||||||
| 				version = "LuaJIT", |  | ||||||
| 			}, |  | ||||||
| 			-- Make the server aware of Neovim runtime files |  | ||||||
| 			workspace = { |  | ||||||
| 				checkThirdParty = false, |  | ||||||
| 				library = { |  | ||||||
| 					vim.env.VIMRUNTIME, |  | ||||||
| 					-- Depending on the usage, you might want to add additional paths here. |  | ||||||
| 					-- "${3rd}/luv/library" |  | ||||||
| 					-- "${3rd}/busted/library", |  | ||||||
| 				}, |  | ||||||
| 				-- or pull in all of 'runtimepath'. NOTE: this is a lot slower |  | ||||||
| 				-- library = vim.api.nvim_get_runtime_file("", true) |  | ||||||
| 			}, |  | ||||||
| 		}) |  | ||||||
| 	end, |  | ||||||
| 	settings = { |  | ||||||
| 		Lua = {}, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| require("lspconfig").pyright.setup({ |  | ||||||
| 	capabilities = capabilities, |  | ||||||
| 	handlers = { |  | ||||||
| 		["textDocument/publishDiagnostics"] = function(...) |  | ||||||
| 			vim.lsp.diagnostic.on_publish_diagnostics(...) |  | ||||||
|  |  | ||||||
| 			local window = vim.api.nvim_get_current_win() |  | ||||||
| 			vim.diagnostic.setloclist({ open_loclist = true }) |  | ||||||
| 			vim.api.nvim_set_current_win(window) |  | ||||||
| 		end, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| require("lspconfig").nil_ls.setup({ |  | ||||||
| 	capabilities = capabilities, |  | ||||||
| 	settings = { |  | ||||||
| 		nix = { |  | ||||||
| 			flake = { |  | ||||||
| 				autoArchive = true, |  | ||||||
| 			}, |  | ||||||
| 		}, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| function ToggleLocList() |  | ||||||
| 	local winid = vim.fn.getloclist(0, { winid = 0 }).winid |  | ||||||
| 	if winid == 0 then |  | ||||||
| 		local window = vim.api.nvim_get_current_win() |  | ||||||
| 		vim.cmd.lopen() |  | ||||||
| 		vim.api.nvim_set_current_win(window) |  | ||||||
| 	else |  | ||||||
| 		vim.cmd.lclose() |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| do |  | ||||||
| 	local whichkey_status, whichkey = pcall(require, "which-key") |  | ||||||
| 	if whichkey_status then |  | ||||||
| 		whichkey.add({ |  | ||||||
| 			{ "<leader>l", desc = "loclist-related commands" }, |  | ||||||
| 			{ "<leader>lp", vim.diagnostic.goto_prev, desc = "Go to previous entry in loclist" }, |  | ||||||
| 			{ "<leader>ln", vim.diagnostic.goto_next, desc = "Go to next entry in loclist" }, |  | ||||||
| 			{ "<leader>ll", ToggleLocList, desc = "Toggle loclist" }, |  | ||||||
| 			{ "<leader>lf", vim.diagnostic.open_float, desc = "Open current loclist entry in floating window" }, |  | ||||||
| 		}) |  | ||||||
| 	else |  | ||||||
| 		vim.keymap.set("n", "<leader>lp", vim.diagnostic.goto_prev) |  | ||||||
| 		vim.keymap.set("n", "<leader>ln", vim.diagnostic.goto_next) |  | ||||||
| 		vim.keymap.set("n", "<leader>ll", ToggleLocList) |  | ||||||
| 		vim.keymap.set("n", "<leader>lf", vim.diagnostic.open_float) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- Use LspAttach autocommand to only map the following keys |  | ||||||
| -- after the language server attaches to the current buffer |  | ||||||
| vim.api.nvim_create_autocmd("LspAttach", { |  | ||||||
| 	group = vim.api.nvim_create_augroup("UserLspConfig", {}), |  | ||||||
| 	callback = function(ev) |  | ||||||
| 		local whichkey = require("which-key") |  | ||||||
| 		-- Enable completion triggered by <c-x><c-o> |  | ||||||
| 		vim.bo[ev.buf].omnifunc = "v:lua.vim.lsp.omnifunc" |  | ||||||
|  |  | ||||||
| 		-- Buffer local mappings. |  | ||||||
| 		-- See `:help vim.lsp.*` for documentation on any of the below functions |  | ||||||
| 		whichkey.add({ |  | ||||||
| 			{ "g", desc = "Go-to related commands" }, |  | ||||||
| 			{ "gD", vim.lsp.buf.declaration, desc = "Go to declaration" }, |  | ||||||
| 			{ "gd", vim.lsp.buf.definition, desc = "Go to definition" }, |  | ||||||
| 			{ "gi", vim.lsp.buf.implementation, desc = "Go to implementation" }, |  | ||||||
| 			{ |  | ||||||
| 				"gr", |  | ||||||
| 				function() |  | ||||||
| 					require("telescope.builtin").lsp_references() |  | ||||||
| 				end, |  | ||||||
| 				desc = "Find references", |  | ||||||
| 			}, |  | ||||||
| 			{ "gK", vim.lsp.buf.hover, desc = "Display information about symbol under cursor" }, |  | ||||||
| 		}) |  | ||||||
| 		whichkey.add({ |  | ||||||
| 			{ "<C-k>", vim.lsp.buf.signature_help, desc = "Display signature information about symbol under cursor" }, |  | ||||||
| 		}) |  | ||||||
| 		whichkey.add({ |  | ||||||
| 			{ "<leader>w", desc = "Workspace-related commands" }, |  | ||||||
| 			{ "<leader>wa", vim.lsp.buf.add_workspace_folder, desc = "Add a path to the workspace folders list" }, |  | ||||||
| 			{ "<leader>wr", vim.lsp.buf.add_workspace_folder, desc = "Remove a path from the workspace folders list" }, |  | ||||||
| 			{ |  | ||||||
| 				"<leader>wl", |  | ||||||
| 				function() |  | ||||||
| 					print(vim.inspect(vim.lsp.buf.list_workspace_folders())) |  | ||||||
| 				end, |  | ||||||
| 				desc = "Show the workspace folders list", |  | ||||||
| 			}, |  | ||||||
| 			{ |  | ||||||
| 				"<leader>f", |  | ||||||
| 				function() |  | ||||||
| 					vim.lsp.buf.format({ async = true }) |  | ||||||
| 				end, |  | ||||||
| 				desc = "Autoformat", |  | ||||||
| 			}, |  | ||||||
| 			{ "<leader>ca", vim.lsp.buf.code_action, desc = "Select a code action" }, |  | ||||||
| 			{ "<leader>rn", vim.lsp.buf.rename, desc = "Rename variable" }, |  | ||||||
| 			{ "<leader>D", vim.lsp.buf.type_definition, desc = "Go to type definition" }, |  | ||||||
| 		}) |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
| @@ -1,45 +0,0 @@ | |||||||
| local cmp = require("cmp") |  | ||||||
|  |  | ||||||
| cmp.setup({ |  | ||||||
| 	snippet = { |  | ||||||
| 		-- REQUIRED - you must specify a snippet engine |  | ||||||
| 		expand = function(args) |  | ||||||
| 			vim.snippet.expand(args.body) |  | ||||||
| 		end, |  | ||||||
| 	}, |  | ||||||
| 	window = { |  | ||||||
| 		completion = cmp.config.window.bordered(), |  | ||||||
| 		documentation = cmp.config.window.bordered(), |  | ||||||
| 	}, |  | ||||||
| 	mapping = cmp.mapping.preset.insert({ |  | ||||||
| 		["<C-b>"] = cmp.mapping.scroll_docs(-4), |  | ||||||
| 		["<C-f>"] = cmp.mapping.scroll_docs(4), |  | ||||||
| 		["<C-Space>"] = cmp.mapping.complete(), |  | ||||||
| 		["<C-e>"] = cmp.mapping.abort(), |  | ||||||
| 		["<CR>"] = cmp.mapping.confirm({ select = true }), -- Accept currently selected item. Set `select` to `false` to only confirm explicitly selected items. |  | ||||||
| 	}), |  | ||||||
| 	sources = cmp.config.sources({ |  | ||||||
| 		{ name = "nvim_lsp" }, |  | ||||||
| 	}, { |  | ||||||
| 		{ name = "buffer" }, |  | ||||||
| 	}), |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| -- Use buffer source for `/` and `?` (if you enabled `native_menu`, this won't work anymore). |  | ||||||
| cmp.setup.cmdline({ "/", "?" }, { |  | ||||||
| 	mapping = cmp.mapping.preset.cmdline(), |  | ||||||
| 	sources = { |  | ||||||
| 		{ name = "buffer" }, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| -- Use cmdline & path source for ':' (if you enabled `native_menu`, this won't work anymore). |  | ||||||
| cmp.setup.cmdline(":", { |  | ||||||
| 	mapping = cmp.mapping.preset.cmdline(), |  | ||||||
| 	sources = cmp.config.sources({ |  | ||||||
| 		{ name = "path" }, |  | ||||||
| 	}, { |  | ||||||
| 		{ name = "cmdline" }, |  | ||||||
| 	}), |  | ||||||
| 	matching = { disallow_symbol_nonprefix_matching = false }, |  | ||||||
| }) |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| require("dap-python").setup("%PYTHONENV%/bin/python") |  | ||||||
|  |  | ||||||
| do |  | ||||||
| 	local whichkey = require("which-key") |  | ||||||
| 	whichkey.add({ |  | ||||||
| 		{ "<localleader>pd", desc = "Debugger-related commands" }, |  | ||||||
| 		{ "<localleader>pdt", desc = "Tests" }, |  | ||||||
| 		{ "<localleader>pdtf", require("dap-python").test_class, desc = "Run Python tests in the current file" }, |  | ||||||
| 		{ "<localleader>pdtc", require("dap-python").test_method, desc = "Run the Python test under the cursor" }, |  | ||||||
| 	}) |  | ||||||
| end |  | ||||||
| @@ -1,65 +0,0 @@ | |||||||
| local dap = require("dap") |  | ||||||
| local dap_ui = require("dap.ui.widgets") |  | ||||||
| dap.adapters.coreclr = { |  | ||||||
| 	type = "executable", |  | ||||||
| 	command = "netcoredbg", |  | ||||||
| 	args = { "--interpreter=vscode" }, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| dap.configurations.fsharp = { |  | ||||||
| 	{ |  | ||||||
| 		type = "coreclr", |  | ||||||
| 		name = "launch - netcoredbg", |  | ||||||
| 		request = "launch", |  | ||||||
| 		program = function() |  | ||||||
| 			return vim.fn.input("Path to dll: ", vim.fn.getcwd() .. "/bin/Debug/", "file") |  | ||||||
| 		end, |  | ||||||
| 	}, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| dap.configurations.cs = { |  | ||||||
| 	{ |  | ||||||
| 		type = "coreclr", |  | ||||||
| 		name = "launch - netcoredbg", |  | ||||||
| 		request = "launch", |  | ||||||
| 		program = function() |  | ||||||
| 			return vim.fn.input("Path to dll: ", vim.fn.getcwd() .. "/bin/Debug/", "file") |  | ||||||
| 		end, |  | ||||||
| 	}, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| do |  | ||||||
| 	local whichkey = require("which-key") |  | ||||||
| 	whichkey.add({ |  | ||||||
| 		{ "<localleader>d", desc = "Debugger-related commands" }, |  | ||||||
| 		{ "<localleader>do", dap.step_over, desc = "Step over" }, |  | ||||||
| 		{ "<localleader>di", dap.step_into, desc = "Step into" }, |  | ||||||
| 		{ "<localleader>dc", dap.continue, desc = "Continue" }, |  | ||||||
| 		{ "<localleader>dC", dap.run_last, desc = "Run with last debug configuration" }, |  | ||||||
| 		{ "<localleader>db", dap.toggle_breakpoint, desc = "Toggle breakpoint" }, |  | ||||||
| 		{ "<localleader>dr", dap.repl.open, desc = "Open debug repl" }, |  | ||||||
| 		{ "<localleader>dv", desc = "Commands to view debugger state" }, |  | ||||||
| 		{ |  | ||||||
| 			"<localleader>dvv", |  | ||||||
| 			function() |  | ||||||
| 				dap_ui.hover() |  | ||||||
| 			end, |  | ||||||
| 			desc = "View value of expression under cursor", |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"<localleader>dvs", |  | ||||||
| 			function() |  | ||||||
| 				dap_ui.sidebar(dap_ui.scopes).open() |  | ||||||
| 			end, |  | ||||||
| 			desc = "View values of all variables in all scopes", |  | ||||||
| 		}, |  | ||||||
| 		{ |  | ||||||
| 			"<localleader>dvf", |  | ||||||
| 			function() |  | ||||||
| 				dap_ui.sidebar(dap_ui.frames).open() |  | ||||||
| 			end, |  | ||||||
| 			desc = "View stack frames", |  | ||||||
| 		}, |  | ||||||
| 		{ "<localleader>dt", dap.terminate, desc = "Terminate/stop/end debug session" }, |  | ||||||
| 	}) |  | ||||||
| end |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| require("nvim-lightbulb").setup({ |  | ||||||
| 	autocmd = { enabled = true }, |  | ||||||
| 	ignore = { |  | ||||||
| 		clients = { |  | ||||||
| 			-- This one is really noisy |  | ||||||
| 			"lua_ls", |  | ||||||
| 		}, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
| @@ -1,64 +0,0 @@ | |||||||
| local function pytest_on_line(_, _, _) end |  | ||||||
| local function pytest_on_complete(_, code, _) |  | ||||||
| 	if code ~= 0 then |  | ||||||
| 		print("Exit code " .. code) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function RunPythonTestAtCursor() |  | ||||||
| 	local api = vim.api |  | ||||||
|  |  | ||||||
| 	-- Get the current buffer and cursor position |  | ||||||
| 	local bufnr = api.nvim_get_current_buf() |  | ||||||
| 	local line_nr = api.nvim_win_get_cursor(0)[1] |  | ||||||
| 	local filename = api.nvim_buf_get_name(bufnr) |  | ||||||
|  |  | ||||||
| 	-- Read the file content |  | ||||||
| 	local lines = api.nvim_buf_get_lines(bufnr, 0, -1, false) |  | ||||||
|  |  | ||||||
| 	-- Find the test function |  | ||||||
| 	local test_name = nil |  | ||||||
| 	for i = line_nr, 1, -1 do |  | ||||||
| 		local line = lines[i] |  | ||||||
| 		if line:match("^def test_") then |  | ||||||
| 			test_name = line:match("^def (%S+)%(") |  | ||||||
| 			break |  | ||||||
| 		end |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	if test_name then |  | ||||||
| 		-- Run pytest for the found test function |  | ||||||
| 		local context = BuildUtils.create_window() |  | ||||||
| 		BuildUtils.run( |  | ||||||
| 			"pytest", |  | ||||||
| 			{ filename .. "::" .. test_name }, |  | ||||||
| 			"Run PyTest (" .. test_name .. ")", |  | ||||||
| 			context, |  | ||||||
| 			pytest_on_line, |  | ||||||
| 			pytest_on_complete |  | ||||||
| 		) |  | ||||||
| 	else |  | ||||||
| 		print("No test function found at or above line " .. line_nr) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function RunPythonTestsInFile() |  | ||||||
| 	local file_path = vim.fn.expand("%:p") |  | ||||||
| 	local context = BuildUtils.create_window() |  | ||||||
| 	BuildUtils.run("pytest", { file_path }, "Run PyTest", context, pytest_on_line, pytest_on_complete) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function RunAllPythonTests() |  | ||||||
| 	local context = BuildUtils.create_window() |  | ||||||
| 	BuildUtils.run("pytest", {}, "Run PyTest", context, pytest_on_line, pytest_on_complete) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| do |  | ||||||
| 	local whichkey = require("which-key") |  | ||||||
| 	whichkey.add({ |  | ||||||
| 		{ "<localleader>pt", desc = "Run Python tests" }, |  | ||||||
| 		{ "<localleader>ptf", RunPythonTestsInFile, desc = "Run Python tests in the current file" }, |  | ||||||
| 		{ "<localleader>pta", RunAllPythonTests, desc = "Run all Python tests" }, |  | ||||||
| 		{ "<localleader>ptc", RunPythonTestAtCursor, desc = "Run the Python test under the cursor" }, |  | ||||||
| 	}) |  | ||||||
| end |  | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| require("roslyn").setup({ |  | ||||||
| 	on_attach = function(_, _) end, |  | ||||||
| 	capabilities = vim.lsp.protocol.make_client_capabilities(), |  | ||||||
| }) |  | ||||||
| @@ -1,8 +0,0 @@ | |||||||
| require("tokyonight").setup({ |  | ||||||
| 	style = "night", |  | ||||||
| 	on_colors = function(colors) |  | ||||||
| 		colors.border = "#565f89" |  | ||||||
| 	end, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| vim.cmd([[colorscheme tokyonight]]) |  | ||||||
| @@ -1,9 +0,0 @@ | |||||||
| require("nvim-treesitter.configs").setup({ |  | ||||||
| 	-- Automatically install missing parsers when entering buffer |  | ||||||
| 	-- Recommendation: set to false if you don't have `tree-sitter` CLI installed locally |  | ||||||
| 	auto_install = false, |  | ||||||
|  |  | ||||||
| 	highlight = { |  | ||||||
| 		enable = true, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
| @@ -1,99 +0,0 @@ | |||||||
| local venv_selector = require("venv-selector") |  | ||||||
|  |  | ||||||
| venv_selector.setup({ |  | ||||||
| 	changed_venv_hooks = { venv_selector.hooks.pyright }, |  | ||||||
| 	name = { "venv", ".venv" }, |  | ||||||
| 	search_venv_managers = true, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| vim.api.nvim_create_autocmd("VimEnter", { |  | ||||||
| 	desc = "Auto select virtualenv Nvim open", |  | ||||||
| 	pattern = "*", |  | ||||||
| 	callback = function() |  | ||||||
| 		-- Mystery: this seems to be being called twice whenever we open nvim |  | ||||||
| 		local venv = vim.fn.findfile("pyproject.toml", vim.fn.getcwd() .. ";") |  | ||||||
| 		if venv ~= "" then |  | ||||||
| 			require("venv-selector").retrieve_from_cache() |  | ||||||
| 		end |  | ||||||
| 	end, |  | ||||||
| 	once = true, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| function SelectVenv() |  | ||||||
| 	local old_path = vim.fn.getenv("PATH") |  | ||||||
| 	vim.cmd("VenvSelectCached") |  | ||||||
| 	local new_path = vim.fn.getenv("PATH") |  | ||||||
| 	if old_path == new_path then |  | ||||||
| 		-- Failed to source venv. Get the user to choose one. |  | ||||||
| 		vim.cmd("VenvSelect") |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| local function find_requirements_txt(start_path) |  | ||||||
| 	local path = vim.fn.fnamemodify(start_path, ":p") |  | ||||||
| 	while path and #path > 1 do |  | ||||||
| 		local req_path = path .. "requirements.txt" |  | ||||||
| 		if vim.fn.filereadable(req_path) ~= 0 then |  | ||||||
| 			return req_path |  | ||||||
| 		end |  | ||||||
| 		path = vim.fn.fnamemodify(path, ":h") |  | ||||||
| 	end |  | ||||||
| 	return nil |  | ||||||
| end |  | ||||||
|  |  | ||||||
| -- TODO: make this one work |  | ||||||
| local function load_venv(venv_dir) |  | ||||||
| 	require("venv-selector.venv").load() |  | ||||||
| 	require("venv-selector.venv").set_venv_and_system_paths(venv_dir) |  | ||||||
| 	require("venv-selector.venv").cache_venv(venv_dir) |  | ||||||
| end |  | ||||||
|  |  | ||||||
| function CreateVenv() |  | ||||||
| 	local requirements_path = find_requirements_txt(vim.fn.getcwd()) |  | ||||||
| 	local venv_dir |  | ||||||
| 	if not requirements_path then |  | ||||||
| 		print("requirements.txt not found; creating fresh venv in current working directory.") |  | ||||||
| 		venv_dir = vim.fn.getcwd() .. "/.venv" |  | ||||||
| 	else |  | ||||||
| 		venv_dir = vim.fn.fnamemodify(requirements_path, ":h") .. "/.venv" |  | ||||||
| 	end |  | ||||||
|  |  | ||||||
| 	print("Creating virtual environment in " .. venv_dir) |  | ||||||
|  |  | ||||||
| 	-- Create virtual environment |  | ||||||
| 	vim.fn.system("python -m venv " .. vim.fn.shellescape(venv_dir)) |  | ||||||
|  |  | ||||||
| 	-- Install requirements |  | ||||||
| 	if requirements_path then |  | ||||||
| 		print("Installing requirements from " .. requirements_path) |  | ||||||
| 		local context = BuildUtils.create_window() |  | ||||||
| 		BuildUtils.run( |  | ||||||
| 			venv_dir .. "/bin/python", |  | ||||||
| 			{ "-m", "pip", "install", "-r", requirements_path }, |  | ||||||
| 			"venv creation", |  | ||||||
| 			context, |  | ||||||
| 			function(_, _, _) end, |  | ||||||
| 			function(_, _, _) |  | ||||||
| 				load_venv(venv_dir) |  | ||||||
| 			end |  | ||||||
| 		) |  | ||||||
| 	else |  | ||||||
| 		load_venv(venv_dir) |  | ||||||
| 	end |  | ||||||
| end |  | ||||||
|  |  | ||||||
| do |  | ||||||
| 	local whichkey = require("which-key") |  | ||||||
| 	whichkey.add({ |  | ||||||
| 		{ "<localleader>pv", desc = "Python virtual environment-related commands" }, |  | ||||||
| 		{ "<localleader>pvc", CreateVenv, desc = "Create virtual environment" }, |  | ||||||
| 		{ "<localleader>pvl", SelectVenv, desc = "Load virtual environment" }, |  | ||||||
| 		{ |  | ||||||
| 			"<localleader>pvo", |  | ||||||
| 			function() |  | ||||||
| 				vim.cmd("VenvSelect") |  | ||||||
| 			end, |  | ||||||
| 			desc = "Choose (override) new virtual environment", |  | ||||||
| 		}, |  | ||||||
| 	}) |  | ||||||
| end |  | ||||||
| @@ -1,40 +0,0 @@ | |||||||
| require("which-key").setup({ |  | ||||||
| 	plugins = { |  | ||||||
| 		marks = true, -- shows a list of your marks on ' and ` |  | ||||||
| 		registers = true, -- shows your registers on " in NORMAL or <C-r> in INSERT mode |  | ||||||
| 		-- the presets plugin, adds help for a bunch of default keybindings in Neovim |  | ||||||
| 		-- No actual key bindings are created |  | ||||||
| 		spelling = { |  | ||||||
| 			enabled = true, -- enabling this will show WhichKey when pressing z= to select spelling suggestions |  | ||||||
| 			suggestions = 20, -- how many suggestions should be shown in the list? |  | ||||||
| 		}, |  | ||||||
| 		presets = { |  | ||||||
| 			operators = true, -- adds help for operators like d, y, ... |  | ||||||
| 			motions = true, -- adds help for motions |  | ||||||
| 			text_objects = true, -- help for text objects triggered after entering an operator |  | ||||||
| 			windows = true, -- default bindings on <c-w> |  | ||||||
| 			nav = true, -- misc bindings to work with windows |  | ||||||
| 			z = true, -- bindings for folds, spelling and others prefixed with z |  | ||||||
| 			g = true, -- bindings for prefixed with g |  | ||||||
| 		}, |  | ||||||
| 	}, |  | ||||||
| 	icons = { |  | ||||||
| 		breadcrumb = "»", -- symbol used in the command line area that shows your active key combo |  | ||||||
| 		separator = "➜", -- symbol used between a key and it's label |  | ||||||
| 		group = "+", -- symbol prepended to a group |  | ||||||
| 	}, |  | ||||||
| 	layout = { |  | ||||||
| 		height = { min = 4, max = 25 }, -- min and max height of the columns |  | ||||||
| 		width = { min = 20, max = 50 }, -- min and max width of the columns |  | ||||||
| 		spacing = 3, -- spacing between columns |  | ||||||
| 		align = "left", -- align columns left, center or right |  | ||||||
| 	}, |  | ||||||
| 	show_help = true, -- show a help message in the command line for using WhichKey |  | ||||||
| 	show_keys = true, -- show the currently pressed key and its label as a message in the command line |  | ||||||
| 	-- disable the WhichKey popup for certain buf types and file types. |  | ||||||
| 	-- Disabled by default for Telescope |  | ||||||
| 	disable = { |  | ||||||
| 		buftypes = {}, |  | ||||||
| 		filetypes = {}, |  | ||||||
| 	}, |  | ||||||
| }) |  | ||||||
| @@ -1,13 +0,0 @@ | |||||||
| output Unknown-1 scale 2 |  | ||||||
| input * { |  | ||||||
|     xkb_layout "gb" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| # capture all screens to clipboard     |  | ||||||
| bindsym Shift+Print exec @@GRIM@@ - | @@WL-COPY@@ |  | ||||||
|      |  | ||||||
| # capture the specified screen area to clipboard     |  | ||||||
| bindsym Shift+Alt+Print exec @@GRIM@@ -g "$(@@SLURP@@)" - | @@WL-COPY@@ |  | ||||||
|      |  | ||||||
| # capture the focused monitor to clipboard     |  | ||||||
| bindsym Shift+Control+Print exec @@GRIM@@ -o $(swaymsg -t get_outputs | jq -r '.[] | select(.focused) | .name') - | @@WL-COPY@@ |  | ||||||
| @@ -12,9 +12,8 @@ with pkgs.vscode-extensions; | |||||||
|     rust-lang.rust-analyzer |     rust-lang.rust-analyzer | ||||||
|     github.vscode-pull-request-github |     github.vscode-pull-request-github | ||||||
|     shardulm94.trailing-spaces |     shardulm94.trailing-spaces | ||||||
|  |     nvarner.typst-lsp | ||||||
|     arrterian.nix-env-selector |     arrterian.nix-env-selector | ||||||
|     # Doesn't build on arm64 |  | ||||||
|     # vadimcn.vscode-lldb |  | ||||||
|   ] |   ] | ||||||
|   ++ pkgs.vscode-utils.extensionsFromVscodeMarketplace [ |   ++ pkgs.vscode-utils.extensionsFromVscodeMarketplace [ | ||||||
|     { |     { | ||||||
| @@ -84,3 +83,23 @@ with pkgs.vscode-extensions; | |||||||
|       sha256 = "sha256-lLLa8SN+Sf9Tbi7HeWYWa2KhPQFJyQWrf9l3EUljwYo="; |       sha256 = "sha256-lLLa8SN+Sf9Tbi7HeWYWa2KhPQFJyQWrf9l3EUljwYo="; | ||||||
|     } |     } | ||||||
|   ] |   ] | ||||||
|  |   ++ [ | ||||||
|  |     (let | ||||||
|  |       vsix = builtins.fetchurl { | ||||||
|  |         name = "vadimcn-vscode-lldb.zip"; | ||||||
|  |         url = "https://github.com/vadimcn/codelldb/releases/download/v1.9.0/codelldb-aarch64-darwin.vsix"; | ||||||
|  |         sha256 = "sha256:1kxrxxlzasa9jl73lqh3n36fzpdgh2hbxpzp8fk6xyzcc5vm9zfb"; | ||||||
|  |       }; | ||||||
|  |     in | ||||||
|  |       pkgs.vscode-utils.buildVscodeExtension | ||||||
|  |       { | ||||||
|  |         vsix = vsix; | ||||||
|  |         src = vsix; | ||||||
|  |         vscodeExtPublisher = "vadimcn"; | ||||||
|  |         vscodeExtName = "vscode-lldb"; | ||||||
|  |         vscodeExtUniqueId = "vadimcn-vscode-lldb"; | ||||||
|  |         publisher = "vadimcn"; | ||||||
|  |         version = "1.9.0"; | ||||||
|  |         name = "vadimcn-vscode-lldb-1.9.0"; | ||||||
|  |       }) | ||||||
|  |   ] | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								mbsync.nix
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								mbsync.nix
									
									
									
									
									
								
							| @@ -1,11 +0,0 @@ | |||||||
| {pkgs}: |  | ||||||
| pkgs.buildEnv { |  | ||||||
|   name = "isync-oauth2"; |  | ||||||
|   paths = [pkgs.isync]; |  | ||||||
|   pathsToLink = ["/bin"]; |  | ||||||
|   nativeBuildInputs = [pkgs.makeWrapper]; |  | ||||||
|   postBuild = '' |  | ||||||
|     wrapProgram "$out/bin/mbsync" \ |  | ||||||
|       --prefix SASL_PATH : "${pkgs.cyrus_sasl}/lib/sasl2:${pkgs.cyrus-sasl-xoauth2}/lib/sasl2" |  | ||||||
|   ''; |  | ||||||
| } |  | ||||||
							
								
								
									
										10
									
								
								overlays.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								overlays.nix
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | |||||||
|  | [ | ||||||
|  |   (self: super: { | ||||||
|  |     # https://github.com/NixOS/nixpkgs/issues/153304 | ||||||
|  |     alacritty = super.alacritty.overrideAttrs ( | ||||||
|  |       o: rec { | ||||||
|  |         doCheck = false; | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |   }) | ||||||
|  | ] | ||||||
							
								
								
									
										14
									
								
								python.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								python.nix
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | {pkgs}: let | ||||||
|  |   my-python-packages = python-packages: | ||||||
|  |     with python-packages; [ | ||||||
|  |       pip | ||||||
|  |       mathlibtools | ||||||
|  |     ]; | ||||||
|  | in let | ||||||
|  |   packageOverrides = self: super: { | ||||||
|  |     # Test failures on darwin ("windows-1252"); just skip pytest | ||||||
|  |     # (required for elan) | ||||||
|  |     beautifulsoup4 = super.beautifulsoup4.overridePythonAttrs (old: {pytestCheckPhase = "true";}); | ||||||
|  |   }; | ||||||
|  | in | ||||||
|  |   (pkgs.python3.override {inherit packageOverrides;}).withPackages my-python-packages | ||||||
							
								
								
									
										2
									
								
								server-home.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								server-home.nix
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | {nixpkgs, ...}: { | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user