from pathlib import Path
import pytest
from unit.applications.proto import ApplicationProto
from unit.option import option
from unit.utils import waitforfiles
prerequisites = {'modules': {'njs': 'any'}}
client = ApplicationProto()
@pytest.fixture(autouse=True)
def setup_method_fixture(temp_dir):
assert 'success' in client.conf(
{
"listeners": {"*:8080": {"pass": "routes"}},
"routes": [{"action": {"share": f"{temp_dir}/assets$uri"}}],
}
)
def create_files(*files):
assets_dir = f'{option.temp_dir}/assets/'
Path(assets_dir).mkdir(exist_ok=True)
_ = [Path(assets_dir + f).touch() for f in files]
waitforfiles(*[assets_dir + f for f in files])
def set_share(share):
assert 'success' in client.conf(share, 'routes/0/action/share')
def check_expression(expression, url='/'):
set_share(f'"`{option.temp_dir}/assets{expression}`"')
assert client.get(url=url)['status'] == 200
def test_njs_template_string(temp_dir):
create_files('str', '`string`', '`backtick', 'l1\nl2')
check_expression('/str')
check_expression('/\\\\`backtick')
check_expression('/l1\\nl2')
set_share(f'"{temp_dir}/assets/`string`"')
assert client.get()['status'] == 200
def test_njs_template_expression():
create_files('str', 'localhost')
check_expression('${uri}', '/str')
check_expression('${uri}${host}')
check_expression('${uri + host}')
check_expression('${uri + `${host}`}')
def test_njs_iteration():
create_files('Connection,Host', 'close,localhost')
check_expression('/${Object.keys(headers).sort().join()}')
check_expression('/${Object.values(headers).sort().join()}')
def test_njs_variables(temp_dir):
create_files('str', 'localhost', '127.0.0.1')
check_expression('/${host}')
check_expression('/${remoteAddr}')
check_expression('/${headers.Host}')
set_share(f'"`{temp_dir}/assets/${{cookies.foo}}`"')
assert (
client.get(headers={'Cookie': 'foo=str', 'Connection': 'close'})[
'status'
]
== 200
), 'cookies'
set_share(f'"`{temp_dir}/assets/${{args.foo}}`"')
assert client.get(url='/?foo=str')['status'] == 200, 'args'
check_expression('/${vars.header_host}')
set_share(f'"`{temp_dir}/assets/${{vars[\\"arg_foo\\"]}}`"')
assert client.get(url='/?foo=str')['status'] == 200, 'vars'
set_share(f'"`{temp_dir}/assets/${{vars.non_exist}}`"')
assert client.get()['status'] == 404, 'undefined'
create_files('undefined')
assert client.get()['status'] == 200, 'undefined 2'
def test_njs_variables_cacheable(temp_dir):
create_files('str')
def check_rewrite(rewrite, uri):
assert 'success' in client.conf(
[
{
"action": {
"rewrite": rewrite,
"share": f"`{temp_dir}/assets{uri}`",
},
},
],
'routes',
)
assert client.get()['status'] == 200
check_rewrite('/str', '${uri}')
check_rewrite('/str', '${vars.uri}')
def test_njs_variables_cacheable_access_log(findall, temp_dir):
assert 'success' in client.conf({"return": 200}, 'routes/0/action')
assert 'success' in client.conf(
{
'path': f'{temp_dir}/access.log',
'format': '`${vars.host}, ${vars.status}\n`',
},
'access_log'
), 'access_log configure'
reqs = 50
for _ in range(reqs):
client.get()
assert len(findall(r'localhost, 200', 'access.log')) == reqs
def test_njs_invalid(skip_alert):
skip_alert(r'js exception:')
def check_invalid(template):
assert 'error' in client.conf(template, 'routes/0/action/share')
check_invalid('"`a"')
check_invalid('"`a``"')
check_invalid('"`a`/"')
check_invalid('"`${vars.}`"')
def check_invalid_resolve(template):
assert 'success' in client.conf(template, 'routes/0/action/share')
assert client.get()['status'] == 500
check_invalid_resolve('"`${a}`"')
check_invalid_resolve('"`${uri.a.a}`"')
check_invalid_resolve('"`${vars.a.a}`"')