{ pkgs, lib, lua, }: let inherit (lib.generators) toLua; requiredLuaModules = drvs: let modules = lib.filter hasLuaModule drvs; in lib.unique ([ lua ] ++ modules ++ lib.concatLists (lib.catAttrs "requiredLuaModules" modules)); # Check whether a derivation provides a lua module. hasLuaModule = drv: drv ? luaModule; # Use this to override the arguments passed to buildLuarocksPackage overrideLuarocks = drv: f: (drv.override ( args: args // { buildLuarocksPackage = drv: (args.buildLuarocksPackage drv).override f; } )) // { overrideScope = scope: overrideLuarocks (drv.overrideScope scope) f; }; in rec { inherit overrideLuarocks; inherit hasLuaModule requiredLuaModules; luaPathList = [ "share/lua/${lua.luaversion}/?.lua" "share/lua/${lua.luaversion}/?/init.lua" ]; luaCPathList = [ "lib/lua/${lua.luaversion}/?.so" ]; # generate paths without a prefix luaPathRelStr = lib.concatStringsSep ";" luaPathList; luaCPathRelStr = lib.concatStringsSep ";" luaCPathList; # generate LUA_(C)PATH value for a specific derivation, i.e., with absolute paths genLuaPathAbsStr = drv: lib.concatMapStringsSep ";" (x: "${drv}/${x}") luaPathList; genLuaCPathAbsStr = drv: lib.concatMapStringsSep ";" (x: "${drv}/${x}") luaCPathList; # Generate a LUA_PATH with absolute paths # genLuaPathAbs = drv: # lib.concatStringsSep ";" (map (x: "${drv}/x") luaPathList); luaAtLeast = lib.versionAtLeast lua.luaversion; luaOlder = lib.versionOlder lua.luaversion; isLua51 = (lib.versions.majorMinor lua.version) == "5.1"; isLua52 = (lib.versions.majorMinor lua.version) == "5.2"; isLua53 = lua.luaversion == "5.3"; isLuaJIT = lib.getName lua == "luajit"; /* generates the relative path towards the folder where seems stable even when using lua_modules_path = "" Example: getDataFolder luaPackages.stdlib => stdlib-41.2.2-1-rocks/stdlib/41.2.2-1/doc */ getDataFolder = drv: "${drv.pname}-${drv.version}-rocks/${drv.pname}/${drv.version}"; /* Convert derivation to a lua module. so that luaRequireModules can be run later */ toLuaModule = drv: drv.overrideAttrs (oldAttrs: { # Use passthru in order to prevent rebuilds when possible. passthru = (oldAttrs.passthru or { }) // { luaModule = lua; requiredLuaModules = requiredLuaModules drv.propagatedBuildInputs; }; }); /* generate a luarocks config conforming to: https://github.com/luarocks/luarocks/wiki/Config-file-format The config lists folders where to find lua dependencies Example: generateLuarocksConfig { externalDeps = [ { name = "CRYPTO"; dep = pkgs.openssl; } ]; rocksSubdir = "subdir"; }; Type: generateLuarocksConfig :: AttrSet -> String */ generateLuarocksConfig = { externalDeps ? [ ], # a list of lua derivations requiredLuaRocks ? [ ], ... }@args: let rocksTrees = lib.imap0 (i: dep: { name = "dep-${toString i}"; root = "${dep}"; # packages built by buildLuaPackage or luarocks doesn't contain rocksSubdir # hence a default here rocks_dir = if dep ? rocksSubdir then "${dep}/${dep.rocksSubdir}" else "${dep.pname}-${dep.version}-rocks"; }) requiredLuaRocks; # Explicitly point luarocks to the relevant locations for multiple-output # derivations that are external dependencies, to work around an issue it has # (https://github.com/luarocks/luarocks/issues/766) depVariables = zipAttrsWithLast ( lib.lists.map ( { name, dep }: { "${name}_INCDIR" = "${lib.getDev dep}/include"; "${name}_LIBDIR" = "${lib.getLib dep}/lib"; "${name}_BINDIR" = "${lib.getBin dep}/bin"; } ) externalDeps' ); zipAttrsWithLast = lib.attrsets.zipAttrsWith (name: lib.lists.last); # example externalDeps': [ { name = "CRYPTO"; dep = pkgs.openssl; } ] externalDeps' = lib.filter (dep: !lib.isDerivation dep) externalDeps; externalDepsDirs = map (x: builtins.toString x) (lib.filter (lib.isDerivation) externalDeps); generatedConfig = ( { # first tree is the default target where new rocks are installed, # any other trees in the list are treated as additional sources of installed rocks for matching dependencies. rocks_trees = ( [ { name = "current"; root = "${placeholder "out"}"; rocks_dir = "current"; } ] ++ rocksTrees ); } // lib.optionalAttrs lua.pkgs.isLuaJIT { # Luajit provides some additional functionality built-in; this exposes # that to luarock's dependency system rocks_provided = { jit = "${lua.luaversion}-1"; ffi = "${lua.luaversion}-1"; luaffi = "${lua.luaversion}-1"; bit = "${lua.luaversion}-1"; }; } // { # For single-output external dependencies external_deps_dirs = externalDepsDirs; # Some needed machinery to handle multiple-output external dependencies, # as per https://github.com/luarocks/luarocks/issues/766 variables = depVariables; } // removeAttrs args [ "requiredLuaRocks" "externalDeps" ] ); in generatedConfig; }