xref: /aosp_15_r20/external/curl/tests/http/test_30_vsftpd.py (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker#!/usr/bin/env python3
2*6236dae4SAndroid Build Coastguard Worker# -*- coding: utf-8 -*-
3*6236dae4SAndroid Build Coastguard Worker#***************************************************************************
4*6236dae4SAndroid Build Coastguard Worker#                                  _   _ ____  _
5*6236dae4SAndroid Build Coastguard Worker#  Project                     ___| | | |  _ \| |
6*6236dae4SAndroid Build Coastguard Worker#                             / __| | | | |_) | |
7*6236dae4SAndroid Build Coastguard Worker#                            | (__| |_| |  _ <| |___
8*6236dae4SAndroid Build Coastguard Worker#                             \___|\___/|_| \_\_____|
9*6236dae4SAndroid Build Coastguard Worker#
10*6236dae4SAndroid Build Coastguard Worker# Copyright (C) Daniel Stenberg, <[email protected]>, et al.
11*6236dae4SAndroid Build Coastguard Worker#
12*6236dae4SAndroid Build Coastguard Worker# This software is licensed as described in the file COPYING, which
13*6236dae4SAndroid Build Coastguard Worker# you should have received as part of this distribution. The terms
14*6236dae4SAndroid Build Coastguard Worker# are also available at https://curl.se/docs/copyright.html.
15*6236dae4SAndroid Build Coastguard Worker#
16*6236dae4SAndroid Build Coastguard Worker# You may opt to use, copy, modify, merge, publish, distribute and/or sell
17*6236dae4SAndroid Build Coastguard Worker# copies of the Software, and permit persons to whom the Software is
18*6236dae4SAndroid Build Coastguard Worker# furnished to do so, under the terms of the COPYING file.
19*6236dae4SAndroid Build Coastguard Worker#
20*6236dae4SAndroid Build Coastguard Worker# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
21*6236dae4SAndroid Build Coastguard Worker# KIND, either express or implied.
22*6236dae4SAndroid Build Coastguard Worker#
23*6236dae4SAndroid Build Coastguard Worker# SPDX-License-Identifier: curl
24*6236dae4SAndroid Build Coastguard Worker#
25*6236dae4SAndroid Build Coastguard Worker###########################################################################
26*6236dae4SAndroid Build Coastguard Worker#
27*6236dae4SAndroid Build Coastguard Workerimport difflib
28*6236dae4SAndroid Build Coastguard Workerimport filecmp
29*6236dae4SAndroid Build Coastguard Workerimport logging
30*6236dae4SAndroid Build Coastguard Workerimport os
31*6236dae4SAndroid Build Coastguard Workerimport shutil
32*6236dae4SAndroid Build Coastguard Workerimport pytest
33*6236dae4SAndroid Build Coastguard Worker
34*6236dae4SAndroid Build Coastguard Workerfrom testenv import Env, CurlClient, VsFTPD
35*6236dae4SAndroid Build Coastguard Worker
36*6236dae4SAndroid Build Coastguard Worker
37*6236dae4SAndroid Build Coastguard Workerlog = logging.getLogger(__name__)
38*6236dae4SAndroid Build Coastguard Worker
39*6236dae4SAndroid Build Coastguard Worker
40*6236dae4SAndroid Build Coastguard Worker@pytest.mark.skipif(condition=not Env.has_vsftpd(), reason="missing vsftpd")
41*6236dae4SAndroid Build Coastguard Workerclass TestVsFTPD:
42*6236dae4SAndroid Build Coastguard Worker
43*6236dae4SAndroid Build Coastguard Worker    @pytest.fixture(autouse=True, scope='class')
44*6236dae4SAndroid Build Coastguard Worker    def vsftpd(self, env):
45*6236dae4SAndroid Build Coastguard Worker        vsftpd = VsFTPD(env=env)
46*6236dae4SAndroid Build Coastguard Worker        assert vsftpd.start()
47*6236dae4SAndroid Build Coastguard Worker        yield vsftpd
48*6236dae4SAndroid Build Coastguard Worker        vsftpd.stop()
49*6236dae4SAndroid Build Coastguard Worker
50*6236dae4SAndroid Build Coastguard Worker    def _make_docs_file(self, docs_dir: str, fname: str, fsize: int):
51*6236dae4SAndroid Build Coastguard Worker        fpath = os.path.join(docs_dir, fname)
52*6236dae4SAndroid Build Coastguard Worker        data1k = 1024*'x'
53*6236dae4SAndroid Build Coastguard Worker        flen = 0
54*6236dae4SAndroid Build Coastguard Worker        with open(fpath, 'w') as fd:
55*6236dae4SAndroid Build Coastguard Worker            while flen < fsize:
56*6236dae4SAndroid Build Coastguard Worker                fd.write(data1k)
57*6236dae4SAndroid Build Coastguard Worker                flen += len(data1k)
58*6236dae4SAndroid Build Coastguard Worker        return flen
59*6236dae4SAndroid Build Coastguard Worker
60*6236dae4SAndroid Build Coastguard Worker    @pytest.fixture(autouse=True, scope='class')
61*6236dae4SAndroid Build Coastguard Worker    def _class_scope(self, env, vsftpd):
62*6236dae4SAndroid Build Coastguard Worker        if os.path.exists(vsftpd.docs_dir):
63*6236dae4SAndroid Build Coastguard Worker            shutil.rmtree(vsftpd.docs_dir)
64*6236dae4SAndroid Build Coastguard Worker        if not os.path.exists(vsftpd.docs_dir):
65*6236dae4SAndroid Build Coastguard Worker            os.makedirs(vsftpd.docs_dir)
66*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpd.docs_dir, fname='data-1k', fsize=1024)
67*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpd.docs_dir, fname='data-10k', fsize=10*1024)
68*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpd.docs_dir, fname='data-1m', fsize=1024*1024)
69*6236dae4SAndroid Build Coastguard Worker        self._make_docs_file(docs_dir=vsftpd.docs_dir, fname='data-10m', fsize=10*1024*1024)
70*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname="upload-1k", fsize=1024)
71*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname="upload-100k", fsize=100*1024)
72*6236dae4SAndroid Build Coastguard Worker        env.make_data_file(indir=env.gen_dir, fname="upload-1m", fsize=1024*1024)
73*6236dae4SAndroid Build Coastguard Worker
74*6236dae4SAndroid Build Coastguard Worker    def test_30_01_list_dir(self, env: Env, vsftpd: VsFTPD, repeat):
75*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
76*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/'
77*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_get(urls=[url], with_stats=True)
78*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=1, http_status=226)
79*6236dae4SAndroid Build Coastguard Worker        lines = open(os.path.join(curl.run_dir, 'download_#1.data')).readlines()
80*6236dae4SAndroid Build Coastguard Worker        assert len(lines) == 4, f'list: {lines}'
81*6236dae4SAndroid Build Coastguard Worker
82*6236dae4SAndroid Build Coastguard Worker    # download 1 file, no SSL
83*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
84*6236dae4SAndroid Build Coastguard Worker        'data-1k', 'data-1m', 'data-10m'
85*6236dae4SAndroid Build Coastguard Worker    ])
86*6236dae4SAndroid Build Coastguard Worker    def test_30_02_download_1(self, env: Env, vsftpd: VsFTPD, docname, repeat):
87*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
88*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpd.docs_dir, f'{docname}')
89*6236dae4SAndroid Build Coastguard Worker        count = 1
90*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/{docname}?[0-{count-1}]'
91*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_get(urls=[url], with_stats=True)
92*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
93*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
94*6236dae4SAndroid Build Coastguard Worker
95*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
96*6236dae4SAndroid Build Coastguard Worker        'data-1k', 'data-1m', 'data-10m'
97*6236dae4SAndroid Build Coastguard Worker    ])
98*6236dae4SAndroid Build Coastguard Worker    def test_30_03_download_10_serial(self, env: Env, vsftpd: VsFTPD, docname, repeat):
99*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
100*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpd.docs_dir, f'{docname}')
101*6236dae4SAndroid Build Coastguard Worker        count = 10
102*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/{docname}?[0-{count-1}]'
103*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_get(urls=[url], with_stats=True)
104*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
105*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
106*6236dae4SAndroid Build Coastguard Worker
107*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
108*6236dae4SAndroid Build Coastguard Worker        'data-1k', 'data-1m', 'data-10m'
109*6236dae4SAndroid Build Coastguard Worker    ])
110*6236dae4SAndroid Build Coastguard Worker    def test_30_04_download_10_parallel(self, env: Env, vsftpd: VsFTPD, docname, repeat):
111*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
112*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpd.docs_dir, f'{docname}')
113*6236dae4SAndroid Build Coastguard Worker        count = 10
114*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/{docname}?[0-{count-1}]'
115*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_get(urls=[url], with_stats=True, extra_args=[
116*6236dae4SAndroid Build Coastguard Worker            '--parallel'
117*6236dae4SAndroid Build Coastguard Worker        ])
118*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
119*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
120*6236dae4SAndroid Build Coastguard Worker
121*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.parametrize("docname", [
122*6236dae4SAndroid Build Coastguard Worker        'upload-1k', 'upload-100k', 'upload-1m'
123*6236dae4SAndroid Build Coastguard Worker    ])
124*6236dae4SAndroid Build Coastguard Worker    def test_30_05_upload_1(self, env: Env, vsftpd: VsFTPD, docname, repeat):
125*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
126*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
127*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpd.docs_dir, docname)
128*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
129*6236dae4SAndroid Build Coastguard Worker        count = 1
130*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/'
131*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_upload(urls=[url], fupload=f'{srcfile}', with_stats=True)
132*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
133*6236dae4SAndroid Build Coastguard Worker        self.check_upload(env, vsftpd, docname=docname)
134*6236dae4SAndroid Build Coastguard Worker
135*6236dae4SAndroid Build Coastguard Worker    def _rmf(self, path):
136*6236dae4SAndroid Build Coastguard Worker        if os.path.exists(path):
137*6236dae4SAndroid Build Coastguard Worker            return os.remove(path)
138*6236dae4SAndroid Build Coastguard Worker
139*6236dae4SAndroid Build Coastguard Worker    # check with `tcpdump` if curl causes any TCP RST packets
140*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.skipif(condition=not Env.tcpdump(), reason="tcpdump not available")
141*6236dae4SAndroid Build Coastguard Worker    def test_30_06_shutdownh_download(self, env: Env, vsftpd: VsFTPD, repeat):
142*6236dae4SAndroid Build Coastguard Worker        docname = 'data-1k'
143*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
144*6236dae4SAndroid Build Coastguard Worker        count = 1
145*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/{docname}?[0-{count-1}]'
146*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_get(urls=[url], with_stats=True, with_tcpdump=True)
147*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
148*6236dae4SAndroid Build Coastguard Worker        assert r.tcpdump
149*6236dae4SAndroid Build Coastguard Worker        assert len(r.tcpdump.stats) == 0, 'Unexpected TCP RSTs packets'
150*6236dae4SAndroid Build Coastguard Worker
151*6236dae4SAndroid Build Coastguard Worker    # check with `tcpdump` if curl causes any TCP RST packets
152*6236dae4SAndroid Build Coastguard Worker    @pytest.mark.skipif(condition=not Env.tcpdump(), reason="tcpdump not available")
153*6236dae4SAndroid Build Coastguard Worker    def test_30_07_shutdownh_upload(self, env: Env, vsftpd: VsFTPD, repeat):
154*6236dae4SAndroid Build Coastguard Worker        docname = 'upload-1k'
155*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
156*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
157*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpd.docs_dir, docname)
158*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
159*6236dae4SAndroid Build Coastguard Worker        count = 1
160*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/'
161*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_upload(urls=[url], fupload=f'{srcfile}', with_stats=True, with_tcpdump=True)
162*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
163*6236dae4SAndroid Build Coastguard Worker        assert r.tcpdump
164*6236dae4SAndroid Build Coastguard Worker        assert len(r.tcpdump.stats) == 0, 'Unexpected TCP RSTs packets'
165*6236dae4SAndroid Build Coastguard Worker
166*6236dae4SAndroid Build Coastguard Worker    def test_30_08_active_download(self, env: Env, vsftpd: VsFTPD):
167*6236dae4SAndroid Build Coastguard Worker        docname = 'data-10k'
168*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
169*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(vsftpd.docs_dir, f'{docname}')
170*6236dae4SAndroid Build Coastguard Worker        count = 1
171*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/{docname}?[0-{count-1}]'
172*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_get(urls=[url], with_stats=True, extra_args=[
173*6236dae4SAndroid Build Coastguard Worker            '--ftp-port', '127.0.0.1'
174*6236dae4SAndroid Build Coastguard Worker        ])
175*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
176*6236dae4SAndroid Build Coastguard Worker        self.check_downloads(curl, srcfile, count)
177*6236dae4SAndroid Build Coastguard Worker
178*6236dae4SAndroid Build Coastguard Worker    def test_30_09_active_upload(self, env: Env, vsftpd: VsFTPD):
179*6236dae4SAndroid Build Coastguard Worker        docname = 'upload-1k'
180*6236dae4SAndroid Build Coastguard Worker        curl = CurlClient(env=env)
181*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
182*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpd.docs_dir, docname)
183*6236dae4SAndroid Build Coastguard Worker        self._rmf(dstfile)
184*6236dae4SAndroid Build Coastguard Worker        count = 1
185*6236dae4SAndroid Build Coastguard Worker        url = f'ftp://{env.ftp_domain}:{vsftpd.port}/'
186*6236dae4SAndroid Build Coastguard Worker        r = curl.ftp_upload(urls=[url], fupload=f'{srcfile}', with_stats=True, extra_args=[
187*6236dae4SAndroid Build Coastguard Worker            '--ftp-port', '127.0.0.1'
188*6236dae4SAndroid Build Coastguard Worker        ])
189*6236dae4SAndroid Build Coastguard Worker        r.check_stats(count=count, http_status=226)
190*6236dae4SAndroid Build Coastguard Worker        self.check_upload(env, vsftpd, docname=docname)
191*6236dae4SAndroid Build Coastguard Worker
192*6236dae4SAndroid Build Coastguard Worker    def check_downloads(self, client, srcfile: str, count: int,
193*6236dae4SAndroid Build Coastguard Worker                        complete: bool = True):
194*6236dae4SAndroid Build Coastguard Worker        for i in range(count):
195*6236dae4SAndroid Build Coastguard Worker            dfile = client.download_file(i)
196*6236dae4SAndroid Build Coastguard Worker            assert os.path.exists(dfile)
197*6236dae4SAndroid Build Coastguard Worker            if complete and not filecmp.cmp(srcfile, dfile, shallow=False):
198*6236dae4SAndroid Build Coastguard Worker                diff = "".join(difflib.unified_diff(a=open(srcfile).readlines(),
199*6236dae4SAndroid Build Coastguard Worker                                                    b=open(dfile).readlines(),
200*6236dae4SAndroid Build Coastguard Worker                                                    fromfile=srcfile,
201*6236dae4SAndroid Build Coastguard Worker                                                    tofile=dfile,
202*6236dae4SAndroid Build Coastguard Worker                                                    n=1))
203*6236dae4SAndroid Build Coastguard Worker                assert False, f'download {dfile} differs:\n{diff}'
204*6236dae4SAndroid Build Coastguard Worker
205*6236dae4SAndroid Build Coastguard Worker    def check_upload(self, env, vsftpd: VsFTPD, docname):
206*6236dae4SAndroid Build Coastguard Worker        srcfile = os.path.join(env.gen_dir, docname)
207*6236dae4SAndroid Build Coastguard Worker        dstfile = os.path.join(vsftpd.docs_dir, docname)
208*6236dae4SAndroid Build Coastguard Worker        assert os.path.exists(srcfile)
209*6236dae4SAndroid Build Coastguard Worker        assert os.path.exists(dstfile)
210*6236dae4SAndroid Build Coastguard Worker        if not filecmp.cmp(srcfile, dstfile, shallow=False):
211*6236dae4SAndroid Build Coastguard Worker            diff = "".join(difflib.unified_diff(a=open(srcfile).readlines(),
212*6236dae4SAndroid Build Coastguard Worker                                                b=open(dstfile).readlines(),
213*6236dae4SAndroid Build Coastguard Worker                                                fromfile=srcfile,
214*6236dae4SAndroid Build Coastguard Worker                                                tofile=dstfile,
215*6236dae4SAndroid Build Coastguard Worker                                                n=1))
216*6236dae4SAndroid Build Coastguard Worker            assert False, f'upload {dstfile} differs:\n{diff}'
217