summaryrefslogblamecommitdiffhomepage
path: root/test/test_njs.py
blob: 23d1df1b97e8793e08cb59445b191810c1e95fd3 (plain) (tree)
1
2
3
4
5
6
7
8
9
                        
 
             
 
                                                    
                              
                                   
 
                                           
 
                           
 
 



                                    
                                                        


                                                                        
 
 

                                             
                                         
 
                                                     
                                                  
 
 

                                                                   
 
 


                                                          
 
 

                                                          
 


                                      
 

                                              
 
 

                                    
 



                                          
 
 

                                                      
 

                                                                
 
 

















                                                                         































                                                                
 

















                                                                       








                                                                        
                                 






                                                                          
                                            
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}`"')