garlicd: allow manual experiment executions
This commit is contained in:
		
							parent
							
								
									ceb25e5d18
								
							
						
					
					
						commit
						0b95ea20b7
					
				| @ -29,8 +29,17 @@ mountdir_rel=$(echo "$garlic_sandbox" | sed 's@^/garlic=@@g') | |||||||
| mountdir=$(readlink -f "$mountdir_rel") | mountdir=$(readlink -f "$mountdir_rel") | ||||||
| run="$mountdir/run" | run="$mountdir/run" | ||||||
| completed="$mountdir/completed" | completed="$mountdir/completed" | ||||||
|  | wipe="$mountdir/wipe" | ||||||
| 
 | 
 | ||||||
| for fifo in "$run" "$completed"; do | handle_bad_signal() { | ||||||
|  |   msg "cleaning FIFO pipes" | ||||||
|  |   rm -f "$run" "$completed" "$wipe" | ||||||
|  |   exit 1 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | trap handle_bad_signal SIGINT | ||||||
|  | 
 | ||||||
|  | for fifo in "$run" "$completed" "$wipe"; do | ||||||
|   if [ ! -e "$fifo" ]; then |   if [ ! -e "$fifo" ]; then | ||||||
|     mkfifo "$fifo" |     mkfifo "$fifo" | ||||||
|     # FIXME: Use more resctrictive permissions |     # FIXME: Use more resctrictive permissions | ||||||
| @ -54,19 +63,22 @@ while true; do | |||||||
|   garlic -R "$tre" |   garlic -R "$tre" | ||||||
| 
 | 
 | ||||||
|   msg "Fetching results..." |   msg "Fetching results..." | ||||||
|   results=$(garlic -Fv "$tre") |   garlic -FKv "$tre" | ||||||
| 
 | 
 | ||||||
|   msg "results=\"$results\"" |   echo -n "$tre" >> "$completed" | ||||||
| 
 | 
 | ||||||
|   msg "Searching drv..." |   msg "Waiting for nix to finish the build..." | ||||||
|   drv=$(nix-store -q --deriver $results) |  | ||||||
| 
 | 
 | ||||||
|   msg "drv = \"$drv\"" |   read -r tre2 < "$wipe" | ||||||
|   if [ -z "$drv" ]; then |   if [ "$tre" != "$tre2" ]; then | ||||||
|     msg "Something failed, drv is empty. Check the logs." |     msg "error: trebuchet mismatch" | ||||||
|     exit 1 |     exit 1 | ||||||
|   fi |   fi | ||||||
| 
 | 
 | ||||||
|   echo -n "$drv" >> "$completed" |   msg "Removing temporal files..." | ||||||
|   msg "execution completed :-)" |   garlic -D "$tre" | ||||||
|  | 
 | ||||||
|  |   echo -n "$tre" >> "$completed" | ||||||
|  | 
 | ||||||
|  |   msg "Execution completed :-)" | ||||||
| done | done | ||||||
|  | |||||||
| @ -3,52 +3,77 @@ | |||||||
| }: | }: | ||||||
| 
 | 
 | ||||||
| { | { | ||||||
|   experimentStage |   trebuchet, | ||||||
| , trebuchetStage |   experiment | ||||||
| }: | }: | ||||||
| 
 | 
 | ||||||
| with builtins; | with builtins; | ||||||
| 
 | 
 | ||||||
| #assert typeOf experimentStage == "string"; |  | ||||||
| #assert typeOf trebuchetStage == "string"; |  | ||||||
| 
 |  | ||||||
| let | let | ||||||
|   # We cannot keep the context of the string when called from a derivation, as |   experimentName = baseNameOf (experiment); | ||||||
|   # they will produce a different resultTree derivation vs called from the |   trebuchetName = baseNameOf (trebuchet); | ||||||
|   # garlic script tool. |  | ||||||
|   #_experimentStage = unsafeDiscardStringContext experimentStage; |  | ||||||
|   #_trebuchetStage = unsafeDiscardStringContext trebuchetStage; |  | ||||||
| 
 |  | ||||||
|   experimentName = baseNameOf (experimentStage); |  | ||||||
|   trebuchetName = baseNameOf (trebuchetStage); |  | ||||||
|   garlicTemp = "/tmp/garlic"; |  | ||||||
| in | in | ||||||
|   #assert hasContext _trebuchetStage == false; |  | ||||||
|   #assert hasContext _experimentStage == false; |  | ||||||
|   stdenv.mkDerivation { |   stdenv.mkDerivation { | ||||||
|     name = "resultTree"; |     name = "resultTree"; | ||||||
|     preferLocalBuild = true; |     preferLocalBuild = true; | ||||||
|     __noChroot = true; |  | ||||||
| 
 | 
 | ||||||
|     phases = [ "installPhase" ]; |     phases = [ "installPhase" ]; | ||||||
| 
 | 
 | ||||||
|     installPhase = '' |     installPhase = '' | ||||||
|       exp=${garlicTemp}/${experimentName} |       echo "resultTree: searching for garlicd daemon..." | ||||||
|  |       if [ -e /garlic/run ]; then | ||||||
|  |         echo "resultTree: asking the daemon to run and fetch the experiment" | ||||||
|  | 
 | ||||||
|  |         echo ${trebuchet} >> /garlic/run | ||||||
|  |         echo "resultTree: waiting for experiment results..." | ||||||
|  |         res=$(cat /garlic/completed) | ||||||
|  | 
 | ||||||
|  |         if [ "$res" != "${trebuchet}" ]; then | ||||||
|  |           echo "resultTree: unknown trebuchet received" | ||||||
|  |           exit 1 | ||||||
|  |         fi | ||||||
|  |       else | ||||||
|  |         echo "resultTree: garlicd not detected: /garlic/run not found" | ||||||
|  |         echo "resultTree: assuming results are already in /garlic" | ||||||
|  |       fi | ||||||
|  | 
 | ||||||
|  |       echo "resultTree: attempting to copy the results from /garlic ..." | ||||||
|  | 
 | ||||||
|  |       exp=/garlic/cache/${experimentName} | ||||||
| 
 | 
 | ||||||
|       if [ ! -e "$exp" ]; then |       if [ ! -e "$exp" ]; then | ||||||
|         echo "$exp: not found" |         echo "resultTree: $exp: not found" | ||||||
|         echo "Run the experiment and fetch the results with:" |         echo "resultTree: run the experiment and fetch the results with:" | ||||||
|         echo |         echo | ||||||
|         echo -e "\e[30;48;5;2mgarlic -RFv ${trebuchetStage}\e[0m" |         echo -e "\e[30;48;5;2mgarlic -RFv ${trebuchet}\e[0m" | ||||||
|         echo |         echo | ||||||
|         echo "See garlic(1) for more details." |         echo "resultTree: see garlic(1) for more details." | ||||||
|         echo "cannot continue building $out, aborting" |         echo "resultTree: cannot continue building $out, aborting" | ||||||
|         exit 1 |         exit 1 | ||||||
|       fi |       fi | ||||||
| 
 | 
 | ||||||
|  |       echo "resultTree: copying results from /garlic into the nix store..." | ||||||
|  | 
 | ||||||
|       mkdir -p $out |       mkdir -p $out | ||||||
|       cp -aL $exp $out/ |       cp -aL $exp $out/ | ||||||
|       ln -s ${trebuchetStage} $out/trebuchet |       ln -s ${trebuchet} $out/trebuchet | ||||||
|       ln -s ${experimentStage} $out/experiment |       ln -s ${experiment} $out/experiment | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |       if [ -e /garlic/run ]; then | ||||||
|  |         echo "resultTree: removing temp files..." | ||||||
|  |         echo ${trebuchet} >> /garlic/wipe | ||||||
|  |         echo "resultTree: waiting confimation from daemon..." | ||||||
|  |         cat /garlic/completed > /dev/null | ||||||
|  |       else | ||||||
|  |         echo "resultTree: garlicd not detected: /garlic/run not found" | ||||||
|  |         echo "resultTree: ignoring temp files" | ||||||
|  |       fi | ||||||
|  | 
 | ||||||
|  |       echo "resultTree: successfully copied into the nix store" | ||||||
|  | 
 | ||||||
|  |       echo "  experiment: ${experiment}" | ||||||
|  |       echo "   trebuchet: ${trebuchet}" | ||||||
|  |       echo "  resultTree: $out" | ||||||
|     ''; |     ''; | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -11,7 +11,6 @@ with garlicTools; | |||||||
| 
 | 
 | ||||||
| let | let | ||||||
|   garlicPrefix = "/mnt/garlic"; |   garlicPrefix = "/mnt/garlic"; | ||||||
|   garlicTemp = "/tmp/garlic"; |  | ||||||
| in | in | ||||||
|   stdenv.mkDerivation { |   stdenv.mkDerivation { | ||||||
|     name = "garlic-tool"; |     name = "garlic-tool"; | ||||||
| @ -22,7 +21,7 @@ in | |||||||
| 
 | 
 | ||||||
|     src = ./.; |     src = ./.; | ||||||
| 
 | 
 | ||||||
|     inherit garlicPrefix garlicTemp sshHost; |     inherit garlicPrefix sshHost; | ||||||
| 
 | 
 | ||||||
|     installPhase = '' |     installPhase = '' | ||||||
|       substituteAllInPlace garlic |       substituteAllInPlace garlic | ||||||
|  | |||||||
| @ -1,15 +1,9 @@ | |||||||
| #!/bin/bash -e | #!/bin/bash -e | ||||||
| 
 | 
 | ||||||
| garlicPrefix=@garlicPrefix@ | garlicPrefix=@garlicPrefix@ | ||||||
| garlicTemp=@garlicTemp@ |  | ||||||
| sshHost=@sshHost@ | sshHost=@sshHost@ | ||||||
| PATH=@PATH@ | PATH=@PATH@ | ||||||
| 
 | 
 | ||||||
| #garlicPrefix=/mnt/garlic |  | ||||||
| #garlicTemp=/tmp/garlic |  | ||||||
| #sshHost=mn1 |  | ||||||
| #PATH=/nix/store/yjkcxbf0y1jdlbj0axghlg2fndc4dqkz-patchelf-0.11/bin:/nix/store/6is25fyx29d731idycngl7qmgcax5xng-gcc-wrapper-9.3.0/bin:/nix/store/h986r9i2j9x5z8i5g8aj0z8jdd129wyx-gcc-9.3.0/bin:/nix/store/48dypl6qdsj3vdzh7hjg5qnngfpdcz7h-glibc-2.31-bin/bin:/nix/store/zmac3n79ayg4fdqgznmi2v3lmcprzx4g-coreutils-8.31/bin:/nix/store/dmnnqr6j7kqgcr357b5qwiwvjvg2yyhd-binutils-wrapper-2.31.1/bin:/nix/store/gmi6xrkl95h6iypv00dvdpm3f4md9i6i-binutils-2.31.1/bin:/nix/store/48dypl6qdsj3vdzh7hjg5qnngfpdcz7h-glibc-2.31-bin/bin:/nix/store/zmac3n79ayg4fdqgznmi2v3lmcprzx4g-coreutils-8.31/bin:/nix/store/a41ky6icdgxa54jzps32gfgcrdyx94hs-rsync-3.1.3/bin:/nix/store/sxll2dlamfm32xd2nyfx7v8mlnx0gxks-openssh-8.3p1/bin:/nix/store/3gp7gv5z9jj3g92czxadvgphpwiviv28-nix-2.3.7/bin:/nix/store/zmac3n79ayg4fdqgznmi2v3lmcprzx4g-coreutils-8.31/bin:/nix/store/f2hn65ksj194nmy58nrjikv9r9w25irh-findutils-4.7.0/bin:/nix/store/m39n3m5c7r22b3ma2phnwmp0jj8a5jja-diffutils-3.7/bin:/nix/store/ray7jgwsr5xbxp28wvr427vywd08nz9s-gnused-4.8/bin:/nix/store/gzc092gzsanvym4c6sjgh22dsh9fzj4s-gnugrep-3.4/bin:/nix/store/mn412q9rz9afdrhl9v2ybf605r91wzl2-gawk-5.1.0/bin:/nix/store/xca341k5x5b4hcmi310gjhdlgqm4l56m-gnutar-1.32/bin:/nix/store/zvb8qad72bz6j7ia60dcsf3dfncxxqc7-gzip-1.10/bin:/nix/store/9pb8zp3zyykw09rg60f2nv32plamhd7h-bzip2-1.0.6.0.1-bin/bin:/nix/store/fm2p1d8w9sx4gbaf8qfv2rsailsyhvm3-gnumake-4.3/bin:/nix/store/npfsrhkjww5q7sax7p7ijcrj3wlbrxn7-bash-4.4-p23/bin:/nix/store/72m0m8v6mbp58vbngjgv5pn2scqhs6kk-patch-2.7.6/bin:/nix/store/7vh7fckk2srlkmmkfhs9y85icwm9rhj5-xz-5.2.5-bin/bin |  | ||||||
| 
 |  | ||||||
| usage() { echo "Usage: garlic [-RFwv] trebuchet" 1>&2; exit 1; } | usage() { echo "Usage: garlic [-RFwv] trebuchet" 1>&2; exit 1; } | ||||||
| 
 | 
 | ||||||
| findClosure() { | findClosure() { | ||||||
| @ -32,6 +26,23 @@ findExperiment() { | |||||||
|   grep -o -- "/nix/store/.*-experiment" "$1" |   grep -o -- "/nix/store/.*-experiment" "$1" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | findOutputDir() { | ||||||
|  |   garlic_sandbox=$(nix show-config |\ | ||||||
|  |     grep extra-sandbox-paths |\ | ||||||
|  |     grep -o '/garlic=[^ ]*' || true) | ||||||
|  | 
 | ||||||
|  |   if [ -z "$garlic_sandbox" ]; then | ||||||
|  |     >&2 echo "Missing extra-sandbox-paths /garlic mountpoint" | ||||||
|  |     >&2 echo "Check the ~/.config/nix/nix.conf file" | ||||||
|  |     exit 1 | ||||||
|  |   fi | ||||||
|  | 
 | ||||||
|  |   mountdir_rel=$(echo "$garlic_sandbox" | sed 's@^/garlic=@@g') | ||||||
|  |   mountdir=$(readlink -f "$mountdir_rel") | ||||||
|  | 
 | ||||||
|  |   echo "$mountdir/cache" | ||||||
|  | } | ||||||
|  | 
 | ||||||
| drvFromOutput() { | drvFromOutput() { | ||||||
|   nix-store -q --deriver $1 |   nix-store -q --deriver $1 | ||||||
| } | } | ||||||
| @ -138,7 +149,7 @@ do_fetch() { | |||||||
| 
 | 
 | ||||||
|   test $verbose && >&2 echo "$exp: execution complete, fetching results" |   test $verbose && >&2 echo "$exp: execution complete, fetching results" | ||||||
| 
 | 
 | ||||||
|   mkdir -p $garlicTemp |   mkdir -p "$outputDir" | ||||||
| 
 | 
 | ||||||
|   rsync -rt --copy-links \ |   rsync -rt --copy-links \ | ||||||
|     --include='*/*/garlic_config.json' \ |     --include='*/*/garlic_config.json' \ | ||||||
| @ -147,15 +158,17 @@ do_fetch() { | |||||||
|     --include='*/*/*/.garlic' \ |     --include='*/*/*/.garlic' \ | ||||||
|     --include='*/*/*/.garlic/*' \ |     --include='*/*/*/.garlic/*' \ | ||||||
|     --exclude='*/*/*/*' \ |     --exclude='*/*/*/*' \ | ||||||
|     $exp $garlicTemp |     "$exp" "$outputDir" | ||||||
| 
 | 
 | ||||||
|  |   if [ ! $enableKeep ]; then | ||||||
|     nix-build -E "(with import ./default.nix; \ |     nix-build -E "(with import ./default.nix; \ | ||||||
|       garlic.pp.store { \ |       garlic.pp.store { \ | ||||||
|       experimentStage = import \"$experimentDrv\"; |         trebuchet = (import \"$trebuchetDrv\" ); \ | ||||||
|       trebuchetStage = import \"$trebuchetDrv\"; |         experiment = (import \"$experimentDrv\"); \ | ||||||
|       })" |       })" | ||||||
| 
 | 
 | ||||||
|   rm -rf $garlicTemp/$expName |     rm -rf "$outputDir/$expName" | ||||||
|  |   fi | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| do_run() { | do_run() { | ||||||
| @ -163,17 +176,26 @@ do_run() { | |||||||
|   >&2 $trebuchet |   >&2 $trebuchet | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | do_delete() { | ||||||
|  |   expName=$(basename $experiment) | ||||||
|  |   rm -rf $outputDir/$expName | ||||||
|  | } | ||||||
|  | 
 | ||||||
| waitResults=1 | waitResults=1 | ||||||
| verbose= | verbose= | ||||||
| operation= | operation= | ||||||
| target= | target= | ||||||
| enableRun= | enableRun= | ||||||
| enableFetch= | enableFetch= | ||||||
|  | enableKeep= | ||||||
|  | enableDelete= | ||||||
| 
 | 
 | ||||||
| while getopts "vwRF" o; do | while getopts "vwRFKD" o; do | ||||||
|   case "${o}" in |   case "${o}" in | ||||||
|     R) enableRun=1 ;; |     R) enableRun=1 ;; | ||||||
|     F) enableFetch=1 ;; |     F) enableFetch=1 ;; | ||||||
|  |     K) enableKeep=1 ;; | ||||||
|  |     D) enableDelete=1 ;; | ||||||
|     w) waitResults=0 ;; |     w) waitResults=0 ;; | ||||||
|     v) verbose=1 ;; |     v) verbose=1 ;; | ||||||
|     *) usage ;; |     *) usage ;; | ||||||
| @ -182,7 +204,7 @@ done | |||||||
| shift $((OPTIND-1)) | shift $((OPTIND-1)) | ||||||
| trebuchet="$1" | trebuchet="$1" | ||||||
| 
 | 
 | ||||||
| if [ -z "$enableRun" -a -z "$enableFetch" ]; then | if [ -z "$enableRun" -a -z "$enableFetch" -a -z "$enableDelete" ]; then | ||||||
|   >&2 echo "missing operation" |   >&2 echo "missing operation" | ||||||
|   usage |   usage | ||||||
| fi | fi | ||||||
| @ -196,6 +218,8 @@ checkMountpoint | |||||||
| 
 | 
 | ||||||
| checkTrebuchet $trebuchet | checkTrebuchet $trebuchet | ||||||
| 
 | 
 | ||||||
|  | outputDir=$(findOutputDir) | ||||||
|  | 
 | ||||||
| experiment=$(findExperiment "$trebuchet") | experiment=$(findExperiment "$trebuchet") | ||||||
| checkExperiment $experiment | checkExperiment $experiment | ||||||
| 
 | 
 | ||||||
| @ -204,3 +228,4 @@ experimentDrv=$(drvFromOutput $experiment) | |||||||
| 
 | 
 | ||||||
| if [ $enableRun ]; then do_run; fi | if [ $enableRun ]; then do_run; fi | ||||||
| if [ $enableFetch ]; then do_fetch; fi | if [ $enableFetch ]; then do_fetch; fi | ||||||
|  | if [ $enableDelete ]; then do_delete; fi | ||||||
|  | |||||||
| @ -35,7 +35,10 @@ rec { | |||||||
|       }; |       }; | ||||||
|     }; |     }; | ||||||
|   in trebuchet // rec { |   in trebuchet // rec { | ||||||
|     result = pp.resultFromLauncher (pp.launcher trebuchet); |     result = pp.store { | ||||||
|  |       trebuchet=trebuchet; | ||||||
|  |       experiment=trebuchet.experiment; | ||||||
|  |     }; | ||||||
|     timetable = pp.timetable result; |     timetable = pp.timetable result; | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user