File: server.py

package info (click to toggle)
nxt-python 3.5.1-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 812 kB
  • sloc: python: 6,857; xml: 22; makefile: 20; sh: 4
file content (89 lines) | stat: -rw-r--r-- 2,898 bytes parent folder | download | duplicates (2)
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
# nxt.command.server module -- Serve an interface to the NXT brick
# Copyright (C) 2011  zonedabone, Marcus Wanner
# Copyright (C) 2021  Nicolas Schodet
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
"""Network server for the NXT brick."""

import argparse
import logging
import socket
import traceback

import nxt.locator


def get_parser() -> argparse.ArgumentParser:
    """Return argument parser."""
    p = argparse.ArgumentParser(description=__doc__)
    p.add_argument("-p", "--port", type=int, default=2727, help="bind port")
    nxt.locator.add_arguments(p)
    levels = ("DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL")
    p.add_argument("--log-level", type=str.upper, choices=levels, help="set log level")
    return p


def serve(
    brick: nxt.brick.Brick, channel: socket.socket, details: tuple[str, int]
) -> None:
    """Handles serving the client."""
    print(f"Connection from {details[0]}.")
    run = True
    try:
        while run:
            data = channel.recv(1024)
            if not data:
                break
            code = data[0]
            if code == 0x00 or code == 0x01 or code == 0x02:
                brick._sock.send(data)
                reply = brick._sock.recv()
                channel.send(reply)
            elif code == 0x80 or code == 0x81:
                brick._sock.send(data)
            elif code == 0x98:
                channel.send(brick._sock.type.encode("ascii"))
            elif code == 0x99:
                run = False
            else:
                raise RuntimeError("Bad protocol")
    except Exception:
        traceback.print_exc()
    finally:
        channel.close()
        print("Connection closed.")


def run() -> None:
    """Run command."""
    options = get_parser().parse_args()

    if options.log_level:
        logging.basicConfig(level=options.log_level)

    print("Finding brick...")
    with nxt.locator.find_with_options(options) as brick:
        print(f"Brick found, starting server on port {options.port}.")
        print("Use Ctrl-C to interrupt.")
        server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
        server.bind(("", options.port))
        server.listen(1)
        try:
            # Have the server serve "forever":
            while True:
                channel, details = server.accept()
                serve(brick, channel, details)
        except KeyboardInterrupt:
            pass


if __name__ == "__main__":
    run()