query
On this page

mixRelease

pkgs.beamPackages.mixRelease

Functor
Docs pulled from | This Revision | about 1 hour ago


Contribute
Enhance the ecosystem with your expertise! Contribute to fill the gaps in documentation. Your input can make a difference.

Noogle detected

This is a Functor

Learn about functors

Implementation

The following is the current implementation of this function.

{
  pname,
  version,
  src,
  nativeBuildInputs ? [ ],
  buildInputs ? [ ],
  meta ? { },
  enableDebugInfo ? false,
  mixEnv ? "prod",
  mixTarget ? "host",
  compileFlags ? [ ],
  # Build a particular named release.
  # see https://hexdocs.pm/mix/1.12/Mix.Tasks.Release.html#content
  mixReleaseName ? "",
  # If set, the given escript binary will be copied to the output
  # instead of the release
  escriptBinName ? null,

  # Options to be passed to the Erlang compiler. As documented in the reference
  # manual, these must be valid Erlang terms. They will be turned into an
  # erlang list and set as the ERL_COMPILER_OPTIONS environment variable.
  # See https://www.erlang.org/doc/man/compile
  erlangCompilerOptions ? [ ],

  # Deterministic Erlang builds remove full system paths from debug information
  # among other things to keep builds more reproducible. See their docs for more:
  # https://www.erlang.org/doc/man/compile
  erlangDeterministicBuilds ? true,

  # Mix dependencies provided as a fixed output derivation
  mixFodDeps ? null,

  # Mix dependencies generated by mix2nix
  #
  # This assumes each dependency is built by buildMix or buildRebar3. Each
  # dependency needs to have a setup hook to add the lib path to $ERL_LIBS.
  # This is how Mix finds dependencies.
  mixNixDeps ? { },

  elixir ? inputs.elixir,
  erlang ? inputs.erlang,
  hex ? inputs.hex.override { inherit elixir; },

  # Remove releases/COOKIE
  #
  # People have different views on the nature of cookies. Some believe that they are
  # secrets, while others believe they are just ids for clustering nodes instead of
  # secrets.
  #
  # If you think cookie is secret, you can set this attr to true, then it will be
  # removed from nix store. If not, you can set it to false.
  #
  # For backward compatibility, it is set to true by default.
  #
  # You can always specify a custom cookie by using RELEASE_COOKIE environment
  # variable, regardless of the value of this attr.
  removeCookie ? true,

  # This reduces closure size, but can lead to some hard to understand runtime
  # errors, so use with caution. See e.g.
  # https://github.com/whitfin/cachex/issues/205
  # https://framagit.org/framasoft/mobilizon/-/issues/1169
  stripDebug ? false,

  ...
}@attrs:
let
  # Remove non standard attributes that cannot be coerced to strings
  overridable = removeAttrs attrs [
    "compileFlags"
    "erlangCompilerOptions"
    "mixNixDeps"
  ];
in
assert mixNixDeps != { } -> mixFodDeps == null;
assert stripDebug -> !enableDebugInfo;
assert escriptBinName != null -> mixReleaseName == "";

stdenv.mkDerivation (
  overridable
  // {
    nativeBuildInputs =
      nativeBuildInputs
      ++
        # Erlang/Elixir deps
        [
          erlang
          elixir
          hex
          git
          mixBuildDirHook
        ]
      ++
        # Mix deps
        (builtins.attrValues mixNixDeps)
      ++
        # other compile-time deps
        [
          findutils
          ripgrep
          bbe
          makeWrapper
        ];

    buildInputs = buildInputs ++ lib.optionals (escriptBinName != null) [ erlang ];

    __darwinAllowLocalNetworking = true;

    env = {
      MIX_ENV = mixEnv;
      MIX_TARGET = mixTarget;
      MIX_BUILD_PREFIX = (if mixTarget == "host" then "" else "${mixTarget}_") + "${mixEnv}";
      MIX_DEBUG = if enableDebugInfo then 1 else 0;
      HEX_OFFLINE = 1;

      DEBUG = if enableDebugInfo then 1 else 0; # for Rebar3 compilation
      # The API with `mix local.rebar rebar path` makes a copy of the binary
      # some older dependencies still use rebar.
      MIX_REBAR = "${rebar}/bin/rebar";
      MIX_REBAR3 = "${rebar3}/bin/rebar3";

      ERL_COMPILER_OPTIONS =
        let
          options = erlangCompilerOptions ++ lib.optionals erlangDeterministicBuilds [ "deterministic" ];
        in
        "[${lib.concatStringsSep "," options}]";

      LANG = if stdenv.hostPlatform.isLinux then "C.UTF-8" else "C";
      LC_CTYPE = if stdenv.hostPlatform.isLinux then "C.UTF-8" else "UTF-8";
    }
    // (attrs.env or { });

    postUnpack = ''
      # Mix and Hex
      export MIX_HOME="$TEMPDIR/mix"
      export HEX_HOME="$TEMPDIR/hex"

      # Rebar
      export REBAR_GLOBAL_CONFIG_DIR="$TEMPDIR/rebar3"
      export REBAR_CACHE_DIR="$TEMPDIR/rebar3.cache"

      ${lib.optionalString (mixFodDeps != null) ''
        # Compilation of the dependencies will require that the dependency path is
        # writable, thus a copy to the $TEMPDIR is inevitable here.
        export MIX_DEPS_PATH="$TEMPDIR/deps"
        cp --no-preserve=mode -R "${mixFodDeps}" "$MIX_DEPS_PATH"
      ''}
    ''
    + (attrs.postUnpack or "");

    configurePhase =
      attrs.configurePhase or ''
        runHook preConfigure

        # This is needed for projects that have a specific compile step
        # the dependency needs to be compiled in order for the task
        # to be available.
        #
        # Phoenix projects for example will need compile.phoenix.
        mix deps.compile --no-deps-check --skip-umbrella-children

        # Symlink dependency sources. This is needed for projects that require
        # access to the source of their dependencies. For example, Phoenix
        # projects need javascript assets to build asset bundles.
        ${lib.optionalString (mixNixDeps != { }) ''
          mkdir -p deps

          ${lib.concatMapAttrsStringSep "\n" (name: dep: ''
            dep_path="deps/${name}"
            if [ -d "${dep}/src" ]; then
              ln -sv ${dep}/src $dep_path
            fi
          '') mixNixDeps}
        ''}

        # Symlink deps to build root. Similar to above, but allows for mixFodDeps
        # Phoenix projects to find javascript assets.
        ${lib.optionalString (mixFodDeps != null) ''
          ln -s "$MIX_DEPS_PATH" ./deps
        ''}

        runHook postConfigure
      '';

    buildPhase =
      attrs.buildPhase or ''
        runHook preBuild

        mix compile --no-deps-check ${lib.concatStringsSep " " compileFlags}

        ${lib.optionalString (escriptBinName != null) ''
          mix escript.build --no-deps-check
        ''}

        runHook postBuild
      '';

    installPhase =
      attrs.installPhase or ''
        runHook preInstall

        ${
          if (escriptBinName != null) then
            ''
              mkdir -p $out/bin
              cp ${escriptBinName} $out/bin
            ''
          else
            ''
              mix release ${mixReleaseName} --no-deps-check --path "$out"
            ''
        }

        runHook postInstall
      '';

    postFixup = ''
      echo "removing files for Microsoft Windows"
      rm -f "$out"/bin/*.bat

      echo "wrapping programs in $out/bin with their runtime deps"
      for f in $(find $out/bin/ -type f -executable); do
        wrapProgram "$f" \
          --prefix PATH : ${
            lib.makeBinPath [
              coreutils
              gnused
              gnugrep
              gawk
            ]
          }
      done
    ''
    + lib.optionalString removeCookie ''
      if [ -e $out/releases/COOKIE ]; then
        echo "removing $out/releases/COOKIE"
        rm $out/releases/COOKIE
      fi
    ''
    + ''
      if [ -e $out/erts-* ]; then
        # ERTS is included in the release, then erlang is not required as a runtime dependency.
        # But, erlang is still referenced in some places. To removed references to erlang,
        # following steps are required.

        # 1. remove references to erlang from plain text files
        for file in $(rg "${erlang}/lib/erlang" "$out" --files-with-matches); do
          echo "removing references to erlang in $file"
          substituteInPlace "$file" --replace "${erlang}/lib/erlang" "$out"
        done

        # 2. remove references to erlang from .beam files
        #
        # No need to do anything, because it has been handled by "deterministic" option specified
        # by ERL_COMPILER_OPTIONS.

        # 3. remove references to erlang from normal binary files
        for file in $(rg "${erlang}/lib/erlang" "$out" --files-with-matches --binary --iglob '!*.beam'); do
          echo "removing references to erlang in $file"
          # use bbe to substitute strings in binary files, because using substituteInPlace
          # on binaries will raise errors
          bbe -e "s|${erlang}/lib/erlang|$out|" -o "$file".tmp "$file"
          rm -f "$file"
          mv "$file".tmp "$file"
        done

        # References to erlang should be removed from output after above processing.
      fi
    ''
    + lib.optionalString stripDebug ''
      # Strip debug symbols to avoid hardreferences to "foreign" closures actually
      # not needed at runtime, while at the same time reduce size of BEAM files.
      erl -noinput -eval 'lists:foreach(fun(F) -> io:format("Stripping ~p.~n", [F]), beam_lib:strip(F) end, filelib:wildcard("'"$out"'/**/*.beam"))' -s init stop
    '';
  }
)