Skip to content

Commit

Permalink
get lua version from luarocks itself
Browse files Browse the repository at this point in the history
  • Loading branch information
asottile committed Jan 17, 2022
1 parent 3f8be74 commit 54331dc
Show file tree
Hide file tree
Showing 3 changed files with 28 additions and 145 deletions.
108 changes: 24 additions & 84 deletions pre_commit/languages/lua.py
@@ -1,6 +1,6 @@
import contextlib
import os
import re
import sys
from typing import Generator
from typing import Sequence
from typing import Tuple
Expand All @@ -11,7 +11,6 @@
from pre_commit.envcontext import Var
from pre_commit.hook import Hook
from pre_commit.languages import helpers
from pre_commit.parse_shebang import find_executable
from pre_commit.prefix import Prefix
from pre_commit.util import clean_path_on_failure
from pre_commit.util import cmd_output
Expand All @@ -21,95 +20,38 @@
healthy = helpers.basic_healthy


def _find_lua(language_version: str) -> str: # pragma: win32 no cover
"""Find a lua executable.
Lua doesn't always have a plain `lua` executable.
Some OS vendors will ship the binary as `lua#.#` (e.g., lua5.3)
so discovery is needed to find a valid executable.
"""
if language_version == C.DEFAULT:
choices = ['lua']
for path in os.environ.get('PATH', '').split(os.pathsep):
try:
candidates = os.listdir(path)
except OSError:
# Invalid path on PATH or lacking permissions.
continue

for candidate in candidates:
# The Lua executable might look like `lua#.#` or `lua-#.#`.
if re.search(r'^lua[-]?\d+\.\d+', candidate):
choices.append(candidate)
else:
# Prefer version specific executables first if available.
# This should avoid the corner case where a user requests a language
# version, gets a `lua` executable, but that executable is actually
# for a different version and package.path would patch LUA_PATH
# incorrectly.
choices = [f'lua{language_version}', 'lua-{language_version}', 'lua']

found_exes = [exe for exe in choices if find_executable(exe)]
if found_exes:
return found_exes[0]

raise ValueError(
'No lua executable found on the system paths '
f'for {language_version} version.',
)
def _get_lua_version() -> str: # pragma: win32 no cover
"""Get the Lua version used in file paths."""
_, stdout, _ = cmd_output('luarocks', 'config', '--lua-ver')
return stdout.strip()


def _get_lua_path_version(
lua_executable: str,
) -> str: # pragma: win32 no cover
"""Get the Lua version used in file paths."""
# This could sniff out from _VERSION, but checking package.path should
# provide an answer for *exactly* where lua is looking for packages.
_, stdout, _ = cmd_output(lua_executable, '-e', 'print(package.path)')
sep = os.sep if os.name != 'nt' else os.sep * 2
match = re.search(fr'{sep}lua{sep}(.*?){sep}', stdout)
if match:
return match[1]

raise ValueError('Cannot determine lua version for file paths.')


def get_env_patch(
env: str, language_version: str,
) -> PatchesT: # pragma: win32 no cover
lua = _find_lua(language_version)
version = _get_lua_path_version(lua)
def get_env_patch(d: str) -> PatchesT: # pragma: win32 no cover
version = _get_lua_version()
so_ext = 'dll' if sys.platform == 'win32' else 'so'
return (
('PATH', (os.path.join(env, 'bin'), os.pathsep, Var('PATH'))),
('PATH', (os.path.join(d, 'bin'), os.pathsep, Var('PATH'))),
(
'LUA_PATH', (
os.path.join(env, 'share', 'lua', version, '?.lua;'),
os.path.join(env, 'share', 'lua', version, '?', 'init.lua;;'),
os.path.join(d, 'share', 'lua', version, '?.lua;'),
os.path.join(d, 'share', 'lua', version, '?', 'init.lua;;'),
),
),
(
'LUA_CPATH', (
os.path.join(env, 'lib', 'lua', version, '?.so;;'),
),
'LUA_CPATH',
(os.path.join(d, 'lib', 'lua', version, f'?.{so_ext};;'),),
),
)


def _envdir(prefix: Prefix, version: str) -> str: # pragma: win32 no cover
directory = helpers.environment_dir(ENVIRONMENT_DIR, version)
def _envdir(prefix: Prefix) -> str: # pragma: win32 no cover
directory = helpers.environment_dir(ENVIRONMENT_DIR, C.DEFAULT)
return prefix.path(directory)


@contextlib.contextmanager # pragma: win32 no cover
def in_env(
prefix: Prefix,
language_version: str,
) -> Generator[None, None, None]:
with envcontext(
get_env_patch(
_envdir(prefix, language_version), language_version,
),
):
def in_env(prefix: Prefix) -> Generator[None, None, None]:
with envcontext(get_env_patch(_envdir(prefix))):
yield


Expand All @@ -120,19 +62,17 @@ def install_environment(
) -> None: # pragma: win32 no cover
helpers.assert_version_default('lua', version)

envdir = _envdir(prefix, version)
envdir = _envdir(prefix)
with clean_path_on_failure(envdir):
with in_env(prefix, version):
with in_env(prefix):
# luarocks doesn't bootstrap a tree prior to installing
# so ensure the directory exists.
os.makedirs(envdir, exist_ok=True)

make_cmd = ['luarocks', '--tree', envdir, 'make']
# Older luarocks (e.g., 2.4.2) expect the rockspec as an argument.
filenames = prefix.star('.rockspec')
make_cmd.extend(filenames[:1])

helpers.run_setup_cmd(prefix, tuple(make_cmd))
# Older luarocks (e.g., 2.4.2) expect the rockspec as an arg
for rockspec in prefix.star('.rockspec'):
make_cmd = ('luarocks', '--tree', envdir, 'make', rockspec)
helpers.run_setup_cmd(prefix, make_cmd)

# luarocks can't install multiple packages at once
# so install them individually.
Expand All @@ -146,5 +86,5 @@ def run_hook(
file_args: Sequence[str],
color: bool,
) -> Tuple[int, bytes]: # pragma: win32 no cover
with in_env(hook.prefix, hook.language_version):
with in_env(hook.prefix):
return helpers.run_xargs(hook, hook.cmd, file_args, color=color)
55 changes: 0 additions & 55 deletions tests/languages/lua_test.py

This file was deleted.

10 changes: 4 additions & 6 deletions tests/repository_test.py
Expand Up @@ -17,7 +17,6 @@
from pre_commit.hook import Hook
from pre_commit.languages import golang
from pre_commit.languages import helpers
from pre_commit.languages import lua
from pre_commit.languages import node
from pre_commit.languages import python
from pre_commit.languages import ruby
Expand Down Expand Up @@ -1142,18 +1141,17 @@ def test_lua_hook(tempdir_factory, store):

@skipif_cant_run_lua # pragma: win32 no cover
def test_local_lua_additional_dependencies(store):
lua_entry = lua._find_lua(C.DEFAULT)
config = {
'repo': 'local',
'hooks': [{
'id': 'local-lua',
'name': 'local-lua',
'entry': lua_entry,
'entry': 'luacheck --version',
'language': 'lua',
'args': ['-e', 'require "inspect"; print("hello world")'],
'additional_dependencies': ['inspect'],
'additional_dependencies': ['luacheck'],
}],
}
hook = _get_hook(config, store, 'local-lua')
ret, out = _hook_run(hook, (), color=False)
assert (ret, _norm_out(out)) == (0, b'hello world\n')
assert b'Luacheck' in out
assert ret == 0

0 comments on commit 54331dc

Please sign in to comment.