flake.nix 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700
  1. {
  2. description = "CDK Flake";
  3. inputs = {
  4. nixpkgs.url = "github:NixOS/nixpkgs/nixos-25.11";
  5. rust-overlay = {
  6. url = "github:oxalica/rust-overlay";
  7. inputs = {
  8. nixpkgs.follows = "nixpkgs";
  9. };
  10. };
  11. fenix = {
  12. url = "github:nix-community/fenix";
  13. inputs.nixpkgs.follows = "nixpkgs";
  14. inputs.rust-analyzer-src.follows = "";
  15. };
  16. flake-utils.url = "github:numtide/flake-utils";
  17. crane = {
  18. url = "github:ipetkov/crane";
  19. };
  20. pre-commit-hooks.url = "github:cachix/pre-commit-hooks.nix";
  21. };
  22. outputs =
  23. { self
  24. , nixpkgs
  25. , rust-overlay
  26. , flake-utils
  27. , pre-commit-hooks
  28. , crane
  29. , ...
  30. }@inputs:
  31. flake-utils.lib.eachDefaultSystem (
  32. system:
  33. let
  34. overlays = [ (import rust-overlay) ];
  35. lib = pkgs.lib;
  36. stdenv = pkgs.stdenv;
  37. isDarwin = stdenv.isDarwin;
  38. libsDarwin =
  39. with pkgs;
  40. lib.optionals isDarwin [
  41. # Additional darwin specific inputs can be set here
  42. # Note: Security and SystemConfiguration frameworks are provided by the default SDK
  43. ];
  44. # Dependencies
  45. pkgs = import nixpkgs {
  46. inherit system overlays;
  47. };
  48. # Toolchains
  49. # latest stable
  50. stable_toolchain = pkgs.rust-bin.stable."1.92.0".default.override {
  51. targets = [ "wasm32-unknown-unknown" ]; # wasm
  52. extensions = [
  53. "rustfmt"
  54. "clippy"
  55. "rust-analyzer"
  56. ];
  57. };
  58. # MSRV stable
  59. msrv_toolchain = pkgs.rust-bin.stable."1.85.0".default.override {
  60. targets = [ "wasm32-unknown-unknown" ]; # wasm
  61. extensions = [
  62. "rustfmt"
  63. "clippy"
  64. "rust-analyzer"
  65. ];
  66. };
  67. # Nightly used for formatting
  68. nightly_toolchain = pkgs.rust-bin.selectLatestNightlyWith (
  69. toolchain:
  70. toolchain.default.override {
  71. extensions = [
  72. "rustfmt"
  73. "clippy"
  74. "rust-analyzer"
  75. "rust-src"
  76. ];
  77. targets = [ "wasm32-unknown-unknown" ]; # wasm
  78. }
  79. );
  80. # ========================================
  81. # Crane setup for cached builds
  82. # ========================================
  83. craneLib = (crane.mkLib pkgs).overrideToolchain stable_toolchain;
  84. craneLibMsrv = (crane.mkLib pkgs).overrideToolchain msrv_toolchain;
  85. # Source for crane builds
  86. src = builtins.path {
  87. path = ./.;
  88. name = "cdk-source";
  89. };
  90. # Source for MSRV builds - uses Cargo.lock.msrv with MSRV-compatible deps
  91. srcMsrv = pkgs.runCommand "cdk-source-msrv" { } ''
  92. cp -r ${src} $out
  93. chmod -R +w $out
  94. cp $out/Cargo.lock.msrv $out/Cargo.lock
  95. '';
  96. # Common args for all Crane builds
  97. commonCraneArgs = {
  98. inherit src;
  99. pname = "cdk";
  100. version = "0.14.0";
  101. nativeBuildInputs = with pkgs; [
  102. pkg-config
  103. protobuf
  104. ];
  105. buildInputs = with pkgs; [
  106. openssl
  107. sqlite
  108. zlib
  109. ] ++ libsDarwin;
  110. # Environment variables
  111. PROTOC = "${pkgs.protobuf}/bin/protoc";
  112. PROTOC_INCLUDE = "${pkgs.protobuf}/include";
  113. };
  114. # Common args for MSRV builds - uses srcMsrv with pinned deps
  115. commonCraneArgsMsrv = commonCraneArgs // {
  116. src = srcMsrv;
  117. };
  118. # Build ALL dependencies once - this is what gets cached by Cachix
  119. # Note: We exclude swagger feature as it tries to download assets during build
  120. workspaceDeps = craneLib.buildDepsOnly (commonCraneArgs // {
  121. pname = "cdk-deps";
  122. # Build deps for workspace - swagger excluded (downloads during build)
  123. cargoExtraArgs = "--workspace";
  124. });
  125. # MSRV dependencies (separate cache due to different toolchain)
  126. workspaceDepsMsrv = craneLibMsrv.buildDepsOnly (commonCraneArgsMsrv // {
  127. pname = "cdk-deps-msrv";
  128. cargoExtraArgs = "--workspace";
  129. });
  130. # Helper function to create clippy checks
  131. mkClippy = name: cargoArgs: craneLib.cargoClippy (commonCraneArgs // {
  132. pname = "cdk-clippy-${name}";
  133. cargoArtifacts = workspaceDeps;
  134. cargoClippyExtraArgs = "${cargoArgs} -- -D warnings";
  135. });
  136. # Helper function to create example checks (compile only, no network access in sandbox)
  137. mkExample = name: craneLib.mkCargoDerivation (commonCraneArgs // {
  138. pname = "cdk-example-${name}";
  139. cargoArtifacts = workspaceDeps;
  140. buildPhaseCargoCommand = "cargo build --example ${name}";
  141. # Examples are compiled but not run (no network in Nix sandbox)
  142. installPhaseCommand = "mkdir -p $out";
  143. });
  144. # Helper function to create example packages (outputs binary for running outside sandbox)
  145. mkExamplePackage = name: craneLib.mkCargoDerivation (commonCraneArgs // {
  146. pname = "cdk-example-${name}";
  147. cargoArtifacts = workspaceDeps;
  148. buildPhaseCargoCommand = "cargo build --release --example ${name}";
  149. installPhaseCommand = ''
  150. mkdir -p $out/bin
  151. cp target/release/examples/${name} $out/bin/
  152. '';
  153. });
  154. # Helper function to create MSRV build checks
  155. mkMsrvBuild = name: cargoArgs: craneLibMsrv.cargoBuild (commonCraneArgsMsrv // {
  156. pname = "cdk-msrv-${name}";
  157. cargoArtifacts = workspaceDepsMsrv;
  158. cargoExtraArgs = cargoArgs;
  159. });
  160. # Helper function to create WASM build checks
  161. # WASM builds don't need native libs like openssl
  162. mkWasmBuild = name: cargoArgs: craneLib.cargoBuild ({
  163. inherit src;
  164. pname = "cdk-wasm-${name}";
  165. version = "0.14.0";
  166. cargoArtifacts = workspaceDeps;
  167. cargoExtraArgs = "${cargoArgs} --target wasm32-unknown-unknown";
  168. # WASM doesn't need native build inputs
  169. nativeBuildInputs = with pkgs; [ pkg-config ];
  170. buildInputs = [ ];
  171. # Disable tests for WASM (can't run in sandbox)
  172. doCheck = false;
  173. });
  174. # Doc tests check
  175. docTests = craneLib.cargoTest (commonCraneArgs // {
  176. pname = "cdk-doc-tests";
  177. cargoArtifacts = workspaceDeps;
  178. cargoTestExtraArgs = "--doc";
  179. });
  180. # Strict docs check - build docs with warnings as errors
  181. # Uses mkCargoDerivation for custom RUSTDOCFLAGS
  182. strictDocs = craneLib.mkCargoDerivation (commonCraneArgs // {
  183. pname = "cdk-strict-docs";
  184. cargoArtifacts = workspaceDeps;
  185. buildPhaseCargoCommand = ''
  186. export RUSTDOCFLAGS="-D warnings"
  187. cargo doc --no-deps \
  188. -p cashu \
  189. -p cdk-common \
  190. -p cdk-sql-common \
  191. -p cdk \
  192. -p cdk-redb \
  193. -p cdk-sqlite \
  194. -p cdk-axum \
  195. -p cdk-cln \
  196. -p cdk-lnd \
  197. -p cdk-lnbits \
  198. -p cdk-fake-wallet \
  199. -p cdk-mint-rpc \
  200. -p cdk-payment-processor \
  201. -p cdk-signatory \
  202. -p cdk-cli \
  203. -p cdk-mintd
  204. '';
  205. installPhaseCommand = "mkdir -p $out";
  206. });
  207. # FFI Python tests
  208. ffiTests = craneLib.mkCargoDerivation (commonCraneArgs // {
  209. pname = "cdk-ffi-tests";
  210. cargoArtifacts = workspaceDeps;
  211. nativeBuildInputs = commonCraneArgs.nativeBuildInputs ++ [
  212. pkgs.python311
  213. ];
  214. buildPhaseCargoCommand = ''
  215. # Build the FFI library
  216. cargo build --release --package cdk-ffi --features postgres
  217. # Generate Python bindings
  218. cargo run --bin uniffi-bindgen generate \
  219. --library target/release/libcdk_ffi.so \
  220. --language python \
  221. --out-dir target/bindings/python
  222. # Copy library to bindings directory
  223. cp target/release/libcdk_ffi.so target/bindings/python/
  224. # Run Python tests
  225. python3 crates/cdk-ffi/tests/test_transactions.py
  226. '';
  227. installPhaseCommand = "mkdir -p $out";
  228. });
  229. # ========================================
  230. # Example definitions - single source of truth
  231. # ========================================
  232. exampleChecks = [
  233. "mint-token"
  234. "melt-token"
  235. "p2pk"
  236. "proof-selection"
  237. "wallet"
  238. ];
  239. # ========================================
  240. # Clippy check definitions - single source of truth
  241. # ========================================
  242. clippyChecks = {
  243. # Core crate: cashu
  244. "cashu" = "-p cashu";
  245. "cashu-no-default" = "-p cashu --no-default-features";
  246. "cashu-wallet" = "-p cashu --no-default-features --features wallet";
  247. "cashu-mint" = "-p cashu --no-default-features --features mint";
  248. "cashu-auth" = "-p cashu --no-default-features --features auth";
  249. # Core crate: cdk-common
  250. "cdk-common" = "-p cdk-common";
  251. "cdk-common-no-default" = "-p cdk-common --no-default-features";
  252. "cdk-common-wallet" = "-p cdk-common --no-default-features --features wallet";
  253. "cdk-common-mint" = "-p cdk-common --no-default-features --features mint";
  254. "cdk-common-auth" = "-p cdk-common --no-default-features --features auth";
  255. # Core crate: cdk
  256. "cdk" = "-p cdk";
  257. "cdk-no-default" = "-p cdk --no-default-features";
  258. "cdk-wallet" = "-p cdk --no-default-features --features wallet";
  259. "cdk-mint" = "-p cdk --no-default-features --features mint";
  260. "cdk-auth" = "-p cdk --no-default-features --features auth";
  261. # SQL crates
  262. "cdk-sql-common" = "-p cdk-sql-common";
  263. "cdk-sql-common-wallet" = "-p cdk-sql-common --no-default-features --features wallet";
  264. "cdk-sql-common-mint" = "-p cdk-sql-common --no-default-features --features mint";
  265. # Database crates
  266. "cdk-redb" = "-p cdk-redb";
  267. "cdk-sqlite" = "-p cdk-sqlite";
  268. "cdk-sqlite-sqlcipher" = "-p cdk-sqlite --features sqlcipher";
  269. # HTTP/API layer
  270. # Note: swagger feature excluded - downloads assets during build, incompatible with Nix sandbox
  271. "cdk-axum" = "-p cdk-axum";
  272. "cdk-axum-no-default" = "-p cdk-axum --no-default-features";
  273. "cdk-axum-redis" = "-p cdk-axum --no-default-features --features redis";
  274. # Lightning backends
  275. "cdk-cln" = "-p cdk-cln";
  276. "cdk-lnd" = "-p cdk-lnd";
  277. "cdk-lnbits" = "-p cdk-lnbits";
  278. "cdk-fake-wallet" = "-p cdk-fake-wallet";
  279. "cdk-payment-processor" = "-p cdk-payment-processor";
  280. "cdk-ldk-node" = "-p cdk-ldk-node";
  281. # Other crates
  282. "cdk-signatory" = "-p cdk-signatory";
  283. "cdk-mint-rpc" = "-p cdk-mint-rpc";
  284. "cdk-prometheus" = "-p cdk-prometheus";
  285. "cdk-ffi" = "-p cdk-ffi";
  286. # Binaries: cdk-cli
  287. "bin-cdk-cli" = "--bin cdk-cli";
  288. "bin-cdk-cli-sqlcipher" = "--bin cdk-cli --features sqlcipher";
  289. "bin-cdk-cli-redb" = "--bin cdk-cli --features redb";
  290. # Binaries: cdk-mintd
  291. "bin-cdk-mintd" = "--bin cdk-mintd";
  292. "bin-cdk-mintd-redis" = "--bin cdk-mintd --features redis";
  293. "bin-cdk-mintd-sqlcipher" = "--bin cdk-mintd --features sqlcipher";
  294. "bin-cdk-mintd-lnd-sqlite" = "--bin cdk-mintd --no-default-features --features lnd,sqlite";
  295. "bin-cdk-mintd-cln-postgres" = "--bin cdk-mintd --no-default-features --features cln,postgres";
  296. "bin-cdk-mintd-lnbits-sqlite" = "--bin cdk-mintd --no-default-features --features lnbits,sqlite";
  297. "bin-cdk-mintd-fakewallet-sqlite" = "--bin cdk-mintd --no-default-features --features fakewallet,sqlite";
  298. "bin-cdk-mintd-grpc-processor-sqlite" = "--bin cdk-mintd --no-default-features --features grpc-processor,sqlite";
  299. "bin-cdk-mintd-management-rpc-lnd-sqlite" = "--bin cdk-mintd --no-default-features --features management-rpc,lnd,sqlite";
  300. "bin-cdk-mintd-cln-sqlite" = "--bin cdk-mintd --no-default-features --features cln,sqlite";
  301. "bin-cdk-mintd-lnd-postgres" = "--bin cdk-mintd --no-default-features --features lnd,postgres";
  302. "bin-cdk-mintd-lnbits-postgres" = "--bin cdk-mintd --no-default-features --features lnbits,postgres";
  303. "bin-cdk-mintd-fakewallet-postgres" = "--bin cdk-mintd --no-default-features --features fakewallet,postgres";
  304. "bin-cdk-mintd-grpc-processor-postgres" = "--bin cdk-mintd --no-default-features --features grpc-processor,postgres";
  305. "bin-cdk-mintd-management-rpc-cln-postgres" = "--bin cdk-mintd --no-default-features --features management-rpc,cln,postgres";
  306. "bin-cdk-mintd-auth-sqlite-fakewallet" = "--bin cdk-mintd --no-default-features --features auth,sqlite,fakewallet";
  307. "bin-cdk-mintd-auth-postgres-lnd" = "--bin cdk-mintd --no-default-features --features auth,postgres,lnd";
  308. # Binaries: cdk-mint-cli
  309. "bin-cdk-mint-cli" = "--bin cdk-mint-cli";
  310. };
  311. # ========================================
  312. # MSRV build check definitions
  313. # ========================================
  314. msrvChecks = {
  315. # Core library with all features (except swagger which breaks MSRV)
  316. "cdk-all-features" = "-p cdk --features \"mint,wallet,auth\"";
  317. # Mintd with all backends, databases, and features (no swagger)
  318. "cdk-mintd-all" = "-p cdk-mintd --no-default-features --features \"cln,lnd,lnbits,fakewallet,ldk-node,grpc-processor,sqlite,postgres,auth,redis,management-rpc\"";
  319. # CLI - default features (excludes redb which breaks MSRV)
  320. "cdk-cli" = "-p cdk-cli";
  321. # Minimal builds to ensure no-default-features works
  322. "cdk-wallet-only" = "-p cdk --no-default-features --features wallet";
  323. };
  324. # ========================================
  325. # WASM build check definitions
  326. # ========================================
  327. wasmChecks = {
  328. "cdk" = "-p cdk";
  329. "cdk-no-default" = "-p cdk --no-default-features";
  330. "cdk-wallet" = "-p cdk --no-default-features --features wallet";
  331. };
  332. # Common inputs
  333. envVars = {
  334. # rust analyzer needs NIX_PATH for some reason.
  335. NIX_PATH = "nixpkgs=${inputs.nixpkgs}";
  336. };
  337. # Override clightning to include mako dependency and fix compilation bug
  338. clightningWithMako = pkgs.clightning.overrideAttrs (oldAttrs: {
  339. nativeBuildInputs = (oldAttrs.nativeBuildInputs or [ ]) ++ [
  340. pkgs.python311Packages.mako
  341. ];
  342. # Disable -Werror to work around multiple compilation bugs in 25.09.2 on macOS
  343. # See: https://github.com/ElementsProject/lightning/issues/7961
  344. env = (oldAttrs.env or { }) // {
  345. NIX_CFLAGS_COMPILE = toString ((oldAttrs.env.NIX_CFLAGS_COMPILE or "") + " -Wno-error");
  346. };
  347. });
  348. buildInputs =
  349. with pkgs;
  350. [
  351. # Add additional build inputs here
  352. git
  353. pkg-config
  354. curl
  355. just
  356. protobuf
  357. nixpkgs-fmt
  358. typos
  359. lnd
  360. clightningWithMako
  361. bitcoind
  362. sqlx-cli
  363. mprocs
  364. cargo-outdated
  365. cargo-mutants
  366. # Needed for github ci
  367. libz
  368. ]
  369. ++ libsDarwin;
  370. # PostgreSQL configuration
  371. postgresConf = {
  372. pgUser = "cdk_user";
  373. pgPassword = "cdk_password";
  374. pgDatabase = "cdk_mint";
  375. pgPort = "5432";
  376. };
  377. # Script to start PostgreSQL
  378. startPostgres = pkgs.writeShellScriptBin "start-postgres" ''
  379. set -e
  380. PGDATA="$PWD/.pg_data"
  381. PGPORT="${postgresConf.pgPort}"
  382. PGUSER="${postgresConf.pgUser}"
  383. PGPASSWORD="${postgresConf.pgPassword}"
  384. PGDATABASE="${postgresConf.pgDatabase}"
  385. # Stop any existing instance first
  386. if [ -d "$PGDATA" ] && ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" status > /dev/null 2>&1; then
  387. echo "Stopping existing PostgreSQL instance..."
  388. ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" stop > /dev/null 2>&1
  389. fi
  390. if [ ! -d "$PGDATA" ]; then
  391. echo "Initializing PostgreSQL database..."
  392. ${pkgs.postgresql_16}/bin/initdb -D "$PGDATA" --auth=trust --no-locale --encoding=UTF8
  393. # Configure PostgreSQL
  394. echo "listen_addresses = 'localhost'" >> "$PGDATA/postgresql.conf"
  395. echo "port = $PGPORT" >> "$PGDATA/postgresql.conf"
  396. echo "unix_socket_directories = '$PGDATA'" >> "$PGDATA/postgresql.conf"
  397. # Start temporarily to create user and database
  398. ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" -l "$PGDATA/logfile" start
  399. sleep 2
  400. # Create user and database
  401. ${pkgs.postgresql_16}/bin/createuser -h localhost -p $PGPORT -s "$PGUSER" || true
  402. ${pkgs.postgresql_16}/bin/psql -h localhost -p $PGPORT -c "ALTER USER $PGUSER WITH PASSWORD '$PGPASSWORD';" postgres
  403. ${pkgs.postgresql_16}/bin/createdb -h localhost -p $PGPORT -O "$PGUSER" "$PGDATABASE" || true
  404. ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" stop
  405. echo "PostgreSQL initialized."
  406. fi
  407. echo "Starting PostgreSQL on port $PGPORT..."
  408. ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" -l "$PGDATA/logfile" start
  409. echo "PostgreSQL started. Connection URL: postgresql://$PGUSER:$PGPASSWORD@localhost:$PGPORT/$PGDATABASE"
  410. '';
  411. # Script to stop PostgreSQL
  412. stopPostgres = pkgs.writeShellScriptBin "stop-postgres" ''
  413. PGDATA="$PWD/.pg_data"
  414. if [ -d "$PGDATA" ]; then
  415. echo "Stopping PostgreSQL..."
  416. ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" stop || echo "PostgreSQL was not running."
  417. else
  418. echo "No PostgreSQL data directory found."
  419. fi
  420. '';
  421. # Script to check PostgreSQL status
  422. pgStatus = pkgs.writeShellScriptBin "pg-status" ''
  423. PGDATA="$PWD/.pg_data"
  424. if [ -d "$PGDATA" ]; then
  425. ${pkgs.postgresql_16}/bin/pg_ctl -D "$PGDATA" status
  426. else
  427. echo "No PostgreSQL data directory found. Run 'start-postgres' first."
  428. fi
  429. '';
  430. # Script to connect to PostgreSQL
  431. pgConnect = pkgs.writeShellScriptBin "pg-connect" ''
  432. ${pkgs.postgresql_16}/bin/psql "postgresql://${postgresConf.pgUser}:${postgresConf.pgPassword}@localhost:${postgresConf.pgPort}/${postgresConf.pgDatabase}"
  433. '';
  434. # Common arguments can be set here to avoid repeating them later
  435. nativeBuildInputs = [
  436. #Add additional build inputs here
  437. ]
  438. ++ lib.optionals isDarwin [
  439. # Additional darwin specific native inputs can be set here
  440. ];
  441. in
  442. {
  443. # Expose deps for explicit cache warming
  444. packages = {
  445. deps = workspaceDeps;
  446. deps-msrv = workspaceDepsMsrv;
  447. }
  448. # Example packages (binaries that can be run outside sandbox with network access)
  449. // (builtins.listToAttrs (map (name: { name = "example-${name}"; value = mkExamplePackage name; }) exampleChecks));
  450. checks =
  451. # Generate clippy checks from clippyChecks attrset
  452. (builtins.mapAttrs (name: args: mkClippy name args) clippyChecks)
  453. # Generate MSRV build checks (prefixed with msrv-)
  454. // (builtins.listToAttrs (map (name: { name = "msrv-${name}"; value = mkMsrvBuild name msrvChecks.${name}; }) (builtins.attrNames msrvChecks)))
  455. # Generate WASM build checks (prefixed with wasm-)
  456. // (builtins.listToAttrs (map (name: { name = "wasm-${name}"; value = mkWasmBuild name wasmChecks.${name}; }) (builtins.attrNames wasmChecks)))
  457. # Generate example checks from exampleChecks list
  458. // (builtins.listToAttrs (map (name: { name = "example-${name}"; value = mkExample name; }) exampleChecks))
  459. // {
  460. # Doc tests
  461. doc-tests = docTests;
  462. # Strict docs check
  463. strict-docs = strictDocs;
  464. # FFI Python tests
  465. ffi-tests = ffiTests;
  466. # Pre-commit checks
  467. pre-commit-check =
  468. let
  469. # this is a hack based on https://github.com/cachix/pre-commit-hooks.nix/issues/126
  470. # we want to use our own rust stuff from oxalica's overlay
  471. _rust = pkgs.rust-bin.stable.latest.default;
  472. rust = pkgs.buildEnv {
  473. name = _rust.name;
  474. inherit (_rust) meta;
  475. buildInputs = [ pkgs.makeWrapper ];
  476. paths = [ _rust ];
  477. pathsToLink = [
  478. "/"
  479. "/bin"
  480. ];
  481. postBuild = ''
  482. for i in $out/bin/*; do
  483. wrapProgram "$i" --prefix PATH : "$out/bin"
  484. done
  485. '';
  486. };
  487. in
  488. pre-commit-hooks.lib.${system}.run {
  489. src = ./.;
  490. hooks = {
  491. rustfmt = {
  492. enable = true;
  493. entry = lib.mkForce "${rust}/bin/cargo-fmt fmt --all -- --config format_code_in_doc_comments=true --check --color always";
  494. };
  495. nixpkgs-fmt.enable = true;
  496. typos.enable = true;
  497. commitizen.enable = true; # conventional commits
  498. };
  499. };
  500. };
  501. devShells =
  502. let
  503. # pre-commit-checks
  504. _shellHook = (self.checks.${system}.pre-commit-check.shellHook or "");
  505. # devShells
  506. msrv = pkgs.mkShell (
  507. {
  508. shellHook = "
  509. cargo update
  510. cargo update home --precise 0.5.11
  511. cargo update typed-index-collections --precise 3.3.0
  512. ${_shellHook}
  513. ";
  514. buildInputs = buildInputs ++ [ msrv_toolchain ];
  515. inherit nativeBuildInputs;
  516. }
  517. // envVars
  518. );
  519. stable = pkgs.mkShell (
  520. {
  521. shellHook = ''
  522. ${_shellHook}
  523. # Needed for github ci
  524. export LD_LIBRARY_PATH=${
  525. pkgs.lib.makeLibraryPath [
  526. pkgs.zlib
  527. ]
  528. }:$LD_LIBRARY_PATH
  529. # PostgreSQL environment variables
  530. export CDK_MINTD_DATABASE_URL="postgresql://${postgresConf.pgUser}:${postgresConf.pgPassword}@localhost:${postgresConf.pgPort}/${postgresConf.pgDatabase}"
  531. echo ""
  532. echo "PostgreSQL commands available:"
  533. echo " start-postgres - Initialize and start PostgreSQL"
  534. echo " stop-postgres - Stop PostgreSQL (run before exiting)"
  535. echo " pg-status - Check PostgreSQL status"
  536. echo " pg-connect - Connect to PostgreSQL with psql"
  537. echo ""
  538. '';
  539. buildInputs = buildInputs ++ [
  540. stable_toolchain
  541. pkgs.postgresql_16
  542. startPostgres
  543. stopPostgres
  544. pgStatus
  545. pgConnect
  546. ];
  547. inherit nativeBuildInputs;
  548. }
  549. // envVars
  550. );
  551. nightly = pkgs.mkShell (
  552. {
  553. shellHook = ''
  554. ${_shellHook}
  555. # Needed for github ci
  556. export LD_LIBRARY_PATH=${
  557. pkgs.lib.makeLibraryPath [
  558. pkgs.zlib
  559. ]
  560. }:$LD_LIBRARY_PATH
  561. '';
  562. buildInputs = buildInputs ++ [ nightly_toolchain ];
  563. inherit nativeBuildInputs;
  564. }
  565. // envVars
  566. );
  567. # Shell with Docker for integration tests
  568. integration = pkgs.mkShell (
  569. {
  570. shellHook = ''
  571. ${_shellHook}
  572. # Ensure Docker is available
  573. if ! command -v docker &> /dev/null; then
  574. echo "Docker is not installed or not in PATH"
  575. echo "Please install Docker to run integration tests"
  576. exit 1
  577. fi
  578. echo "Docker is available at $(which docker)"
  579. echo "Docker version: $(docker --version)"
  580. '';
  581. buildInputs = buildInputs ++ [
  582. stable_toolchain
  583. pkgs.docker-client
  584. pkgs.python311
  585. ];
  586. inherit nativeBuildInputs;
  587. }
  588. // envVars
  589. );
  590. # Shell for FFI development (Python bindings)
  591. ffi = pkgs.mkShell (
  592. {
  593. shellHook = ''
  594. ${_shellHook}
  595. echo "FFI development shell"
  596. echo " just ffi-test - Run Python FFI tests"
  597. echo " just ffi-dev-python - Launch Python REPL with CDK FFI"
  598. '';
  599. buildInputs = buildInputs ++ [
  600. stable_toolchain
  601. pkgs.python311
  602. ];
  603. inherit nativeBuildInputs;
  604. }
  605. // envVars
  606. );
  607. in
  608. {
  609. inherit
  610. msrv
  611. stable
  612. nightly
  613. integration
  614. ffi
  615. ;
  616. default = stable;
  617. };
  618. }
  619. );
  620. }