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

                                                        
 
                              
 


                                
                                        
                                    



                                  
 






                                                                     
 


                                                                      
 



















                                         
 
 

                                         
 
                                                  
 


                                                                
 
 

                                               
 
                               
 

                                                                      
 
 

                                                
 
                       
 

                                                                       
 
 


                                        
 


                                                        
 
 

                                       
 
                                                      
 
 

                                             
 
                                                   
 
 

                                             
 


                                                             
 
 

                                              
 


                                                              
 
 

                                                        
 


                                                              
 
 

                                       
 
                       
 
                                                               
 
 

                                         
 


                                                               
 
 

                                           
 
                               
 
                                                                   
 
 

                                       
 
                               
 
                                                               
 
 


                                         
 
                       
 
                                                                 
 
 








                                                   
 
                                                        
 
 

















                                                                       
 
 

                                                        
 








































                                                                          
             











                                                                             
 

                                                  
 




                                                        
 

                                              
 

                                                                             
 

                          
 

                              
 

                                           
 
 

                                          
 
                                                               
 






                                                
 
 


                                                        
 


                                               
 
 

                                          
 
                                                         
 
 


                                        
 
                                                       
 
 

                                       
 
                                                   
 
 

                                       
 
                                                             
 
 

                                       
 
                              
 
                                                               
 
 


                                                           
 
                                                                  
 






























                                                                            
 
                                               
 
 

                                      
 
                       
 
                                          
 










































                                                                      
 
                                                
 
                                                                           
 
                    
 
                                                        
import re
import subprocess

import pytest
from unit.applications.lang.ruby import ApplicationRuby

prerequisites = {'modules': {'ruby': 'all'}}

client = ApplicationRuby()


def test_ruby_application(date_to_sec_epoch, sec_epoch):
    client.load('variables')

    body = 'Test body string.'

    resp = client.post(
        headers={
            'Host': 'localhost',
            'Content-Type': 'text/html',
            'Custom-Header': 'blah',
            'Connection': 'close',
        },
        body=body,
    )

    assert resp['status'] == 200, 'status'
    headers = resp['headers']
    header_server = headers.pop('Server')
    assert re.search(r'Unit/[\d\.]+', header_server), 'server header'
    assert (
        headers.pop('Server-Software') == header_server
    ), 'server software header'

    date = headers.pop('Date')
    assert date[-4:] == ' GMT', 'date header timezone'
    assert abs(date_to_sec_epoch(date) - sec_epoch) < 5, 'date header'

    assert headers == {
        'Connection': 'close',
        'Content-Length': str(len(body)),
        'Content-Type': 'text/html',
        'Request-Method': 'POST',
        'Request-Uri': '/',
        'Http-Host': 'localhost',
        'Script-Name': '',
        'Server-Protocol': 'HTTP/1.1',
        'Custom-Header': 'blah',
        'Rack-Version': '13',
        'Rack-Url-Scheme': 'http',
        'Rack-Multithread': 'false',
        'Rack-Multiprocess': 'true',
        'Rack-Run-Once': 'false',
        'Rack-Hijack-Q': 'false',
        'Rack-Hijack': '',
        'Rack-Hijack-IO': '',
    }, 'headers'
    assert resp['body'] == body, 'body'


def test_ruby_application_query_string():
    client.load('query_string')

    resp = client.get(url='/?var1=val1&var2=val2')

    assert (
        resp['headers']['Query-String'] == 'var1=val1&var2=val2'
    ), 'Query-String header'


def test_ruby_application_query_string_empty():
    client.load('query_string')

    resp = client.get(url='/?')

    assert resp['status'] == 200, 'query string empty status'
    assert resp['headers']['Query-String'] == '', 'query string empty'


def test_ruby_application_query_string_absent():
    client.load('query_string')

    resp = client.get()

    assert resp['status'] == 200, 'query string absent status'
    assert resp['headers']['Query-String'] == '', 'query string absent'


@pytest.mark.skip('not yet')
def test_ruby_application_server_port():
    client.load('server_port')

    assert (
        client.get()['headers']['Server-Port'] == '7080'
    ), 'Server-Port header'


def test_ruby_application_status_int():
    client.load('status_int')

    assert client.get()['status'] == 200, 'status int'


def test_ruby_application_input_read_empty():
    client.load('input_read_empty')

    assert client.get()['body'] == '', 'read empty'


def test_ruby_application_input_read_parts():
    client.load('input_read_parts')

    assert (
        client.post(body='0123456789')['body'] == '012345678'
    ), 'input read parts'


def test_ruby_application_input_read_buffer():
    client.load('input_read_buffer')

    assert (
        client.post(body='0123456789')['body'] == '0123456789'
    ), 'input read buffer'


def test_ruby_application_input_read_buffer_not_empty():
    client.load('input_read_buffer_not_empty')

    assert (
        client.post(body='0123456789')['body'] == '0123456789'
    ), 'input read buffer not empty'


def test_ruby_application_input_gets():
    client.load('input_gets')

    body = '0123456789'

    assert client.post(body=body)['body'] == body, 'input gets'


def test_ruby_application_input_gets_2():
    client.load('input_gets')

    assert (
        client.post(body='01234\n56789\n')['body'] == '01234\n'
    ), 'input gets 2'


def test_ruby_application_input_gets_all():
    client.load('input_gets_all')

    body = '\n01234\n56789\n\n'

    assert client.post(body=body)['body'] == body, 'input gets all'


def test_ruby_application_input_each():
    client.load('input_each')

    body = '\n01234\n56789\n\n'

    assert client.post(body=body)['body'] == body, 'input each'


@pytest.mark.skip('not yet')
def test_ruby_application_input_rewind():
    client.load('input_rewind')

    body = '0123456789'

    assert client.post(body=body)['body'] == body, 'input rewind'


@pytest.mark.skip('not yet')
def test_ruby_application_syntax_error(skip_alert):
    skip_alert(
        r'Failed to parse rack script',
        r'syntax error',
        r'new_from_string',
        r'parse_file',
    )
    client.load('syntax_error')

    assert client.get()['status'] == 500, 'syntax error'


def test_ruby_application_errors_puts(wait_for_record):
    client.load('errors_puts')

    assert client.get()['status'] == 200

    assert (
        wait_for_record(r'\[error\].+Error in application') is not None
    ), 'errors puts'


def test_ruby_application_errors_puts_int(wait_for_record):
    client.load('errors_puts_int')

    assert client.get()['status'] == 200

    assert (
        wait_for_record(r'\[error\].+1234567890') is not None
    ), 'errors puts int'


def test_ruby_application_errors_write(wait_for_record):
    client.load('errors_write')

    assert client.get()['status'] == 200
    assert (
        wait_for_record(r'\[error\].+Error in application') is not None
    ), 'errors write'


def test_ruby_application_errors_write_to_s_custom():
    client.load('errors_write_to_s_custom')

    assert client.get()['status'] == 200, 'errors write to_s custom'


def test_ruby_application_errors_write_int(wait_for_record):
    client.load('errors_write_int')

    assert client.get()['status'] == 200
    assert (
        wait_for_record(r'\[error\].+1234567890') is not None
    ), 'errors write int'


def test_ruby_application_at_exit(wait_for_record):
    client.load('at_exit')

    assert client.get()['status'] == 200

    assert 'success' in client.conf({"listeners": {}, "applications": {}})

    assert (
        wait_for_record(r'\[error\].+At exit called\.') is not None
    ), 'at exit'


def test_ruby_application_encoding():
    client.load('encoding')

    try:
        locales = (
            subprocess.check_output(
                ['locale', '-a'],
                stderr=subprocess.STDOUT,
            )
            .decode()
            .split('\n')
        )

    except (FileNotFoundError, subprocess.CalledProcessError):
        pytest.skip('require locale')

    def get_locale(pattern):
        return next(
            (l for l in locales if re.match(pattern, l.upper()) is not None),
            None,
        )

    utf8 = get_locale(r'.*UTF[-_]?8')
    iso88591 = get_locale(r'.*ISO[-_]?8859[-_]?1')

    def check_locale(enc):
        assert 'success' in client.conf(
            {"LC_CTYPE": enc, "LC_ALL": ""},
            '/config/applications/encoding/environment',
        )

        resp = client.get()
        assert resp['status'] == 200, 'status'

        enc_default = re.sub(r'[-_]', '', resp['headers']['X-Enc']).upper()
        assert enc_default == re.sub(r'[-_]', '', enc.split('.')[-1]).upper()

    if utf8:
        check_locale(utf8)

    if iso88591:
        check_locale(iso88591)

    if not utf8 and not iso88591:
        pytest.skip('no available locales')


def test_ruby_application_header_custom():
    client.load('header_custom')

    resp = client.post(body="\ntc=one,two\ntc=three,four,\n\n")

    assert resp['headers']['Custom-Header'] == [
        '',
        'tc=one,two',
        'tc=three,four,',
        '',
        '',
    ], 'header custom'


@pytest.mark.skip('not yet')
def test_ruby_application_header_custom_non_printable():
    client.load('header_custom')

    assert (
        client.post(body='\b')['status'] == 500
    ), 'header custom non printable'


def test_ruby_application_header_status():
    client.load('header_status')

    assert client.get()['status'] == 200, 'header status'


@pytest.mark.skip('not yet')
def test_ruby_application_header_rack():
    client.load('header_rack')

    assert client.get()['status'] == 500, 'header rack'


def test_ruby_application_body_empty():
    client.load('body_empty')

    assert client.get()['body'] == '', 'body empty'


def test_ruby_application_body_array():
    client.load('body_array')

    assert client.get()['body'] == '0123456789', 'body array'


def test_ruby_application_body_large():
    client.load('mirror')

    body = '0123456789' * 1000

    assert client.post(body=body)['body'] == body, 'body large'


@pytest.mark.skip('not yet')
def test_ruby_application_body_each_error(wait_for_record):
    client.load('body_each_error')

    assert client.get()['status'] == 500, 'body each error status'

    assert (
        wait_for_record(r'\[error\].+Failed to run ruby script') is not None
    ), 'body each error'


def test_ruby_application_body_file():
    client.load('body_file')

    assert client.get()['body'] == 'body\n', 'body file'


def test_ruby_keepalive_body():
    client.load('mirror')

    assert client.get()['status'] == 200, 'init'

    body = '0123456789' * 500
    (resp, sock) = client.post(
        headers={
            'Host': 'localhost',
            'Connection': 'keep-alive',
        },
        start=True,
        body=body,
        read_timeout=1,
    )

    assert resp['body'] == body, 'keep-alive 1'

    body = '0123456789'
    resp = client.post(sock=sock, body=body)

    assert resp['body'] == body, 'keep-alive 2'


def test_ruby_application_constants():
    client.load('constants')

    resp = client.get()

    assert resp['status'] == 200, 'status'

    headers = resp['headers']
    assert len(headers['X-Copyright']) > 0, 'RUBY_COPYRIGHT'
    assert len(headers['X-Description']) > 0, 'RUBY_DESCRIPTION'
    assert len(headers['X-Engine']) > 0, 'RUBY_ENGINE'
    assert len(headers['X-Engine-Version']) > 0, 'RUBY_ENGINE_VERSION'
    assert len(headers['X-Patchlevel']) > 0, 'RUBY_PATCHLEVEL'
    assert len(headers['X-Platform']) > 0, 'RUBY_PLATFORM'
    assert len(headers['X-Release-Date']) > 0, 'RUBY_RELEASE_DATE'
    assert len(headers['X-Revision']) > 0, 'RUBY_REVISION'
    assert len(headers['X-Version']) > 0, 'RUBY_VERSION'


def test_ruby_application_threads():
    client.load('threads')

    assert 'success' in client.conf(
        '4', 'applications/threads/threads'
    ), 'configure 4 threads'

    socks = []

    for _ in range(4):
        sock = client.get(
            headers={
                'Host': 'localhost',
                'X-Delay': '2',
                'Connection': 'close',
            },
            no_recv=True,
        )

        socks.append(sock)

    threads = set()

    for sock in socks:
        resp = client.recvall(sock).decode('utf-8')

        client.log_in(resp)

        resp = client._resp_to_dict(resp)

        assert resp['status'] == 200, 'status'

        threads.add(resp['headers']['X-Thread'])

        assert resp['headers']['Rack-Multithread'] == 'true', 'multithread'

        sock.close()

    assert len(socks) == len(threads), 'threads differs'