public inbox for passt-dev@passt.top
 help / color / mirror / code / Atom feed
blob 5399bb54bfcfb4565733d1558ef30cbe6bb77911 5670 bytes (raw)

  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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
 
#! /usr/bin/env avocado-runner-avocado-classless

# SPDX-License-Identifier: GPL-2.0-or-later
#
# Copyright Red Hat
# Author: David Gibson <david@gibson.dropbear.id.au>

"""
Test A Simple Socket Transport

transfer.py - Helpers for testing data transfers
"""

import contextlib
import ipaddress
import time

from avocado_classless.test import assert_eq, test_output

from tasst.address import LOOPBACK4, LOOPBACK6
from tasst.nstool import unshare_site


# HACK: how long to wait for the server to be ready and listening (s)
SERVER_READY_DELAY = 0.05  # 1/20th of a second


# socat needs IPv6 addresses in square brackets
def socat_fmt(ip):
    if isinstance(ip, ipaddress.IPv6Address):
        return f'[{ip}]'
    if isinstance(ip, ipaddress.IPv4Address):
        return f'{ip}'
    raise TypeError


def socat_upload(datafile, cs, ss, connect, listen):
    cs.require_cmds('socat', 'cat')
    ss.require_cmds('socat')

    with ss.bg(f'socat -u {listen} STDOUT', verbose=False) as server:
        time.sleep(SERVER_READY_DELAY)
        cs.fg(f'socat -u OPEN:{datafile} {connect}')
        res = server.run()
    srcdata = cs.output(f'cat {datafile}', verbose=False)
    assert_eq(srcdata, res.stdout)


def socat_download(datafile, cs, ss, connect, listen):
    cs.require_cmds('socat')
    ss.require_cmds('socat', 'cat')

    with ss.bg(f'socat -u OPEN:{datafile} {listen}'):
        time.sleep(SERVER_READY_DELAY)
        dstdata = cs.output(f'socat -u {connect} STDOUT', verbose=False)
    srcdata = ss.output(f'cat {datafile}', verbose=False)
    assert_eq(srcdata, dstdata)


# pylint: disable=R0913
def _tcp_socat(connectip, connectport, listenip, listenport, fromip):
    v6 = isinstance(connectip, ipaddress.IPv6Address)
    if listenport is None:
        listenport = connectport
    if v6:
        connect = f'TCP6:[{connectip}]:{connectport},ipv6only'
        listen = f'TCP6-LISTEN:{listenport},ipv6only'
    else:
        connect = f'TCP4:{connectip}:{connectport}'
        listen = f'TCP4-LISTEN:{listenport}'
    if listenip is not None:
        listen += f',bind={socat_fmt(listenip)}'
    if fromip is not None:
        connect += f',bind={socat_fmt(fromip)}'
    return (connect, listen)


def tcp_upload(datafile, cs, ss, connectip, connectport,
               listenip=None, listenport=None, fromip=None):
    connect, listen = _tcp_socat(connectip, connectport, listenip, listenport,
                                 fromip)
    socat_upload(datafile, cs, ss, connect, listen)


def tcp_download(datafile, cs, ss, connectip, connectport,
                 listenip=None, listenport=None, fromip=None):
    connect, listen = _tcp_socat(connectip, connectport, listenip, listenport,
                                 fromip)
    socat_download(datafile, cs, ss, connect, listen)


def udp_transfer(datafile, cs, ss, connectip, connectport,
                 listenip=None, listenport=None, fromip=None):
    v6 = isinstance(connectip, ipaddress.IPv6Address)
    if listenport is None:
        listenport = connectport
    if v6:
        connect = f'UDP6:[{connectip}]:{connectport},ipv6only,shut-null'
        listen = f'UDP6-LISTEN:{listenport},ipv6only,null-eof'
    else:
        connect = f'UDP4:{connectip}:{connectport},shut-null'
        listen = f'UDP4-LISTEN:{listenport},null-eof'
    if listenip is not None:
        listen += f',bind={socat_fmt(listenip)}'
    if fromip is not None:
        connect += f',bind={socat_fmt(fromip)}'

    socat_upload(datafile, cs, ss, connect, listen)


def test_tcp_uploads(datafile, ip4, ip6, port,
                     listenip4=None, listenip6=None, listenport=None,
                     fromip4=None, fromip6=None):
    if listenport is None:
        listenport = port

    def dec(sitefn):
        def tcp4_upload(sites):
            with sites as (cs, ss):
                tcp_upload(datafile, cs, ss, ip4, port,
                           listenip=listenip4, listenport=listenport,
                           fromip=fromip4)

        def tcp6_upload(sites):
            with sites as (cs, ss):
                tcp_upload(datafile, cs, ss, ip6, port,
                           listenip=listenip6, listenport=listenport,
                           fromip=fromip6)

        return test_output(tcp4_upload, tcp6_upload)(sitefn)

    return dec


def test_udp_transfers(datafile, ip4, ip6, port,
                       listenip4=None, listenip6=None, listenport=None,
                       fromip4=None, fromip6=None):
    if listenport is None:
        listenport = port

    def dec(sitefn):
        def udp4_transfer(sites):
            with sites as (cs, ss):
                udp_transfer(datafile, cs, ss, ip4, port,
                             listenip=listenip4, listenport=listenport,
                             fromip=fromip4)

        def udp6_transfer(sites):
            with sites as (cs, ss):
                udp_transfer(datafile, cs, ss, ip6, port,
                             listenip=listenip6, listenport=listenport,
                             fromip=fromip6)
        return test_output(udp4_transfer, udp6_transfer)(sitefn)

    return dec


def test_transfers(ip4, ip6, port, **kwargs):
    def dec(sitefn):
        sitefn = test_tcp_uploads('small.bin', ip4, ip6, port,
                                  **kwargs)(sitefn)
        sitefn = test_udp_transfers('medium.bin', ip4, ip6, port,
                                    **kwargs)(sitefn)
        return sitefn

    return dec


@test_transfers(ip4=LOOPBACK4, ip6=LOOPBACK6, port=10000)
@contextlib.contextmanager
def local_only():
    with unshare_site('ns', '-Un') as ns:
        ns.ifup('lo')
        yield (ns, ns)

debug log:

solving 5399bb54 ...
found 5399bb54 in https://archives.passt.top/passt-dev/20230627025429.2209702-23-david@gibson.dropbear.id.au/
found 788c1d52 in https://archives.passt.top/passt-dev/20230627025429.2209702-22-david@gibson.dropbear.id.au/

applying [1/2] https://archives.passt.top/passt-dev/20230627025429.2209702-22-david@gibson.dropbear.id.au/
diff --git a/test/tasst/transfer.py b/test/tasst/transfer.py
new file mode 100644
index 00000000..788c1d52


applying [2/2] https://archives.passt.top/passt-dev/20230627025429.2209702-23-david@gibson.dropbear.id.au/
diff --git a/test/tasst/transfer.py b/test/tasst/transfer.py
index 788c1d52..5399bb54 100644

Checking patch test/tasst/transfer.py...
Applied patch test/tasst/transfer.py cleanly.
Checking patch test/tasst/transfer.py...
Applied patch test/tasst/transfer.py cleanly.

index at:
100644 5399bb54bfcfb4565733d1558ef30cbe6bb77911	test/tasst/transfer.py

Code repositories for project(s) associated with this public inbox

	https://passt.top/passt

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for IMAP folder(s).