|
| 1 | +#!/usr/bin/env bash |
| 2 | +set -euo pipefail |
| 3 | + |
| 4 | +if [ "$#" -ne 1 ]; then |
| 5 | + echo "usage: $0 <conda-executable>" >&2 |
| 6 | + exit 2 |
| 7 | +fi |
| 8 | + |
| 9 | +conda_executable=$1 |
| 10 | + |
| 11 | +script_dir=$(CDPATH= cd -- "$(dirname -- "$0")" && pwd) |
| 12 | +envs_dir="$script_dir/.conda-envs" |
| 13 | +pkgs_dir="$script_dir/.conda-pkgs" |
| 14 | +setup_tag="qdata-flow-compat-v1" |
| 15 | + |
| 16 | +hash_file() { |
| 17 | + if command -v sha256sum >/dev/null 2>&1; then |
| 18 | + sha256sum "$1" | awk '{print $1}' |
| 19 | + elif command -v shasum >/dev/null 2>&1; then |
| 20 | + shasum -a 256 "$1" | awk '{print $1}' |
| 21 | + else |
| 22 | + echo "missing sha256 tool" >&2 |
| 23 | + exit 2 |
| 24 | + fi |
| 25 | +} |
| 26 | + |
| 27 | +has_conda_meta() { |
| 28 | + local env_prefix=$1 |
| 29 | + local package_name=$2 |
| 30 | + |
| 31 | + compgen -G "$env_prefix/conda-meta/$package_name-*.json" >/dev/null |
| 32 | +} |
| 33 | + |
| 34 | +capture_versions() { |
| 35 | + local env_prefix=$1 |
| 36 | + |
| 37 | + CONDA_PKGS_DIRS="$pkgs_dir" "$conda_executable" run --prefix "$env_prefix" \ |
| 38 | + Rscript -e 'pkgs <- c("Rcpp", "RcppParallel"); stopifnot(all(vapply(pkgs, requireNamespace, logical(1), quietly = TRUE))); cat(as.character(packageVersion("Rcpp")), "\n", as.character(packageVersion("RcppParallel")), "\n", sep = "")' |
| 39 | +} |
| 40 | + |
| 41 | +write_stamp() { |
| 42 | + local env_kind=$1 |
| 43 | + local source_kind=$2 |
| 44 | + local env_prefix="$envs_dir/$env_kind" |
| 45 | + local env_file="$script_dir/environment-$env_kind.yml" |
| 46 | + local stamp_file="$env_prefix/.qdata-flow-compat-stamp" |
| 47 | + local versions |
| 48 | + local rcpp_version |
| 49 | + local rcppparallel_version |
| 50 | + |
| 51 | + versions=$(capture_versions "$env_prefix") |
| 52 | + rcpp_version=$(printf '%s\n' "$versions" | sed -n '1p') |
| 53 | + rcppparallel_version=$(printf '%s\n' "$versions" | sed -n '2p') |
| 54 | + |
| 55 | + cat > "$stamp_file" <<EOF |
| 56 | +STAMP_ENV_HASH=$(hash_file "$env_file") |
| 57 | +STAMP_SETUP_TAG=$setup_tag |
| 58 | +STAMP_SOURCE_KIND=$source_kind |
| 59 | +STAMP_RCPP_VERSION=$rcpp_version |
| 60 | +STAMP_RCPPPARALLEL_VERSION=$rcppparallel_version |
| 61 | +EOF |
| 62 | +} |
| 63 | + |
| 64 | +env_is_ready() { |
| 65 | + local env_kind=$1 |
| 66 | + local source_kind=$2 |
| 67 | + local env_prefix="$envs_dir/$env_kind" |
| 68 | + local env_file="$script_dir/environment-$env_kind.yml" |
| 69 | + local stamp_file="$env_prefix/.qdata-flow-compat-stamp" |
| 70 | + local versions |
| 71 | + local rcpp_version |
| 72 | + local rcppparallel_version |
| 73 | + |
| 74 | + if [ ! -x "$env_prefix/bin/Rscript" ] || [ ! -f "$stamp_file" ]; then |
| 75 | + return 1 |
| 76 | + fi |
| 77 | + |
| 78 | + STAMP_ENV_HASH= |
| 79 | + STAMP_SETUP_TAG= |
| 80 | + STAMP_SOURCE_KIND= |
| 81 | + STAMP_RCPP_VERSION= |
| 82 | + STAMP_RCPPPARALLEL_VERSION= |
| 83 | + # shellcheck disable=SC1090 |
| 84 | + . "$stamp_file" |
| 85 | + |
| 86 | + if [ "$STAMP_ENV_HASH" != "$(hash_file "$env_file")" ]; then |
| 87 | + return 1 |
| 88 | + fi |
| 89 | + if [ "$STAMP_SETUP_TAG" != "$setup_tag" ]; then |
| 90 | + return 1 |
| 91 | + fi |
| 92 | + if [ "$STAMP_SOURCE_KIND" != "$source_kind" ]; then |
| 93 | + return 1 |
| 94 | + fi |
| 95 | + |
| 96 | + case "$source_kind" in |
| 97 | + conda) |
| 98 | + has_conda_meta "$env_prefix" r-rcppparallel || return 1 |
| 99 | + ;; |
| 100 | + cran) |
| 101 | + if has_conda_meta "$env_prefix" r-rcppparallel; then |
| 102 | + return 1 |
| 103 | + fi |
| 104 | + ;; |
| 105 | + *) |
| 106 | + echo "unexpected source kind: $source_kind" >&2 |
| 107 | + exit 2 |
| 108 | + ;; |
| 109 | + esac |
| 110 | + |
| 111 | + if [ "$env_kind" = mixed ]; then |
| 112 | + has_conda_meta "$env_prefix" tbb-devel || return 1 |
| 113 | + fi |
| 114 | + |
| 115 | + if ! versions=$(capture_versions "$env_prefix"); then |
| 116 | + return 1 |
| 117 | + fi |
| 118 | + |
| 119 | + rcpp_version=$(printf '%s\n' "$versions" | sed -n '1p') |
| 120 | + rcppparallel_version=$(printf '%s\n' "$versions" | sed -n '2p') |
| 121 | + |
| 122 | + [ "$rcpp_version" = "$STAMP_RCPP_VERSION" ] || return 1 |
| 123 | + [ "$rcppparallel_version" = "$STAMP_RCPPPARALLEL_VERSION" ] || return 1 |
| 124 | +} |
| 125 | + |
| 126 | +rebuild_env() { |
| 127 | + local env_kind=$1 |
| 128 | + local env_file="$script_dir/environment-$env_kind.yml" |
| 129 | + local env_prefix="$envs_dir/$env_kind" |
| 130 | + local log_file |
| 131 | + |
| 132 | + rm -rf "$env_prefix" |
| 133 | + mkdir -p "$envs_dir" "$pkgs_dir" |
| 134 | + log_file=$(mktemp) |
| 135 | + |
| 136 | + if CONDA_PKGS_DIRS="$pkgs_dir" "$conda_executable" env create --prefix "$env_prefix" --file "$env_file" --yes 2>&1 | tee "$log_file"; then |
| 137 | + rm -f "$log_file" |
| 138 | + return |
| 139 | + fi |
| 140 | + |
| 141 | + if grep -Eiq 'SafetyError|corrupt|incorrect size' "$log_file"; then |
| 142 | + echo "Detected corrupt local conda package cache, clearing $pkgs_dir and retrying once" |
| 143 | + rm -rf "$pkgs_dir" "$env_prefix" |
| 144 | + mkdir -p "$pkgs_dir" |
| 145 | + CONDA_PKGS_DIRS="$pkgs_dir" "$conda_executable" env create --prefix "$env_prefix" --file "$env_file" --yes |
| 146 | + rm -f "$log_file" |
| 147 | + return |
| 148 | + fi |
| 149 | + |
| 150 | + rm -f "$log_file" |
| 151 | + return 1 |
| 152 | +} |
| 153 | + |
| 154 | +install_cran_rcppparallel() { |
| 155 | + local env_kind=$1 |
| 156 | + local env_prefix="$envs_dir/$env_kind" |
| 157 | + |
| 158 | + CONDA_PKGS_DIRS="$pkgs_dir" "$conda_executable" run --prefix "$env_prefix" Rscript -e 'options(repos = c(CRAN = "https://cloud.r-project.org")); install.packages("RcppParallel", type = "source")' |
| 159 | +} |
| 160 | + |
| 161 | +ensure_env() { |
| 162 | + local env_kind=$1 |
| 163 | + local source_kind=$2 |
| 164 | + |
| 165 | + if env_is_ready "$env_kind" "$source_kind"; then |
| 166 | + echo "Reusing $env_kind env" |
| 167 | + return |
| 168 | + fi |
| 169 | + |
| 170 | + echo "Rebuilding $env_kind env" |
| 171 | + rebuild_env "$env_kind" |
| 172 | + if [ "$source_kind" = cran ]; then |
| 173 | + install_cran_rcppparallel "$env_kind" |
| 174 | + fi |
| 175 | + write_stamp "$env_kind" "$source_kind" |
| 176 | +} |
| 177 | + |
| 178 | +run_probe() { |
| 179 | + local env_kind=$1 |
| 180 | + local env_prefix="$envs_dir/$env_kind" |
| 181 | + |
| 182 | + CONDA_PKGS_DIRS="$pkgs_dir" "$conda_executable" run --prefix "$env_prefix" Rscript "$script_dir/tbb_flow_compat.R" "$script_dir/tbb_flow_compat_probe.cpp" |
| 183 | +} |
| 184 | + |
| 185 | +echo "Testing RcppParallel from conda (oneTBB)" |
| 186 | +ensure_env conda conda |
| 187 | +run_probe conda |
| 188 | + |
| 189 | +echo "Testing RcppParallel from CRAN (TBB 2019)" |
| 190 | +ensure_env cran cran |
| 191 | +run_probe cran |
| 192 | + |
| 193 | +echo "Testing RcppParallel from CRAN with system tbb-devel (mixed header test)" |
| 194 | +ensure_env mixed cran |
| 195 | +run_probe mixed |
0 commit comments