summaryrefslogtreecommitdiff
path: root/examples/server/tests/features/environment.py
blob: 82104e9202e5e05e227e6f111025b0bf49077979 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
import os
import signal
import socket
import sys
import time
import traceback
from contextlib import closing

import psutil


def before_scenario(context, scenario):
    context.debug = 'DEBUG' in os.environ and os.environ['DEBUG'] == 'ON'
    if context.debug:
        print("DEBUG=ON\n")
    print(f"\x1b[33;42mStarting new scenario: {scenario.name}!\x1b[0m\n")
    port = 8080
    if 'PORT' in os.environ:
        port = int(os.environ['PORT'])
    if is_server_listening("localhost", port):
        assert False, "Server already started"


def after_scenario(context, scenario):
    try:
        if 'server_process' not in context or context.server_process is None:
            return
        if scenario.status == "failed":
            if 'GITHUB_ACTIONS' in os.environ:
                print(f"\x1b[33;101mSCENARIO FAILED: {scenario.name} server logs:\x1b[0m\n\n")
                if os.path.isfile('llama.log'):
                    with closing(open('llama.log', 'r')) as f:
                        for line in f:
                            print(line)
            if not is_server_listening(context.server_fqdn, context.server_port):
                print("\x1b[33;101mERROR: Server stopped listening\x1b[0m\n")

        if not pid_exists(context.server_process.pid):
            assert False, f"Server not running pid={context.server_process.pid} ..."

        server_graceful_shutdown(context)

        # Wait few for socket to free up
        time.sleep(0.05)

        attempts = 0
        while pid_exists(context.server_process.pid) or is_server_listening(context.server_fqdn, context.server_port):
            server_kill(context)
            time.sleep(0.1)
            attempts += 1
            if attempts > 5:
                server_kill_hard(context)
    except:
        exc = sys.exception()
        print("error in after scenario: \n")
        print(exc)
        print("*** print_tb: \n")
        traceback.print_tb(exc.__traceback__, file=sys.stdout)


def server_graceful_shutdown(context):
    print(f"shutting down server pid={context.server_process.pid} ...\n")
    if os.name == 'nt':
        os.kill(context.server_process.pid, signal.CTRL_C_EVENT)
    else:
        os.kill(context.server_process.pid, signal.SIGINT)


def server_kill(context):
    print(f"killing server pid={context.server_process.pid} ...\n")
    context.server_process.kill()


def server_kill_hard(context):
    pid = context.server_process.pid
    path = context.server_path

    print(f"Server dangling exits, hard killing force {pid}={path}...\n")
    try:
        psutil.Process(pid).kill()
    except psutil.NoSuchProcess:
        return False
    return True


def is_server_listening(server_fqdn, server_port):
    with closing(socket.socket(socket.AF_INET, socket.SOCK_STREAM)) as sock:
        result = sock.connect_ex((server_fqdn, server_port))
        _is_server_listening = result == 0
        if _is_server_listening:
            print(f"server is listening on {server_fqdn}:{server_port}...\n")
        return _is_server_listening


def pid_exists(pid):
    try:
        psutil.Process(pid)
    except psutil.NoSuchProcess:
        return False
    return True