mirror of
				https://github.com/Smaug123/nix-dotfiles
				synced 2025-10-26 08:38:40 +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" | ||||
|     steps: | ||||
|       - name: "Checkout" | ||||
|         uses: "actions/checkout@v5" | ||||
|         uses: "actions/checkout@v4" | ||||
|       - 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 }}" } | ||||
|       - name: "Check flake" | ||||
|         run: "nix flake check --all-systems" | ||||
|  | ||||
|   all-required-checks-complete: | ||||
|     if:  ${{ always() }} | ||||
|     runs-on: "ubuntu-latest" | ||||
|     steps: | ||||
|       - uses: Smaug123/all-required-checks-complete-action@05b40a8c47ef0b175ea326e9abb09802cb67b44e | ||||
|         with: | ||||
|           needs-context: ${{ toJSON(needs) }} | ||||
|     needs: [ "flake-check" ] | ||||
|       - run: "echo \"All required checks complete.\"" | ||||
|     needs: | ||||
|       - "flake-check" | ||||
|   | ||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,4 +3,3 @@ result | ||||
| bin/ | ||||
| obj/ | ||||
| .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) | ||||
|  | ||||
| Copyright (c) 2024 Patrick Stevens | ||||
| Copyright (c) 2016 Amir Salihefendic | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
|   | ||||
| @@ -1,11 +1,17 @@ | ||||
| {pkgs, ...}: let | ||||
|   mbsync = import ./mbsync.nix {inherit pkgs;}; | ||||
|   python = import ./python.nix {inherit pkgs;}; | ||||
| in { | ||||
|   nix.useDaemon = true; | ||||
|  | ||||
|   # List packages installed in system profile. To search by name, run: | ||||
|   # $ nix-env -qaP | grep wget | ||||
|  | ||||
|   environment.systemPackages = [ | ||||
|     pkgs.python3 | ||||
|     pkgs.alacritty | ||||
|     pkgs.rustup | ||||
|     pkgs.libiconv | ||||
|     pkgs.clang | ||||
|     python | ||||
|   ]; | ||||
|  | ||||
|   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. | ||||
|   programs.zsh.enable = true; | ||||
|   programs.gnupg.agent.enable = true; | ||||
|  | ||||
|   # Use a custom configuration.nix location. | ||||
|   # $ darwin-rebuild switch -I darwin-config=$HOME/.config/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. | ||||
|   services.nix-daemon.enable = true; | ||||
|   nix.package = pkgs.nixVersions.stable; | ||||
|   nix.gc.automatic = true; | ||||
|   nix.nixPath = ["darwin=/nix/store/zq4v3pi2wsfsrjkpk71kcn8srhbwjabf-nix-darwin"]; | ||||
|  | ||||
|   # Sandbox causes failure: https://github.com/NixOS/nix/issues/4119 | ||||
|   nix.settings.sandbox = false; | ||||
|  | ||||
|   # Optimising store leads to transient build failures https://github.com/NixOS/nix/issues/7273 | ||||
|   nix.extraOptions = '' | ||||
|     auto-optimise-store = false | ||||
|     auto-optimise-store = true | ||||
|     experimental-features = nix-command flakes | ||||
|     extra-experimental-features = ca-derivations | ||||
|     max-jobs = auto  # Allow building multiple derivations in parallel | ||||
| @@ -127,6 +50,4 @@ in { | ||||
|   # Used for backwards compatibility, please read the changelog before changing. | ||||
|   # $ darwin-rebuild changelog | ||||
|   system.stateVersion = 4; | ||||
|  | ||||
|   system.primaryUser = "patrick"; | ||||
| } | ||||
|   | ||||
							
								
								
									
										189
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										189
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							| @@ -3,14 +3,15 @@ | ||||
|     "apple-silicon": { | ||||
|       "inputs": { | ||||
|         "flake-compat": "flake-compat", | ||||
|         "nixpkgs": "nixpkgs" | ||||
|         "nixpkgs": "nixpkgs", | ||||
|         "rust-overlay": "rust-overlay" | ||||
|       }, | ||||
|       "locked": { | ||||
|         "lastModified": 1759818599, | ||||
|         "narHash": "sha256-4Go3gVl3E+geWMcFQ+06qlkO/lJlSvS9dyhYiXLWYq0=", | ||||
|         "lastModified": 1705557527, | ||||
|         "narHash": "sha256-DuxxHTQ/W5KToFLWG4FUF8hLldNo9eXlbt7JgvhrMnY=", | ||||
|         "owner": "tpwrules", | ||||
|         "repo": "nixos-apple-silicon", | ||||
|         "rev": "24ab28e47b586f741910b3a2f0428f3523a0fff3", | ||||
|         "rev": "6e324ab06cb27a19409ebc1dc2664bf1e585490a", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -26,11 +27,11 @@ | ||||
|         ] | ||||
|       }, | ||||
|       "locked": { | ||||
|         "lastModified": 1760721282, | ||||
|         "narHash": "sha256-aAHphQbU9t/b2RRy2Eb8oMv+I08isXv2KUGFAFn7nCo=", | ||||
|         "lastModified": 1705915768, | ||||
|         "narHash": "sha256-+Jlz8OAqkOwJlioac9wtpsCnjgGYUhvLpgJR/5tP9po=", | ||||
|         "owner": "lnl7", | ||||
|         "repo": "nix-darwin", | ||||
|         "rev": "c3211fcd0c56c11ff110d346d4487b18f7365168", | ||||
|         "rev": "1e706ef323de76236eb183d7784f3bd57255ec0b", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -40,13 +41,35 @@ | ||||
|         "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": { | ||||
|       "locked": { | ||||
|         "lastModified": 1746162366, | ||||
|         "narHash": "sha256-5SSSZ/oQkwfcAz/o/6TlejlVGqeK08wyREBQ5qFFPhM=", | ||||
|         "lastModified": 1688025799, | ||||
|         "narHash": "sha256-ktpB4dRtnksm9F5WawoIkEneh1nrEvuxb5lJFt1iOyw=", | ||||
|         "owner": "nix-community", | ||||
|         "repo": "flake-compat", | ||||
|         "rev": "0f158086a2ecdbb138cd0429410e44994f1b7e4b", | ||||
|         "rev": "8bf105319d44f6b9f0d764efa4fdef9f1cc9ba1c", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -60,11 +83,29 @@ | ||||
|         "systems": "systems" | ||||
|       }, | ||||
|       "locked": { | ||||
|         "lastModified": 1731533236, | ||||
|         "narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=", | ||||
|         "lastModified": 1705309234, | ||||
|         "narHash": "sha256-uNRRNRKmJyCRC/8y1RqBkqWBLM034y4qN7EprSdmgyA=", | ||||
|         "owner": "numtide", | ||||
|         "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" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -80,11 +121,11 @@ | ||||
|         ] | ||||
|       }, | ||||
|       "locked": { | ||||
|         "lastModified": 1760887455, | ||||
|         "narHash": "sha256-/xU8iYZjolWbMUNBQF6af5zgGs73Qw21WMgz1tLs3Yw=", | ||||
|         "lastModified": 1706134977, | ||||
|         "narHash": "sha256-KwNb1Li3K6vuVwZ77tFjZ89AWBo7AiCs9t0Cens4BsM=", | ||||
|         "owner": "nix-community", | ||||
|         "repo": "home-manager", | ||||
|         "rev": "aeabc1ac63e6ebb8ba4714c4abdfe0556f2de765", | ||||
|         "rev": "6359d40f6ec0b72a38e02b333f343c3d4929ec10", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -107,27 +148,59 @@ | ||||
|     }, | ||||
|     "nixpkgs": { | ||||
|       "locked": { | ||||
|         "lastModified": 1759756860, | ||||
|         "narHash": "sha256-7QLUQZein1TbpouPOAsTka8vQLujHRmmTqze2IkdrqY=", | ||||
|         "owner": "NixOS", | ||||
|         "lastModified": 1705316053, | ||||
|         "narHash": "sha256-J2Ey5mPFT8gdfL2XC0JTZvKaBw/b2pnyudEXFvl+dQM=", | ||||
|         "owner": "nixos", | ||||
|         "repo": "nixpkgs", | ||||
|         "rev": "d3baaf296366efdea6737124f05b65d1cf25fa7c", | ||||
|         "rev": "c3e128f3c0ecc1fb04aef9f72b3dcc2f6cecf370", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
|         "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", | ||||
|         "type": "github" | ||||
|       } | ||||
|     }, | ||||
|     "nixpkgs_2": { | ||||
|       "locked": { | ||||
|         "lastModified": 1760872779, | ||||
|         "narHash": "sha256-c5C907Raf9eY8f1NUXYeju9aUDlm227s/V0OptEbypA=", | ||||
|         "lastModified": 1706006310, | ||||
|         "narHash": "sha256-nDPz0fj0IFcDhSTlXBU2aixcnGs2Jm4Zcuoj0QtmiXQ=", | ||||
|         "owner": "NixOS", | ||||
|         "repo": "nixpkgs", | ||||
|         "rev": "63bdb5d90fa2fa11c42f9716ad1e23565613b07c", | ||||
|         "rev": "b43bb235efeab5324c5e486882ef46749188eee2", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -139,11 +212,11 @@ | ||||
|     }, | ||||
|     "nixpkgs_3": { | ||||
|       "locked": { | ||||
|         "lastModified": 1734254970, | ||||
|         "narHash": "sha256-yZzYWWWeOqSFvIirHzY1SJiuSBPmXIYJFhyt+1zkv8A=", | ||||
|         "lastModified": 1695033101, | ||||
|         "narHash": "sha256-RQ4m+ycjdLdass7Hr4+Lzwnjw7wGhcUkKqWiJS3YxPM=", | ||||
|         "owner": "nixos", | ||||
|         "repo": "nixpkgs", | ||||
|         "rev": "d388ee0ec8c623389ab3a7caead258a94cec14de", | ||||
|         "rev": "d941d9491804e0ca01e03468dbf6f8d3a7919a16", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -156,11 +229,50 @@ | ||||
|       "inputs": { | ||||
|         "apple-silicon": "apple-silicon", | ||||
|         "darwin": "darwin", | ||||
|         "emacs": "emacs", | ||||
|         "home-manager": "home-manager", | ||||
|         "nixpkgs": "nixpkgs_2", | ||||
|         "sops-nix": "sops-nix", | ||||
|         "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": { | ||||
|       "locked": { | ||||
|         "lastModified": 1681028828, | ||||
| @@ -176,18 +288,33 @@ | ||||
|         "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": { | ||||
|       "inputs": { | ||||
|         "flake-utils": "flake-utils", | ||||
|         "flake-utils": "flake-utils_2", | ||||
|         "model": "model", | ||||
|         "nixpkgs": "nixpkgs_3" | ||||
|       }, | ||||
|       "locked": { | ||||
|         "lastModified": 1743962136, | ||||
|         "narHash": "sha256-YsKxkEGqGE+c0L+k8Vczq9UHpzSktR9/tm3zrF7abzo=", | ||||
|         "lastModified": 1704121968, | ||||
|         "narHash": "sha256-N8FJb+ohJ4Qt/m5RoAbwm3RP4VRjl+hA6PUCfjPhZo8=", | ||||
|         "owner": "Smaug123", | ||||
|         "repo": "whisper.cpp", | ||||
|         "rev": "e1faa8b19ead213f507dbabda45ac54b8765a6eb", | ||||
|         "rev": "04f8e0cdc73abe7c593b2c9405f0f590c51de95a", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
|   | ||||
							
								
								
									
										62
									
								
								flake.nix
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								flake.nix
									
									
									
									
									
								
							| @@ -14,6 +14,14 @@ | ||||
|       # url = "github:Smaug123/nix-darwin/extract"; | ||||
|       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 = { | ||||
|       url = "github:tpwrules/nixos-apple-silicon"; | ||||
|     }; | ||||
| @@ -23,19 +31,23 @@ | ||||
|   }; | ||||
|  | ||||
|   outputs = { | ||||
|     self, | ||||
|     darwin, | ||||
|     emacs, | ||||
|     nixpkgs, | ||||
|     home-manager, | ||||
|     sops-nix, | ||||
|     apple-silicon, | ||||
|     whisper, | ||||
|     ... | ||||
|   }: let | ||||
|   } @ inputs: let | ||||
|     config = { | ||||
|       # contentAddressedByDefault = true; | ||||
|       allowUnfree = true; | ||||
|     }; | ||||
|     systems = ["aarch64-darwin" "aarch64-linux" "x86_64-linux"]; | ||||
|   in let | ||||
|     overlays = [emacs.overlay] ++ import ./overlays.nix; | ||||
|     recursiveMerge = attrList: let | ||||
|       f = attrPath: | ||||
|         builtins.zipAttrsWith (n: values: | ||||
| @@ -50,40 +62,10 @@ | ||||
|       f [] attrList; | ||||
|   in { | ||||
|     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 | ||||
|         system = "aarch64-linux"; | ||||
|       in let | ||||
|         pkgs = import nixpkgs {inherit system config;}; | ||||
|         pkgs = import nixpkgs {inherit system config overlays;}; | ||||
|       in | ||||
|         nixpkgs.lib.nixosSystem { | ||||
|           inherit system; | ||||
| @@ -92,9 +74,6 @@ | ||||
|               nixpkgs = pkgs; | ||||
|               username = "patrick"; | ||||
|               dotnet = pkgs.dotnet-sdk_8; | ||||
|               mbsync = import ./mbsync.nix {inherit pkgs;}; | ||||
|               secretsPath = "/home/patrick/.secrets/"; | ||||
|               machinename = "earthworm"; | ||||
|             }; | ||||
|           in [ | ||||
|             ./home-manager/earthworm-config.nix | ||||
| @@ -103,7 +82,7 @@ | ||||
|             { | ||||
|               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)]; | ||||
|               home-manager.users.patrick = recursiveMerge [(import ./home-manager/earthworm.nix args) (import ./home-manager/home.nix args)]; | ||||
|             } | ||||
|           ]; | ||||
|         }; | ||||
| @@ -111,7 +90,7 @@ | ||||
|     darwinConfigurations = let | ||||
|       system = "aarch64-darwin"; | ||||
|     in let | ||||
|       pkgs = import nixpkgs {inherit system config;}; | ||||
|       pkgs = import nixpkgs {inherit system config overlays;}; | ||||
|     in { | ||||
|       nixpkgs = pkgs; | ||||
|       patrick = darwin.lib.darwinSystem { | ||||
| @@ -122,12 +101,10 @@ | ||||
|             username = "patrick"; | ||||
|             dotnet = pkgs.dotnet-sdk_8; | ||||
|             whisper = whisper.packages.${system}; | ||||
|             mbsync = import ./mbsync.nix {inherit pkgs;}; | ||||
|             secretsPath = "/Users/patrick/.secrets/"; | ||||
|             machinename = "darwin"; | ||||
|           }; | ||||
|         in [ | ||||
|           ./darwin-configuration.nix | ||||
|           sops-nix.nixosModules.sops | ||||
|           home-manager.darwinModules.home-manager | ||||
|           { | ||||
|             home-manager.useGlobalPkgs = true; | ||||
| @@ -144,12 +121,11 @@ | ||||
|         pkgs.stdenvNoCC.mkDerivation { | ||||
|           name = "fmt-check"; | ||||
|           src = ./.; | ||||
|           nativeBuildInputs = [pkgs.alejandra pkgs.shellcheck pkgs.shfmt pkgs.stylua]; | ||||
|           nativeBuildInputs = [pkgs.alejandra pkgs.shellcheck pkgs.shfmt]; | ||||
|           checkPhase = '' | ||||
|             find . -type f -name '*.sh' | xargs shfmt -d -s -i 2 -ci | ||||
|             alejandra -c . | ||||
|             find . -type f -name '*.sh' -exec shellcheck -x {} \; | ||||
|             find . -type f -name '*.lua' -exec stylua --check {} \; | ||||
|           ''; | ||||
|           installPhase = "mkdir $out"; | ||||
|           dontBuild = true; | ||||
| @@ -167,7 +143,7 @@ | ||||
|           pkgs = import nixpkgs {inherit config system;}; | ||||
|         in { | ||||
|           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 = { | ||||
|     profiles.default = { | ||||
|       userSettings = { | ||||
|         "lean.leanpkgPath" = "/Users/${username}/.elan/toolchains/stable/bin/leanpkg"; | ||||
|         "lean.executablePath" = "/Users/${username}/.elan/toolchains/stable/bin/lean"; | ||||
|         "lean.memoryLimit" = 16384; | ||||
|         "latex-workshop.view.pdf.viewer" = "tab"; | ||||
|         "lean4.toolchainPath" = "/Users/${username}/.elan/toolchains/leanprover--lean4---nightly-2022-12-16"; | ||||
|         "git.openRepositoryInParentFolders" = "always"; | ||||
|       }; | ||||
|     userSettings = { | ||||
|       "lean.leanpkgPath" = "/Users/${username}/.elan/toolchains/stable/bin/leanpkg"; | ||||
|       "lean.executablePath" = "/Users/${username}/.elan/toolchains/stable/bin/lean"; | ||||
|       "lean.memoryLimit" = 16384; | ||||
|       "latex-workshop.view.pdf.viewer" = "tab"; | ||||
|       "lean4.toolchainPath" = "/Users/${username}/.elan/toolchains/leanprover--lean4---nightly-2022-12-16"; | ||||
|     }; | ||||
|   }; | ||||
|  | ||||
|   | ||||
| @@ -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 = [ | ||||
|     ../hardware/earthworm.nix | ||||
|   ]; | ||||
|  | ||||
|   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"; | ||||
|       } | ||||
|     ]; | ||||
|   }; | ||||
|   hardware.asahi.peripheralFirmwareDirectory = ../firmware; | ||||
|  | ||||
|   boot.loader.systemd-boot.enable = true; | ||||
|   boot.loader.efi.canTouchEfiVariables = false; | ||||
| @@ -51,10 +18,6 @@ | ||||
|   networking = { | ||||
|     hostName = "earthworm"; | ||||
|     networkmanager.enable = true; | ||||
|     wireless.iwd = { | ||||
|       enable = true; | ||||
|       settings.General.EnableNetworkConfiguration = true; | ||||
|     }; | ||||
|   }; | ||||
|  | ||||
|   time.timeZone = "Europe/London"; | ||||
| @@ -75,7 +38,7 @@ | ||||
|   ]; | ||||
|  | ||||
|   environment.loginShellInit = '' | ||||
|     [[ "$(tty)" == /dev/tty1 ]] && export WLR_RENDER_NO_EXPLICIT_SYNC=1 && sway | ||||
|     [[ "$(tty)" == /dev/tty1 ]] && sway | ||||
|   ''; | ||||
|  | ||||
|   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, | ||||
|   machinename, | ||||
|   username, | ||||
|   mbsync, | ||||
|   dotnet, | ||||
|   secretsPath, | ||||
|   ... | ||||
| }: { | ||||
|   # Let Home Manager install and manage itself. | ||||
| @@ -24,55 +21,64 @@ | ||||
|   # changes in each release. | ||||
|   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 = [ | ||||
|     # ./modules/agda.nix | ||||
|     # ./modules/emacs.nix | ||||
|     ./modules/ghostty.nix | ||||
|     ./modules/direnv.nix | ||||
|     ./modules/tmux.nix | ||||
|     ./modules/zsh.nix | ||||
|     ./modules/ripgrep.nix | ||||
|     ./modules/rust.nix | ||||
|     ./modules/posix-sh.nix | ||||
|     (import ./modules/mail.nix | ||||
|       { | ||||
|         inherit mbsync secretsPath; | ||||
|         pkgs = nixpkgs; | ||||
|       }) | ||||
|   ]; | ||||
|   programs.zsh = { | ||||
|     enable = true; | ||||
|     autocd = true; | ||||
|     enableAutosuggestions = true; | ||||
|     enableCompletion = true; | ||||
|     history = { | ||||
|       expireDuplicatesFirst = true; | ||||
|     }; | ||||
|     oh-my-zsh = { | ||||
|       enable = true; | ||||
|       plugins = ["git" "macos" "dircycle" "timer"]; | ||||
|       theme = "robbyrussell"; | ||||
|     }; | ||||
|     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 = { | ||||
|     enable = true; | ||||
|     enableZshIntegration = true; | ||||
|   }; | ||||
|  | ||||
|   programs.difftastic = { | ||||
|     enable = true; | ||||
|     git.enable = true; | ||||
|   }; | ||||
|   programs.git = { | ||||
|     package = nixpkgs.gitAndTools.gitFull; | ||||
|     enable = true; | ||||
|     settings = { | ||||
|       alias = { | ||||
|         co = "checkout"; | ||||
|         st = "status"; | ||||
|       }; | ||||
|       user = { | ||||
|         email = "3138005+Smaug123@users.noreply.github.com"; | ||||
|         name = "Smaug123"; | ||||
|       }; | ||||
|       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!"; | ||||
|     userName = "Smaug123"; | ||||
|     userEmail = "patrick+github@patrickstevens.co.uk"; | ||||
|     aliases = { | ||||
|       co = "checkout"; | ||||
|       st = "status"; | ||||
|     }; | ||||
|     delta = {enable = true;}; | ||||
|     extraConfig = { | ||||
|       core = { | ||||
|         autocrlf = "input"; | ||||
|       }; | ||||
| @@ -82,14 +88,6 @@ | ||||
|       push = { | ||||
|         default = "current"; | ||||
|         autoSetupRemote = true; | ||||
|         followTags = true; | ||||
|       }; | ||||
|       fetch = { | ||||
|         prune = true; | ||||
|         all = true; | ||||
|       }; | ||||
|       help = { | ||||
|         autocorrect = "prompt"; | ||||
|       }; | ||||
|       pull = { | ||||
|         rebase = false; | ||||
| @@ -97,15 +95,6 @@ | ||||
|       init = { | ||||
|         defaultBranch = "main"; | ||||
|       }; | ||||
|       branch = { | ||||
|         sort = "-committerdate"; | ||||
|       }; | ||||
|       column = { | ||||
|         ui = "auto"; | ||||
|       }; | ||||
|       tag = { | ||||
|         sort = "version:refname"; | ||||
|       }; | ||||
|       advice = { | ||||
|         addIgnoredFile = false; | ||||
|       }; | ||||
| @@ -123,227 +112,163 @@ | ||||
|       }; | ||||
|       diff = { | ||||
|         colorMoved = "default"; | ||||
|         algorithm = "histogram"; | ||||
|         renames = true; | ||||
|       }; | ||||
|       "protocol.file" = { | ||||
|         allow = "always"; | ||||
|       }; | ||||
|       url."git@github.com:" = { | ||||
|         insteadOf = "https://github.com/"; | ||||
|       }; | ||||
|     }; | ||||
|   }; | ||||
|  | ||||
|   programs.vscode = { | ||||
|     enable = true; | ||||
|     enableExtensionUpdateCheck = true; | ||||
|     enableUpdateCheck = true; | ||||
|     package = nixpkgs.vscode; | ||||
|     profiles.default = { | ||||
|       extensions = import ./vscode-extensions.nix {pkgs = nixpkgs;}; | ||||
|       enableExtensionUpdateCheck = true; | ||||
|       enableUpdateCheck = true; | ||||
|       userSettings = { | ||||
|         workbench.colorTheme = "Default"; | ||||
|         "files.Exclude" = { | ||||
|           "**/.git" = true; | ||||
|           "**/.DS_Store" = true; | ||||
|           "**/Thumbs.db" = true; | ||||
|           "**/*.olean" = true; | ||||
|           "**/result" = true; | ||||
|         }; | ||||
|         "git.path" = "${nixpkgs.git}/bin/git"; | ||||
|         "update.mode" = "none"; | ||||
|         "explorer.confirmDelete" = false; | ||||
|     extensions = import ./vscode-extensions.nix {pkgs = nixpkgs;}; | ||||
|     userSettings = { | ||||
|       workbench.colorTheme = "Default"; | ||||
|       "files.Exclude" = { | ||||
|         "**/.git" = true; | ||||
|         "**/.DS_Store" = true; | ||||
|         "**/Thumbs.db" = true; | ||||
|         "**/*.olean" = true; | ||||
|         "**/result" = true; | ||||
|       }; | ||||
|       "git.path" = "${nixpkgs.git}/bin/git"; | ||||
|       "update.mode" = "none"; | ||||
|       "explorer.confirmDelete" = false; | ||||
|     }; | ||||
|   }; | ||||
|  | ||||
|   services.syncthing = { | ||||
|     enable = true; | ||||
|   }; | ||||
|  | ||||
|   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 { | ||||
|     enable = true; | ||||
|     plugins = [ | ||||
|       { | ||||
|         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.molokai | ||||
|       nixpkgs.vimPlugins.tagbar | ||||
|       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; | ||||
|         type = "lua"; | ||||
|         config = builtins.readFile ./nvim/ionide-vim.lua; | ||||
|         config = '' | ||||
|           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; | ||||
|         config = builtins.readFile ./nvim/chadtree.lua; | ||||
|         type = "lua"; | ||||
|         config = "let g:chadtree_settings = {'xdg': v:true}"; | ||||
|       } | ||||
|       { | ||||
|         plugin = nixpkgs.vimPlugins.nvim-cmp; | ||||
|         config = builtins.readFile ./nvim/nvim-cmp.lua; | ||||
|         type = "lua"; | ||||
|         plugin = nixpkgs.vimPlugins.coq_nvim; | ||||
|         config = ''let g:coq_settings = { 'auto_start': v:true, 'xdg': v:true }''; | ||||
|       } | ||||
|       { | ||||
|         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; | ||||
|         config = builtins.readFile ./nvim/nvim-dap.lua; | ||||
|         type = "lua"; | ||||
|       } | ||||
|       { | ||||
|         plugin = nixpkgs.vimPlugins.nvim-dap-python; | ||||
|         config = builtins.replaceStrings ["%PYTHONENV%"] ["${debugPyEnv}"] (builtins.readFile ./nvim/nvim-dap-python.lua); | ||||
|         type = "lua"; | ||||
|         plugin = nixpkgs.vimPlugins.syntastic; | ||||
|         config = ''          let g:syntastic_rust_checkers = ['cargo'] | ||||
|           let g:syntastic_always_populate_loc_list = 1 | ||||
|           let g:syntastic_auto_loc_list = 1 | ||||
|           let g:syntastic_check_on_open = 1 | ||||
|           let g:syntastic_check_on_wq = 0''; | ||||
|       } | ||||
|  | ||||
|       nixpkgs.vimPlugins.tagbar | ||||
|     ]; | ||||
|     viAlias = true; | ||||
|     vimAlias = true; | ||||
|     vimdiffAlias = 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 = | ||||
|     [ | ||||
|       nixpkgs.jq | ||||
|       nixpkgs.difftastic | ||||
|       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 [] | ||||
|     ); | ||||
|   programs.direnv = { | ||||
|     enable = true; | ||||
|     enableZshIntegration = true; | ||||
|     nix-direnv.enable = true; | ||||
|   }; | ||||
|  | ||||
|   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.".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 | ||||
|     github.vscode-pull-request-github | ||||
|     shardulm94.trailing-spaces | ||||
|     nvarner.typst-lsp | ||||
|     arrterian.nix-env-selector | ||||
|     # Doesn't build on arm64 | ||||
|     # vadimcn.vscode-lldb | ||||
|   ] | ||||
|   ++ pkgs.vscode-utils.extensionsFromVscodeMarketplace [ | ||||
|     { | ||||
| @@ -84,3 +83,23 @@ with pkgs.vscode-extensions; | ||||
|       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